diff options
Diffstat (limited to 'node_modules/tslint/lib/rules/noUnsafeAnyRule.js')
-rw-r--r-- | node_modules/tslint/lib/rules/noUnsafeAnyRule.js | 218 |
1 files changed, 218 insertions, 0 deletions
diff --git a/node_modules/tslint/lib/rules/noUnsafeAnyRule.js b/node_modules/tslint/lib/rules/noUnsafeAnyRule.js new file mode 100644 index 000000000..f07eff92b --- /dev/null +++ b/node_modules/tslint/lib/rules/noUnsafeAnyRule.js @@ -0,0 +1,218 @@ +"use strict"; +/** + * @license + * Copyright 2017 Palantir Technologies, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +Object.defineProperty(exports, "__esModule", { value: true }); +var tslib_1 = require("tslib"); +var ts = require("typescript"); +var Lint = require("../index"); +var Rule = (function (_super) { + tslib_1.__extends(Rule, _super); + function Rule() { + return _super !== null && _super.apply(this, arguments) || this; + } + Rule.prototype.applyWithProgram = function (sourceFile, program) { + return this.applyWithFunction(sourceFile, function (ctx) { return walk(ctx, program.getTypeChecker()); }); + }; + return Rule; +}(Lint.Rules.TypedRule)); +/* tslint:disable:object-literal-sort-keys */ +Rule.metadata = { + ruleName: "no-unsafe-any", + description: (_a = ["\n Warns when using an expression of type 'any' in a dynamic way.\n Uses are only allowed if they would work for `{} | null | undefined`.\n Type casts and tests are allowed.\n Expressions that work on all values (such as `\"\" + x`) are allowed."], _a.raw = ["\n Warns when using an expression of type 'any' in a dynamic way.\n Uses are only allowed if they would work for \\`{} | null | undefined\\`.\n Type casts and tests are allowed.\n Expressions that work on all values (such as \\`\"\" + x\\`) are allowed."], Lint.Utils.dedent(_a)), + optionsDescription: "Not configurable.", + options: null, + optionExamples: [true], + type: "functionality", + typescriptOnly: true, + requiresTypeInfo: true, +}; +/* tslint:enable:object-literal-sort-keys */ +Rule.FAILURE_STRING = "Unsafe use of expression of type 'any'."; +exports.Rule = Rule; +function walk(ctx, checker) { + if (ctx.sourceFile.isDeclarationFile) { + // Not possible in a declaration file. + return; + } + return ts.forEachChild(ctx.sourceFile, cb); + /** @param anyOk If true, this node will be allowed to be of type *any*. (But its children might not.) */ + function cb(node, anyOk) { + switch (node.kind) { + case ts.SyntaxKind.ParenthesizedExpression: + // Don't warn on a parenthesized expression, warn on its contents. + return cb(node.expression, anyOk); + case ts.SyntaxKind.Parameter: { + var _a = node, type = _a.type, initializer = _a.initializer; + if (initializer !== undefined) { + return cb(initializer, /*anyOk*/ type !== undefined && type.kind === ts.SyntaxKind.AnyKeyword); + } + return; + } + case ts.SyntaxKind.LabeledStatement: + // Ignore label + return cb(node.statement); + case ts.SyntaxKind.BreakStatement: // Ignore label + // Ignore types + case ts.SyntaxKind.InterfaceDeclaration: + case ts.SyntaxKind.TypeAliasDeclaration: + case ts.SyntaxKind.QualifiedName: + case ts.SyntaxKind.TypePredicate: + case ts.SyntaxKind.TypeOfExpression: + // Ignore imports + case ts.SyntaxKind.ImportEqualsDeclaration: + case ts.SyntaxKind.ImportDeclaration: + case ts.SyntaxKind.ExportDeclaration: + // These show as type "any" if in type position. + case ts.SyntaxKind.NumericLiteral: + case ts.SyntaxKind.StringLiteral: + return; + // Recurse through these, but ignore the immediate child because it is allowed to be 'any'. + case ts.SyntaxKind.DeleteExpression: + case ts.SyntaxKind.ExpressionStatement: + case ts.SyntaxKind.TypeAssertionExpression: + case ts.SyntaxKind.AsExpression: + case ts.SyntaxKind.TemplateSpan: // Allow stringification (works on all values). Note: tagged templates handled differently. + case ts.SyntaxKind.ThrowStatement: { + var expression = node.expression; + return cb(expression, /*anyOk*/ true); + } + case ts.SyntaxKind.PropertyAssignment: { + // Only check RHS. + var _b = node, name = _b.name, initializer = _b.initializer; + // The LHS will be 'any' if the RHS is, so just handle the RHS. + // Still need to check the LHS in case it is a computed key. + cb(name, /*anyOk*/ true); + cb(initializer); + return; + } + case ts.SyntaxKind.PropertyDeclaration: { + var _c = node, name = _c.name, initializer = _c.initializer; + if (initializer !== undefined) { + return cb(initializer, /*anyOk*/ isNodeAny(name, checker)); + } + return; + } + case ts.SyntaxKind.TaggedTemplateExpression: { + var _d = node, tag = _d.tag, template = _d.template; + cb(tag); + if (template.kind === ts.SyntaxKind.TemplateExpression) { + for (var _i = 0, _e = template.templateSpans; _i < _e.length; _i++) { + var expression = _e[_i].expression; + checkContextual(expression); + } + } + // Also check the template expression itself + check(); + return; + } + case ts.SyntaxKind.CallExpression: + case ts.SyntaxKind.NewExpression: { + var _f = node, expression = _f.expression, args = _f.arguments; + cb(expression); + if (args !== undefined) { + for (var _g = 0, args_1 = args; _g < args_1.length; _g++) { + var arg = args_1[_g]; + checkContextual(arg); + } + } + // Also check the call expression itself + check(); + return; + } + case ts.SyntaxKind.PropertyAccessExpression: + // Don't warn for right hand side; this is redundant if we warn for the access itself. + cb(node.expression); + check(); + return; + case ts.SyntaxKind.VariableDeclaration: + return checkVariableDeclaration(node); + case ts.SyntaxKind.BinaryExpression: + return checkBinaryExpression(node); + case ts.SyntaxKind.ReturnStatement: { + var expression = node.expression; + if (expression !== undefined) { + return checkContextual(expression); + } + return; + } + default: + if (!(ts.isExpression(node) && check())) { + return ts.forEachChild(node, cb); + } + return; + } + function check() { + var isUnsafe = anyOk !== true && isNodeAny(node, checker); + if (isUnsafe) { + ctx.addFailureAtNode(node, Rule.FAILURE_STRING); + } + return isUnsafe; + } + } + /** OK for this value to be 'any' if that's its contextual type. */ + function checkContextual(arg) { + return cb(arg, /*anyOk*/ isAny(checker.getContextualType(arg))); + } + // Allow `const x = foo;` and `const x: any = foo`, but not `const x: Foo = foo;`. + function checkVariableDeclaration(_a) { + var type = _a.type, initializer = _a.initializer; + // Always allow the LHS to be `any`. Just don't allow RHS to be `any` when LHS isn't. + // TODO: handle destructuring + if (initializer !== undefined) { + return cb(initializer, /*anyOk*/ type === undefined || type.kind === ts.SyntaxKind.AnyKeyword); + } + return; + } + function checkBinaryExpression(node) { + var _a = node, left = _a.left, right = _a.right, operatorToken = _a.operatorToken; + // Allow equality since all values support equality. + if (Lint.getEqualsKind(operatorToken) !== undefined) { + return; + } + switch (operatorToken.kind) { + case ts.SyntaxKind.InstanceOfKeyword: + return cb(right); + case ts.SyntaxKind.CommaToken: + cb(left, /*anyOk*/ true); + return cb(right, /*anyOk*/ true); + case ts.SyntaxKind.EqualsToken: + // Allow assignment if the lhs is also *any*. + // TODO: handle destructuring + cb(right, /*anyOk*/ isNodeAny(left, checker)); + return; + case ts.SyntaxKind.PlusToken: // Allow implicit stringification + case ts.SyntaxKind.PlusEqualsToken: + var anyOk = isStringLike(left, checker) + || (isStringLike(right, checker) && operatorToken.kind === ts.SyntaxKind.PlusToken); + cb(left, anyOk); + return cb(right, anyOk); + default: + cb(left); + return cb(right); + } + } +} +function isNodeAny(node, checker) { + return isAny(checker.getTypeAtLocation(node)); +} +function isStringLike(expr, checker) { + return Lint.isTypeFlagSet(checker.getTypeAtLocation(expr), ts.TypeFlags.StringLike); +} +function isAny(type) { + return type !== undefined && Lint.isTypeFlagSet(type, ts.TypeFlags.Any); +} +var _a; |