mirror of
https://github.com/microsoft/vscode.git
synced 2025-12-10 00:27:05 -06:00
Wth node 20.18, we can now run these typescript files directly instead of having to use ts-node
61 lines
2.2 KiB
TypeScript
61 lines
2.2 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';
|
|
import type * as ESTree from 'estree';
|
|
import { TSESTree } from '@typescript-eslint/utils';
|
|
|
|
/**
|
|
* 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 default 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: ESTree.BinaryExpression) {
|
|
const node = inNode as TSESTree.BinaryExpression;
|
|
// Check if we're inside a type predicate function
|
|
const ancestors = context.sourceCode.getAncestors(node as ESTree.Node);
|
|
|
|
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,
|
|
messageId: 'noInOperator'
|
|
});
|
|
}
|
|
|
|
return {
|
|
['BinaryExpression[operator="in"]']: checkInOperator,
|
|
};
|
|
}
|
|
};
|