@cdklabs/eslint-plugin 1.4.0 → 1.4.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -4,9 +4,10 @@ exports.defaultOptions = exports.meta = void 0;
|
|
|
4
4
|
exports.create = create;
|
|
5
5
|
const utils_1 = require("@typescript-eslint/utils");
|
|
6
6
|
const eslint_utils_1 = require("@typescript-eslint/utils/eslint-utils");
|
|
7
|
+
const typescript_1 = require("typescript");
|
|
7
8
|
exports.meta = {
|
|
8
9
|
messages: {
|
|
9
|
-
avoidAccess: "{{ memberAccess }}: this will evaluate
|
|
10
|
+
avoidAccess: "{{ memberAccess }}: this will evaluate a property, which might throw if it's a getter. Prefer using `'{{ prop }}' in {{ obj }}` (don't forget to check for object-ness of {{ obj }} if necessary!)",
|
|
10
11
|
},
|
|
11
12
|
};
|
|
12
13
|
exports.defaultOptions = {};
|
|
@@ -17,6 +18,8 @@ function create(context) {
|
|
|
17
18
|
/**
|
|
18
19
|
* Whether this function is a type predicate
|
|
19
20
|
*
|
|
21
|
+
* This is a short and sweet way to determine whether this is a function we should be checking.
|
|
22
|
+
*
|
|
20
23
|
* ```
|
|
21
24
|
* function isSomething(x: unknown): x is Something { ... }
|
|
22
25
|
* ```
|
|
@@ -30,6 +33,9 @@ function create(context) {
|
|
|
30
33
|
/**
|
|
31
34
|
* Whether this looks like a type coercion function
|
|
32
35
|
*
|
|
36
|
+
* There are a lot more heuristics involved here to determine whether this is actually
|
|
37
|
+
* a function that does a run-time type check. It could look like this for a number of reasons.
|
|
38
|
+
*
|
|
33
39
|
* ```
|
|
34
40
|
* function toSomething(x: SomeType): SomeSubType { ... }
|
|
35
41
|
* ```
|
|
@@ -37,13 +43,37 @@ function create(context) {
|
|
|
37
43
|
function isTypeCoercionFunction(node) {
|
|
38
44
|
const typeChecker = services.program.getTypeChecker();
|
|
39
45
|
const functionType = services.getTypeAtLocation(node);
|
|
40
|
-
const
|
|
41
|
-
|
|
46
|
+
const callSignature = functionType.getCallSignatures()[0];
|
|
47
|
+
if (!callSignature) {
|
|
48
|
+
// Constructors, getters and setters don't have a call signature
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
const firstParameter = callSignature.getParameters()[0];
|
|
52
|
+
if (!firstParameter) {
|
|
53
|
+
// Couldn't resolve first parameter for some reason. This can happen if the argument is 'this'
|
|
54
|
+
return false;
|
|
55
|
+
}
|
|
56
|
+
const argumentType = typeChecker.getTypeOfSymbol(firstParameter);
|
|
57
|
+
const returnType = callSignature.getReturnType();
|
|
58
|
+
// If the output type is void this is not a type coercion
|
|
59
|
+
// eslint-disable-next-line no-bitwise
|
|
60
|
+
if ((returnType.getFlags() & typescript_1.TypeFlags.VoidLike) !== 0) {
|
|
61
|
+
return false;
|
|
62
|
+
}
|
|
63
|
+
// If the output type doesn't have a symbol, it's an anonymous type (`return { x: 3, y: 42 };`)
|
|
64
|
+
// Which is most likely not a coercion function.
|
|
65
|
+
if (returnType.getSymbol() == null || returnType.getSymbol()?.escapedName === '__object') {
|
|
66
|
+
return false;
|
|
67
|
+
}
|
|
68
|
+
// Object methods are never type testing functions (static class methods might be)
|
|
69
|
+
if (node.parent.type === utils_1.AST_NODE_TYPES.MethodDefinition && !node.parent.static) {
|
|
70
|
+
return false;
|
|
71
|
+
}
|
|
42
72
|
// If the return type is assignable to the input type, then this looks like
|
|
43
|
-
// a downcast which probably means it's a type
|
|
44
|
-
return typeChecker.isTypeAssignableTo(returnType, argumentType)
|
|
73
|
+
// a downcast which probably means it's a type coercion function.
|
|
74
|
+
return (typeChecker.isTypeAssignableTo(returnType, argumentType)
|
|
75
|
+
&& !typeChecker.isTypeAssignableTo(argumentType, returnType));
|
|
45
76
|
}
|
|
46
|
-
// eslint-disable-next-line @stylistic/max-len
|
|
47
77
|
function maybeInteresting(node) {
|
|
48
78
|
if (node.params.length !== 1 || node.params[0].type !== utils_1.AST_NODE_TYPES.Identifier) {
|
|
49
79
|
return undefined;
|
|
@@ -103,7 +133,7 @@ function create(context) {
|
|
|
103
133
|
&& node.parent.parent.operator === '!') {
|
|
104
134
|
dangerous = true;
|
|
105
135
|
}
|
|
106
|
-
if (node.parent.type === utils_1.AST_NODE_TYPES.LogicalExpression) {
|
|
136
|
+
if (node.parent.type === utils_1.AST_NODE_TYPES.LogicalExpression && ['&&', '||'].includes(node.parent.operator)) {
|
|
107
137
|
dangerous = true;
|
|
108
138
|
}
|
|
109
139
|
if ([utils_1.AST_NODE_TYPES.ReturnStatement, utils_1.AST_NODE_TYPES.IfStatement, utils_1.AST_NODE_TYPES.ConditionalExpression].includes(node.parent.type)) {
|
|
@@ -131,4 +161,4 @@ function create(context) {
|
|
|
131
161
|
'MemberExpression': onlyInInterestingFunction(memberExpression),
|
|
132
162
|
};
|
|
133
163
|
}
|
|
134
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"no-evaluating-typeguard.js","sourceRoot":"","sources":["../../src/rules/no-evaluating-typeguard.ts"],"names":[],"mappings":"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","sourcesContent":["import { AST_NODE_TYPES, TSESTree } from '@typescript-eslint/utils';\n\nimport { getParserServices } from '@typescript-eslint/utils/eslint-utils';\nimport { Rule } from 'eslint';\nimport type { MemberExpression } from 'estree';\nimport NodeParentExtension = Rule.NodeParentExtension;\n\nexport const meta: Rule.RuleMetaData = {\n  messages: {\n    avoidAccess: \"{{ memberAccess }}: this will evaluate '{{ prop }}', which might throw if it's a getter. Prefer using `'{{ prop }}' in {{ obj }}` (don't forget to check for object-ness of {{ obj }} if necessary!)\",\n  },\n};\n\nexport const defaultOptions = {};\n\n\ninterface InterestingFunction {\n  readonly argumentName: string;\n}\n\nexport function create(context: Rule.RuleContext): Rule.RuleListener {\n  const services = getParserServices(context as any);\n\n  const functionStack: Array<InterestingFunction | undefined> = [];\n  //  const classDeclarations: Map<string, ClassDeclaration> = new Map();\n\n  /**\n   * Whether this function is a type predicate\n   *\n   * ```\n   * function isSomething(x: unknown): x is Something { ... }\n   * ```\n   */\n  function isUserDefinedTypeGuard(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) {\n    if (node.params.length !== 1 || node.params[0].type !== AST_NODE_TYPES.Identifier) { return false; }\n    return node.returnType?.typeAnnotation.type === AST_NODE_TYPES.TSTypePredicate;\n  }\n\n  /**\n   * Whether this looks like a type coercion function\n   *\n   * ```\n   * function toSomething(x: SomeType): SomeSubType { ... }\n   * ```\n   */\n  function isTypeCoercionFunction(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) {\n    const typeChecker = services.program.getTypeChecker();\n\n    const functionType = services.getTypeAtLocation(node);\n    const argumentType = typeChecker.getTypeOfSymbol(functionType.getCallSignatures()[0].getParameters()[0]);\n    const returnType = functionType.getCallSignatures()[0].getReturnType();\n\n    // If the return type is assignable to the input type, then this looks like\n    // a downcast which probably means it's a type coerction function.\n    return typeChecker.isTypeAssignableTo(returnType, argumentType);\n  }\n\n  // eslint-disable-next-line @stylistic/max-len\n  function maybeInteresting(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression): InterestingFunction | undefined {\n    if (node.params.length !== 1 || node.params[0].type !== AST_NODE_TYPES.Identifier) { return undefined; }\n    if (isUserDefinedTypeGuard(node) || isTypeCoercionFunction(node)) {\n      return {\n        argumentName: (node.params[0] as TSESTree.Identifier).name,\n      };\n    }\n    return undefined;\n  }\n\n  /**\n   * Enter a function and see if it's interesting\n   */\n  function enterFunction(node: (TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) & NodeParentExtension) {\n    functionStack.push(maybeInteresting(node));\n  }\n\n  function exitFunction() {\n    functionStack.pop();\n  }\n\n  function currentInteresting(): InterestingFunction | undefined {\n    return functionStack.length > 0 ? functionStack[functionStack.length - 1] : undefined;\n  }\n\n  function onlyInInterestingFunction<A extends Function>(fn: A): A {\n    return ((...args: any[]) => {\n      if (currentInteresting()) {\n        return fn(...args);\n      }\n      return undefined;\n    }) as any;\n  }\n\n  /**\n   * For member expressions in interesting functions\n   *\n   * - If their LHS refers to the only function argument; AND\n   *   - If their parent is a `typeof` node; OR\n   *   - The member expression is part of a boolean expression; OR\n   *   - The member expression is in a bang-bang operator (!!); OR\n   *   - The expression is by itself in an `if` or `return` or `?:` ternary\n   *\n   * Then that is a potential problem because it will evaluate a member\n   * that we potentially only wanted to test for presence.\n   */\n  function memberExpression(node: TSESTree.MemberExpression) {\n    const current = currentInteresting()!;\n\n    // Only property access on the primary argument\n    if ((node.object.type !== AST_NODE_TYPES.Identifier) || (node.object as TSESTree.Identifier).name !== current.argumentName) {\n      return;\n    }\n\n    let dangerous = false;\n    if (node.parent.type === AST_NODE_TYPES.UnaryExpression && node.parent.operator === 'typeof') {\n      dangerous = true;\n    }\n    if (node.parent.type === AST_NODE_TYPES.UnaryExpression\n      && node.parent.operator === '!'\n      && node.parent.parent.type === AST_NODE_TYPES.UnaryExpression\n      && node.parent.parent.operator === '!') {\n      dangerous = true;\n    }\n    if (node.parent.type === AST_NODE_TYPES.LogicalExpression) {\n      dangerous = true;\n    }\n    if ([AST_NODE_TYPES.ReturnStatement, AST_NODE_TYPES.IfStatement, AST_NODE_TYPES.ConditionalExpression].includes(node.parent.type)) {\n      dangerous = true;\n    }\n\n    if (dangerous) {\n      const eslintNode = node as MemberExpression;\n      context.report({\n        node: eslintNode,\n        messageId: 'avoidAccess',\n        loc: eslintNode.loc ?? { line: 0, column: 0 },\n        data: {\n          memberAccess: context.sourceCode.getText(eslintNode),\n          obj: current.argumentName,\n          prop: (node.property as TSESTree.PrivateIdentifier | TSESTree.Identifier).name,\n        },\n      });\n    }\n  }\n\n  return {\n    'FunctionDeclaration': enterFunction as any,\n    'FunctionExpression': enterFunction as any,\n    'FunctionDeclaration:exit': exitFunction,\n    'FunctionExpression:exit': exitFunction,\n    'MemberExpression': onlyInInterestingFunction(memberExpression) as any,\n  };\n}\n"]}
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"no-evaluating-typeguard.js","sourceRoot":"","sources":["../../src/rules/no-evaluating-typeguard.ts"],"names":[],"mappings":"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","sourcesContent":["import { AST_NODE_TYPES, TSESTree } from '@typescript-eslint/utils';\nimport { getParserServices } from '@typescript-eslint/utils/eslint-utils';\nimport { Rule } from 'eslint';\nimport type { MemberExpression } from 'estree';\nimport { TypeFlags } from 'typescript';\nimport NodeParentExtension = Rule.NodeParentExtension;\n\nexport const meta: Rule.RuleMetaData = {\n  messages: {\n    avoidAccess: \"{{ memberAccess }}: this will evaluate a property, which might throw if it's a getter. Prefer using `'{{ prop }}' in {{ obj }}` (don't forget to check for object-ness of {{ obj }} if necessary!)\",\n  },\n};\n\nexport const defaultOptions = {};\n\n\ninterface InterestingFunction {\n  readonly argumentName: string;\n}\n\nexport function create(context: Rule.RuleContext): Rule.RuleListener {\n  const services = getParserServices(context as any);\n\n  const functionStack: Array<InterestingFunction | undefined> = [];\n  //  const classDeclarations: Map<string, ClassDeclaration> = new Map();\n\n  /**\n   * Whether this function is a type predicate\n   *\n   * This is a short and sweet way to determine whether this is a function we should be checking.\n   *\n   * ```\n   * function isSomething(x: unknown): x is Something { ... }\n   * ```\n   */\n  function isUserDefinedTypeGuard(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) {\n    if (node.params.length !== 1 || node.params[0].type !== AST_NODE_TYPES.Identifier) { return false; }\n    return node.returnType?.typeAnnotation.type === AST_NODE_TYPES.TSTypePredicate;\n  }\n\n  /**\n   * Whether this looks like a type coercion function\n   *\n   * There are a lot more heuristics involved here to determine whether this is actually\n   * a function that does a run-time type check. It could look like this for a number of reasons.\n   *\n   * ```\n   * function toSomething(x: SomeType): SomeSubType { ... }\n   * ```\n   */\n  function isTypeCoercionFunction(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) {\n    const typeChecker = services.program.getTypeChecker();\n\n    const functionType = services.getTypeAtLocation(node);\n    const callSignature = functionType.getCallSignatures()[0];\n    if (!callSignature) {\n      // Constructors, getters and setters don't have a call signature\n      return false;\n    }\n\n    const firstParameter = callSignature.getParameters()[0];\n    if (!firstParameter) {\n      // Couldn't resolve first parameter for some reason. This can happen if the argument is 'this'\n      return false;\n    }\n\n    const argumentType = typeChecker.getTypeOfSymbol(firstParameter);\n    const returnType = callSignature.getReturnType();\n\n    // If the output type is void this is not a type coercion\n    // eslint-disable-next-line no-bitwise\n    if ((returnType.getFlags() & TypeFlags.VoidLike) !== 0) {\n      return false;\n    }\n\n    // If the output type doesn't have a symbol, it's an anonymous type (`return { x: 3, y: 42 };`)\n    // Which is most likely not a coercion function.\n    if (returnType.getSymbol() == null || returnType.getSymbol()?.escapedName === '__object') {\n      return false;\n    }\n\n    // Object methods are never type testing functions (static class methods might be)\n    if (node.parent.type === AST_NODE_TYPES.MethodDefinition && !node.parent.static) {\n      return false;\n    }\n\n    // If the return type is assignable to the input type, then this looks like\n    // a downcast which probably means it's a type coercion function.\n    return (typeChecker.isTypeAssignableTo(returnType, argumentType)\n      && !typeChecker.isTypeAssignableTo(argumentType, returnType));\n  }\n\n  function maybeInteresting(node: TSESTree.FunctionDeclaration | TSESTree.FunctionExpression): InterestingFunction | undefined {\n    if (node.params.length !== 1 || node.params[0].type !== AST_NODE_TYPES.Identifier) { return undefined; }\n    if (isUserDefinedTypeGuard(node) || isTypeCoercionFunction(node)) {\n      return {\n        argumentName: (node.params[0] as TSESTree.Identifier).name,\n      };\n    }\n    return undefined;\n  }\n\n  /**\n   * Enter a function and see if it's interesting\n   */\n  function enterFunction(node: (TSESTree.FunctionDeclaration | TSESTree.FunctionExpression) & NodeParentExtension) {\n    functionStack.push(maybeInteresting(node));\n  }\n\n  function exitFunction() {\n    functionStack.pop();\n  }\n\n  function currentInteresting(): InterestingFunction | undefined {\n    return functionStack.length > 0 ? functionStack[functionStack.length - 1] : undefined;\n  }\n\n  function onlyInInterestingFunction<A extends Function>(fn: A): A {\n    return ((...args: any[]) => {\n      if (currentInteresting()) {\n        return fn(...args);\n      }\n      return undefined;\n    }) as any;\n  }\n\n  /**\n   * For member expressions in interesting functions\n   *\n   * - If their LHS refers to the only function argument; AND\n   *   - If their parent is a `typeof` node; OR\n   *   - The member expression is part of a boolean expression; OR\n   *   - The member expression is in a bang-bang operator (!!); OR\n   *   - The expression is by itself in an `if` or `return` or `?:` ternary\n   *\n   * Then that is a potential problem because it will evaluate a member\n   * that we potentially only wanted to test for presence.\n   */\n  function memberExpression(node: TSESTree.MemberExpression) {\n    const current = currentInteresting()!;\n\n    // Only property access on the primary argument\n    if ((node.object.type !== AST_NODE_TYPES.Identifier) || (node.object as TSESTree.Identifier).name !== current.argumentName) {\n      return;\n    }\n\n    let dangerous = false;\n    if (node.parent.type === AST_NODE_TYPES.UnaryExpression && node.parent.operator === 'typeof') {\n      dangerous = true;\n    }\n    if (node.parent.type === AST_NODE_TYPES.UnaryExpression\n      && node.parent.operator === '!'\n      && node.parent.parent.type === AST_NODE_TYPES.UnaryExpression\n      && node.parent.parent.operator === '!') {\n      dangerous = true;\n    }\n    if (node.parent.type === AST_NODE_TYPES.LogicalExpression && ['&&', '||'].includes(node.parent.operator)) {\n      dangerous = true;\n    }\n    if ([AST_NODE_TYPES.ReturnStatement, AST_NODE_TYPES.IfStatement, AST_NODE_TYPES.ConditionalExpression].includes(node.parent.type)) {\n      dangerous = true;\n    }\n\n    if (dangerous) {\n      const eslintNode = node as MemberExpression;\n      context.report({\n        node: eslintNode,\n        messageId: 'avoidAccess',\n        loc: eslintNode.loc ?? { line: 0, column: 0 },\n        data: {\n          memberAccess: context.sourceCode.getText(eslintNode),\n          obj: current.argumentName,\n          prop: (node.property as TSESTree.PrivateIdentifier | TSESTree.Identifier).name,\n        },\n      });\n    }\n  }\n\n  return {\n    'FunctionDeclaration': enterFunction as any,\n    'FunctionExpression': enterFunction as any,\n    'FunctionDeclaration:exit': exitFunction,\n    'FunctionExpression:exit': exitFunction,\n    'MemberExpression': onlyInInterestingFunction(memberExpression) as any,\n  };\n}\n"]}
|
package/package.json
CHANGED
|
@@ -62,7 +62,8 @@
|
|
|
62
62
|
},
|
|
63
63
|
"dependencies": {
|
|
64
64
|
"@typescript-eslint/utils": "^8.52.0",
|
|
65
|
-
"fs-extra": "^11.3.3"
|
|
65
|
+
"fs-extra": "^11.3.3",
|
|
66
|
+
"typescript": "^5.9.3"
|
|
66
67
|
},
|
|
67
68
|
"engines": {
|
|
68
69
|
"node": ">= 18.12.0"
|
|
@@ -72,7 +73,7 @@
|
|
|
72
73
|
"publishConfig": {
|
|
73
74
|
"access": "public"
|
|
74
75
|
},
|
|
75
|
-
"version": "1.4.
|
|
76
|
+
"version": "1.4.2",
|
|
76
77
|
"jest": {
|
|
77
78
|
"coverageProvider": "v8",
|
|
78
79
|
"testMatch": [
|