UNPKG

simplr-tslint

Version:

A set of TSLint rules used in SimplrJS projects.

98 lines 15.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); const ts = require("typescript"); const Lint = require("tslint"); const changeCase = require("change-case"); const BACKING_FIELD_PREFIX = "_"; class Rule extends Lint.Rules.AbstractRule { static accessorFailureMessageFactory(expectedName) { return `Accessor name must be the same as a backing field. Expected name "${expectedName}".`; } apply(sourceFile) { return this.applyWithWalker(new BackingFieldsWalker(sourceFile, this.getOptions())); } } Rule.usageFailureMessage = "Backing field can only be used in getter and setter."; exports.Rule = Rule; class BackingFieldsWalker extends Lint.RuleWalker { checkPropertyPrefix(name) { return name === BACKING_FIELD_PREFIX + changeCase.camelCase(name); } removePrefix(name) { return name.substring(BACKING_FIELD_PREFIX.length, name.length); } accessorNameEquals(accessorName, backingFieldName) { return BACKING_FIELD_PREFIX + accessorName === backingFieldName; } isMemberOfClassDeclaration(classDeclaration, name) { for (const member of classDeclaration.members) { // Property if (ts.isPropertyDeclaration(member) && member.modifiers != null && member.modifiers.findIndex(x => x.kind === ts.SyntaxKind.PrivateKeyword) !== -1 && member.name.getText() === name) { return true; } // Constructor Parameter Property. if (ts.isConstructorDeclaration(member)) { for (const parameter of member.parameters) { if (parameter.modifiers != null && parameter.modifiers.findIndex(x => x.kind === ts.SyntaxKind.PrivateKeyword) !== -1 && parameter.name.getText() === name) { return true; } } } } return false; } visitSourceFile(node) { // This rule should only work in source files. if (!node.isDeclarationFile) { super.visitSourceFile(node); } } visitPropertyAccessExpression(node) { super.visitPropertyAccessExpression(node); // "this._something" if (node.expression.kind !== ts.SyntaxKind.ThisKeyword) { return; } const name = node.name.getText(); if (!this.checkPropertyPrefix(name)) { return; } let currentParentNode = node.parent; let classDeclaration; let accessor; while (currentParentNode != null) { if (ts.isGetAccessorDeclaration(currentParentNode) || ts.isSetAccessorDeclaration(currentParentNode)) { accessor = currentParentNode; } if (ts.isClassDeclaration(currentParentNode)) { classDeclaration = currentParentNode; break; } currentParentNode = currentParentNode.parent; } if (classDeclaration == null || !this.isMemberOfClassDeclaration(classDeclaration, name)) { return; } if (accessor != null) { const accessorNameNode = accessor.name; const casedAccessorName = changeCase.camelCase(accessorNameNode.getText()); if (!this.accessorNameEquals(casedAccessorName, name)) { const expectedAccessorName = this.removePrefix(name); const accessorNameNodeStart = accessorNameNode.getStart(); const accessorNameNodeWidth = accessorNameNode.getWidth(); const fix = new Lint.Replacement(accessorNameNodeStart, accessorNameNodeWidth, expectedAccessorName); this.addFailureAt(accessorNameNodeStart, accessorNameNodeWidth, Rule.accessorFailureMessageFactory(expectedAccessorName), fix); } } else { // Backing field can only be used in GetAccessor and SetAccessor declarations. this.addFailureAtNode(node, Rule.usageFailureMessage); } } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backingFieldRule.js","sourceRoot":"","sources":["../src/backingFieldRule.ts"],"names":[],"mappings":"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","sourcesContent":["import * as ts from \"typescript\";\r\nimport * as Lint from \"tslint\";\r\nimport * as changeCase from \"change-case\";\r\n\r\nconst BACKING_FIELD_PREFIX = \"_\";\r\n\r\nexport class Rule extends Lint.Rules.AbstractRule {\r\n    public static readonly usageFailureMessage: string = \"Backing field can only be used in getter and setter.\";\r\n    public static accessorFailureMessageFactory(expectedName: string): string {\r\n        return `Accessor name must be the same as a backing field. Expected name \"${expectedName}\".`;\r\n    }\r\n\r\n    public apply(sourceFile: ts.SourceFile): Lint.RuleFailure[] {\r\n        return this.applyWithWalker(new BackingFieldsWalker(sourceFile, this.getOptions()));\r\n    }\r\n}\r\n\r\nclass BackingFieldsWalker extends Lint.RuleWalker {\r\n    private checkPropertyPrefix(name: string): boolean {\r\n        return name === BACKING_FIELD_PREFIX + changeCase.camelCase(name);\r\n    }\r\n\r\n    private removePrefix(name: string): string {\r\n        return name.substring(BACKING_FIELD_PREFIX.length, name.length);\r\n    }\r\n\r\n    private accessorNameEquals(accessorName: string, backingFieldName: string): boolean {\r\n        return BACKING_FIELD_PREFIX + accessorName === backingFieldName;\r\n    }\r\n\r\n    private isMemberOfClassDeclaration(classDeclaration: ts.ClassDeclaration, name: string): boolean {\r\n        for (const member of classDeclaration.members) {\r\n            // Property\r\n            if (\r\n                ts.isPropertyDeclaration(member) &&\r\n                member.modifiers != null &&\r\n                member.modifiers.findIndex(x => x.kind === ts.SyntaxKind.PrivateKeyword) !== -1 &&\r\n                member.name.getText() === name\r\n            ) {\r\n                return true;\r\n            }\r\n\r\n            // Constructor Parameter Property.\r\n            if (ts.isConstructorDeclaration(member)) {\r\n                for (const parameter of member.parameters) {\r\n                    if (\r\n                        parameter.modifiers != null &&\r\n                        parameter.modifiers.findIndex(x => x.kind === ts.SyntaxKind.PrivateKeyword) !== -1 &&\r\n                        parameter.name.getText() === name\r\n                    ) {\r\n                        return true;\r\n                    }\r\n                }\r\n            }\r\n        }\r\n\r\n        return false;\r\n    }\r\n\r\n    public visitSourceFile(node: ts.SourceFile): void {\r\n        // This rule should only work in source files.\r\n        if (!node.isDeclarationFile) {\r\n            super.visitSourceFile(node);\r\n        }\r\n    }\r\n\r\n    public visitPropertyAccessExpression(node: ts.PropertyAccessExpression): void {\r\n        super.visitPropertyAccessExpression(node);\r\n        // \"this._something\"\r\n        if (node.expression.kind !== ts.SyntaxKind.ThisKeyword) {\r\n            return;\r\n        }\r\n\r\n        const name = node.name.getText();\r\n        if (!this.checkPropertyPrefix(name)) {\r\n            return;\r\n        }\r\n\r\n        let currentParentNode: ts.Node | undefined = node.parent;\r\n        let classDeclaration: ts.ClassDeclaration | undefined;\r\n        let accessor: ts.GetAccessorDeclaration | ts.SetAccessorDeclaration | undefined;\r\n        while (currentParentNode != null) {\r\n            if (ts.isGetAccessorDeclaration(currentParentNode) || ts.isSetAccessorDeclaration(currentParentNode)) {\r\n                accessor = currentParentNode;\r\n            }\r\n\r\n            if (ts.isClassDeclaration(currentParentNode)) {\r\n                classDeclaration = currentParentNode;\r\n                break;\r\n            }\r\n\r\n            currentParentNode = currentParentNode.parent;\r\n        }\r\n\r\n        if (classDeclaration == null || !this.isMemberOfClassDeclaration(classDeclaration, name)) {\r\n            return;\r\n        }\r\n\r\n        if (accessor != null) {\r\n            const accessorNameNode = accessor.name;\r\n            const casedAccessorName = changeCase.camelCase(accessorNameNode.getText());\r\n\r\n            if (!this.accessorNameEquals(casedAccessorName, name)) {\r\n                const expectedAccessorName = this.removePrefix(name);\r\n                const accessorNameNodeStart = accessorNameNode.getStart();\r\n                const accessorNameNodeWidth = accessorNameNode.getWidth();\r\n                const fix = new Lint.Replacement(accessorNameNodeStart, accessorNameNodeWidth, expectedAccessorName);\r\n\r\n                this.addFailureAt(\r\n                    accessorNameNodeStart,\r\n                    accessorNameNodeWidth,\r\n                    Rule.accessorFailureMessageFactory(expectedAccessorName),\r\n                    fix\r\n                );\r\n            }\r\n        } else {\r\n            // Backing field can only be used in GetAccessor and SetAccessor declarations.\r\n            this.addFailureAtNode(node, Rule.usageFailureMessage);\r\n        }\r\n    }\r\n}\r\n"]}