vscode/.eslint-plugin-local/code-no-in-operator.ts
Johannes Rieken 478b33a153
Add eslint rule for no in-operator (#273779)
* add no-in-operator rule, config, and ignore-list

* add `hasKey` util and tests

* updates

* sample adoption, no `in` in strings.ts
2025-10-28 17:51:49 +01:00

58 lines
2.0 KiB
TypeScript

/*---------------------------------------------------------------------------------------------
* Copyright (c) Microsoft Corporation. All rights reserved.
* Licensed under the MIT License. See License.txt in the project root for license information.
*--------------------------------------------------------------------------------------------*/
import * as eslint from 'eslint';
/**
* Disallows the use of the `in` operator in TypeScript code, except within
* type predicate functions (functions with `arg is Type` return types).
*
* The `in` operator can lead to runtime errors and type safety issues.
* Consider using Object.hasOwn(), hasOwnProperty(), or other safer patterns.
*
* Exception: Type predicate functions are allowed to use the `in` operator
* since they are the standard way to perform runtime type checking.
*/
export = new class NoInOperator implements eslint.Rule.RuleModule {
readonly meta: eslint.Rule.RuleMetaData = {
messages: {
noInOperator: 'The "in" operator should not be used. Use type discriminator properties and classes instead or the `hasKey`-utility.',
},
schema: false,
};
create(context: eslint.Rule.RuleContext): eslint.Rule.RuleListener {
function checkInOperator(inNode: any) {
// Check if we're inside a type predicate function
const ancestors = context.sourceCode.getAncestors(inNode);
for (const ancestor of ancestors) {
if (ancestor.type === 'FunctionDeclaration' ||
ancestor.type === 'FunctionExpression' ||
ancestor.type === 'ArrowFunctionExpression') {
// Check if this function has a type predicate return type
// Type predicates have the form: `arg is SomeType`
if ((ancestor as { returnType?: any }).returnType?.typeAnnotation?.type === 'TSTypePredicate') {
// This is a type predicate function, allow the "in" operator
return;
}
}
}
context.report({
node: inNode,
messageId: 'noInOperator'
});
}
return {
['BinaryExpression[operator="in"]']: checkInOperator,
};
}
};