diff --git a/packages/cspell-eslint-plugin/README.md b/packages/cspell-eslint-plugin/README.md index e0476a220182..fd6ed3a8638a 100644 --- a/packages/cspell-eslint-plugin/README.md +++ b/packages/cspell-eslint-plugin/README.md @@ -21,30 +21,36 @@ This plugin is still in active development. Due to the nature of how files are p ## Options -```ts +````ts interface Options { /** * Number of spelling suggestions to make. * @default 8 */ numSuggestions: number; - /** * Generate suggestions * @default true */ generateSuggestions: boolean; - - /** - * Output debug logs - * @default false - */ - debugMode?: boolean; /** * Ignore import and require names * @default true */ ignoreImports?: boolean; + /** + * Ignore the properties of imported variables, structures, and types. + * + * Example: + * ``` + * import { example } from 'third-party'; + * + * const msg = example.property; // `property` is not spell checked. + * ``` + * + * @default true + */ + ignoreImportProperties?: boolean; /** * Spell check identifiers (variables names, function names, and class names) * @default true @@ -65,8 +71,13 @@ interface Options { * @default true */ checkComments?: boolean; + /** + * Output debug logs + * @default false + */ + debugMode?: boolean; } -``` +```` Example: diff --git a/packages/cspell-eslint-plugin/fixtures/with-errors/creepyData.ts b/packages/cspell-eslint-plugin/fixtures/with-errors/creepyData.ts new file mode 100644 index 000000000000..fd1da2db515c --- /dev/null +++ b/packages/cspell-eslint-plugin/fixtures/with-errors/creepyData.ts @@ -0,0 +1,21 @@ +export interface CreepyExpressions { + muawhahaha: string; + grrrrr: string; + uuuug: string; +} + +export const expressions: CreepyExpressions = { + muawhahaha: 'muawhahaha', + grrrrr: 'grrrrr', + uuuug: 'uuuug', +}; + +export const muawhahaha = expressions.muawhahaha; +export const uuug = expressions.uuuug; +export const grrr = expressions.grrrrr; + +export enum ExpressionCategory { + MUAWHAHAHA = 0, + GRRRRRR, + UUUUUG, +} diff --git a/packages/cspell-eslint-plugin/fixtures/with-errors/importAlias.ts b/packages/cspell-eslint-plugin/fixtures/with-errors/importAlias.ts new file mode 100644 index 000000000000..3f20efaa0dbe --- /dev/null +++ b/packages/cspell-eslint-plugin/fixtures/with-errors/importAlias.ts @@ -0,0 +1,5 @@ +import { uuug as uuug, muawhahaha as evilLaugh, grrr } from './creepyData'; + +console.log(uuug); +console.log(evilLaugh); +console.log(grrr); diff --git a/packages/cspell-eslint-plugin/fixtures/with-errors/imports.ts b/packages/cspell-eslint-plugin/fixtures/with-errors/imports.ts new file mode 100644 index 000000000000..7e6ec0876471 --- /dev/null +++ b/packages/cspell-eslint-plugin/fixtures/with-errors/imports.ts @@ -0,0 +1,7 @@ +import { muawhahaha, expressions } from './creepyData'; +import * as creepy from './creepyData'; + +console.log(creepy.expressions.grrrrr); +console.log(creepy.muawhahaha); +console.log(muawhahaha); +console.log(expressions.uuuug); diff --git a/packages/cspell-eslint-plugin/src/_auto_generated_/options.schema.json b/packages/cspell-eslint-plugin/src/_auto_generated_/options.schema.json index 05644e321bd4..eab433afa72c 100644 --- a/packages/cspell-eslint-plugin/src/_auto_generated_/options.schema.json +++ b/packages/cspell-eslint-plugin/src/_auto_generated_/options.schema.json @@ -10,7 +10,7 @@ }, "checkIdentifiers": { "default": true, - "description": "Spell check identifiers (variables names, function names, and class names)", + "description": "Spell check identifiers (variables names, function names, class names, etc.)", "type": "boolean" }, "checkStringTemplates": { @@ -33,6 +33,11 @@ "description": "Generate suggestions", "type": "boolean" }, + "ignoreImportProperties": { + "default": true, + "description": "Ignore the properties of imported variables, structures, and types.\n\nExample: ``` import { example } from 'third-party';\n\nconst msg = example.property; // `property` is not spell checked. ```", + "type": "boolean" + }, "ignoreImports": { "default": true, "description": "Ignore import and require names", diff --git a/packages/cspell-eslint-plugin/src/cspell-eslint-plugin.ts b/packages/cspell-eslint-plugin/src/cspell-eslint-plugin.ts index be67c4cf2786..08fe58935d02 100644 --- a/packages/cspell-eslint-plugin/src/cspell-eslint-plugin.ts +++ b/packages/cspell-eslint-plugin/src/cspell-eslint-plugin.ts @@ -4,7 +4,7 @@ import assert from 'assert'; import { createTextDocument, CSpellSettings, DocumentValidator, ValidationIssue } from 'cspell-lib'; import type { Rule } from 'eslint'; // eslint-disable-next-line node/no-missing-import -import type { Comment, Identifier, Literal, Node, TemplateElement } from 'estree'; +import type { Comment, Identifier, Literal, Node, TemplateElement, ImportSpecifier } from 'estree'; import { format } from 'util'; import { normalizeOptions } from './options'; import optionsSchema from './_auto_generated_/options.schema.json'; @@ -56,6 +56,7 @@ function log(...args: Parameters) { function create(context: Rule.RuleContext): Rule.RuleListener { const options = normalizeOptions(context.options[0]); + const toIgnore = new Set(); const importedIdentifiers = new Set(); isDebugMode = options.debugMode || false; isDebugMode && logContext(context); @@ -66,8 +67,9 @@ function create(context: Rule.RuleContext): Rule.RuleListener { function checkLiteral(node: Literal & Rule.NodeParentExtension) { if (!options.checkStrings) return; if (typeof node.value === 'string') { - if (options.ignoreImports && isImportOrRequired(node)) return; debugNode(node, node.value); + if (options.ignoreImports && isImportOrRequired(node)) return; + if (options.ignoreImportProperties && isImportedProperty(node)) return; checkNodeText(node, node.value); } } @@ -80,13 +82,25 @@ function create(context: Rule.RuleContext): Rule.RuleListener { } function checkIdentifier(node: Identifier & Rule.NodeParentExtension) { - if (options.ignoreImports && isImportIdentifier(node)) { - importedIdentifiers.add(node.name); - return; + debugNode(node, node.name); + if (options.ignoreImports) { + if (isRawImportIdentifier(node)) { + toIgnore.add(node.name); + return; + } + if (isImportIdentifier(node)) { + importedIdentifiers.add(node.name); + if (isLocalImportIdentifierUnique(node)) { + checkNodeText(node, node.name); + } + return; + } else if (options.ignoreImportProperties && isImportedProperty(node)) { + return; + } } if (!options.checkIdentifiers) return; - if (importedIdentifiers.has(node.name)) return; - debugNode(node, node.name); + if (toIgnore.has(node.name) && !isObjectProperty(node)) return; + if (skipCheckForRawImportIdentifiers(node)) return; checkNodeText(node, node.name); } @@ -113,6 +127,17 @@ function create(context: Rule.RuleContext): Rule.RuleListener { } function isImportIdentifier(node: ASTNode): boolean { + const parent = node.parent; + if (node.type !== 'Identifier' || !parent) return false; + return ( + (parent.type === 'ImportSpecifier' || + parent.type === 'ImportNamespaceSpecifier' || + parent.type === 'ImportDefaultSpecifier') && + parent.local === node + ); + } + + function isRawImportIdentifier(node: ASTNode): boolean { const parent = node.parent; if (node.type !== 'Identifier' || !parent) return false; return ( @@ -121,6 +146,34 @@ function create(context: Rule.RuleContext): Rule.RuleListener { ); } + function isLocalImportIdentifierUnique(node: ASTNode): boolean { + const parent = getImportParent(node); + if (!parent) return true; + const { imported, local } = parent; + if (imported.name !== local.name) return true; + return imported.range?.[0] !== local.range?.[0] && imported.range?.[1] !== local.range?.[1]; + } + + function getImportParent(node: ASTNode): ImportSpecifier | undefined { + const parent = node.parent; + return parent?.type === 'ImportSpecifier' ? parent : undefined; + } + + function skipCheckForRawImportIdentifiers(node: ASTNode): boolean { + if (options.ignoreImports) return false; + const parent = getImportParent(node); + return !!parent && parent.imported === node && !isLocalImportIdentifierUnique(node); + } + + function isImportedProperty(node: ASTNode): boolean { + const obj = findOriginObject(node); + return !!obj && obj.type === 'Identifier' && importedIdentifiers.has(obj.name); + } + + function isObjectProperty(node: ASTNode): boolean { + return node.parent?.type === 'MemberExpression'; + } + function reportIssue(issue: ValidationIssue) { const messageId: MessageIds = issue.isFlagged ? 'wordForbidden' : 'wordUnknown'; const data = { @@ -236,6 +289,19 @@ function create(context: Rule.RuleContext): Rule.RuleListener { return inheritance(node).join(' '); } + /** + * find the origin of a member expression + */ + function findOriginObject(node: ASTNode): ASTNode | undefined { + const parent = node.parent; + if (parent?.type !== 'MemberExpression' || parent.property !== node) return undefined; + let obj = parent.object; + while (obj.type === 'MemberExpression') { + obj = obj.object; + } + return obj; + } + function isFunctionCall(node: ASTNode | undefined, name: string): boolean { return node?.type === 'CallExpression' && node.callee.type === 'Identifier' && node.callee.name === name; } diff --git a/packages/cspell-eslint-plugin/src/index.test.ts b/packages/cspell-eslint-plugin/src/index.test.ts index 0c3ad47dc246..4afaebf23415 100644 --- a/packages/cspell-eslint-plugin/src/index.test.ts +++ b/packages/cspell-eslint-plugin/src/index.test.ts @@ -42,6 +42,7 @@ ruleTester.run('cspell', Rule.rules.spellchecker, { readSample('sample.ts'), readSample('sampleESM.mjs'), readFix('with-errors/strings.ts', { checkStrings: false, checkStringTemplates: false }), + readFix('with-errors/imports.ts'), ], invalid: [ // cspell:ignore Guuide Gallaxy BADD functionn coool @@ -86,6 +87,38 @@ ruleTester.run('cspell', Rule.rules.spellchecker, { readInvalid('with-errors/strings.ts', ['Unknown word: "naaame"', 'Unknown word: "doen\'t"'], { checkStringTemplates: false, }), + // cspell:ignore muawhahaha grrrrr uuuug + readInvalid( + 'with-errors/imports.ts', + [ + 'Unknown word: "muawhahaha"', + 'Unknown word: "grrrrr"', + 'Unknown word: "muawhahaha"', + 'Unknown word: "muawhahaha"', + 'Unknown word: "uuuug"', + ], + { ignoreImports: false } + ), + readInvalid( + 'with-errors/imports.ts', + ['Unknown word: "grrrrr"', 'Unknown word: "muawhahaha"', 'Unknown word: "uuuug"'], + { ignoreImportProperties: false } + ), + // cspell:ignore uuug grrr + readInvalid('with-errors/importAlias.ts', ['Unknown word: "uuug"']), + readInvalid('with-errors/importAlias.ts', ['Unknown word: "uuug"'], { ignoreImportProperties: false }), + readInvalid( + 'with-errors/importAlias.ts', + [ + 'Unknown word: "uuug"', + 'Unknown word: "uuug"', + 'Unknown word: "muawhahaha"', + 'Unknown word: "grrr"', + 'Unknown word: "uuug"', + 'Unknown word: "grrr"', + ], + { ignoreImports: false } + ), ], }); diff --git a/packages/cspell-eslint-plugin/src/options.ts b/packages/cspell-eslint-plugin/src/options.ts index 2c83da390478..2f0161a1ddf1 100644 --- a/packages/cspell-eslint-plugin/src/options.ts +++ b/packages/cspell-eslint-plugin/src/options.ts @@ -25,7 +25,20 @@ export interface Check { */ ignoreImports?: boolean; /** - * Spell check identifiers (variables names, function names, and class names) + * Ignore the properties of imported variables, structures, and types. + * + * Example: + * ``` + * import { example } from 'third-party'; + * + * const msg = example.property; // `property` is not spell checked. + * ``` + * + * @default true + */ + ignoreImportProperties?: boolean; + /** + * Spell check identifiers (variables names, function names, class names, etc.) * @default true */ checkIdentifiers?: boolean; @@ -46,22 +59,23 @@ export interface Check { checkComments?: boolean; } -export const defaultCheckOptions: Check = { +export const defaultCheckOptions: Required = { checkComments: true, checkIdentifiers: true, checkStrings: true, checkStringTemplates: true, ignoreImports: true, + ignoreImportProperties: true, }; -export const defaultOptions: Options = { +export const defaultOptions: Required = { ...defaultCheckOptions, numSuggestions: 8, generateSuggestions: true, debugMode: false, }; -export function normalizeOptions(opts: Options | undefined): Options { - const options: Options = Object.assign({}, defaultOptions, opts || {}); +export function normalizeOptions(opts: Options | undefined): Required { + const options: Required = Object.assign({}, defaultOptions, opts || {}); return options; } diff --git a/test-packages/test-cspell-eslint-plugin/.eslintrc.debug.js b/test-packages/test-cspell-eslint-plugin/.eslintrc.debug.js new file mode 100644 index 000000000000..8f23a1a29566 --- /dev/null +++ b/test-packages/test-cspell-eslint-plugin/.eslintrc.debug.js @@ -0,0 +1,8 @@ +/** + * @type { import("eslint").Linter.Config } + */ +const config = { + extends: ['./.eslintrc.js', 'plugin:@cspell/debug'], +}; + +module.exports = config; diff --git a/test-packages/test-cspell-eslint-plugin/.eslintrc.js b/test-packages/test-cspell-eslint-plugin/.eslintrc.js index 10398dfb527d..22b3b067ab76 100644 --- a/test-packages/test-cspell-eslint-plugin/.eslintrc.js +++ b/test-packages/test-cspell-eslint-plugin/.eslintrc.js @@ -13,8 +13,7 @@ const config = { 'plugin:import/warnings', 'plugin:promise/recommended', 'plugin:prettier/recommended', - // 'plugin:@cspell/recommended', - 'plugin:@cspell/debug', + 'plugin:@cspell/recommended', ], ignorePatterns: ['**/*.d.ts', '**/*.map', '**/coverage/**', '**/dist/**', '**/node_modules/**'], parserOptions: { diff --git a/test-packages/test-cspell-eslint-plugin/fixtures/creepyData.ts b/test-packages/test-cspell-eslint-plugin/fixtures/creepyData.ts new file mode 100644 index 000000000000..fd1da2db515c --- /dev/null +++ b/test-packages/test-cspell-eslint-plugin/fixtures/creepyData.ts @@ -0,0 +1,21 @@ +export interface CreepyExpressions { + muawhahaha: string; + grrrrr: string; + uuuug: string; +} + +export const expressions: CreepyExpressions = { + muawhahaha: 'muawhahaha', + grrrrr: 'grrrrr', + uuuug: 'uuuug', +}; + +export const muawhahaha = expressions.muawhahaha; +export const uuug = expressions.uuuug; +export const grrr = expressions.grrrrr; + +export enum ExpressionCategory { + MUAWHAHAHA = 0, + GRRRRRR, + UUUUUG, +} diff --git a/test-packages/test-cspell-eslint-plugin/src/file.js b/test-packages/test-cspell-eslint-plugin/fixtures/file.js similarity index 100% rename from test-packages/test-cspell-eslint-plugin/src/file.js rename to test-packages/test-cspell-eslint-plugin/fixtures/file.js diff --git a/test-packages/test-cspell-eslint-plugin/fixtures/importAlias.ts b/test-packages/test-cspell-eslint-plugin/fixtures/importAlias.ts new file mode 100644 index 000000000000..3f20efaa0dbe --- /dev/null +++ b/test-packages/test-cspell-eslint-plugin/fixtures/importAlias.ts @@ -0,0 +1,5 @@ +import { uuug as uuug, muawhahaha as evilLaugh, grrr } from './creepyData'; + +console.log(uuug); +console.log(evilLaugh); +console.log(grrr); diff --git a/test-packages/test-cspell-eslint-plugin/src/index.ts b/test-packages/test-cspell-eslint-plugin/fixtures/index.ts similarity index 61% rename from test-packages/test-cspell-eslint-plugin/src/index.ts rename to test-packages/test-cspell-eslint-plugin/fixtures/index.ts index f41d4ef2ec18..c204478808d3 100644 --- a/test-packages/test-cspell-eslint-plugin/src/index.ts +++ b/test-packages/test-cspell-eslint-plugin/fixtures/index.ts @@ -1,4 +1,6 @@ import { reader as myreader, reeder } from './reader'; +import { expressions, muawhahaha } from './creepyData'; +import * as creepy from './creepyData'; /** * This is some sample code to test cspell's eslint-plugin. @@ -16,6 +18,11 @@ function main() { console.log(myreader === reeder ? 'Match' : 'miss'); console.log(myreader(__filename)); + + console.log(expressions.grrrrr); + console.log(expressions['uuuug']); + console.log(muawhahaha); + console.log(creepy.expressions.grrrrr); } main(); diff --git a/test-packages/test-cspell-eslint-plugin/src/myReader.ts b/test-packages/test-cspell-eslint-plugin/fixtures/myReader.ts similarity index 100% rename from test-packages/test-cspell-eslint-plugin/src/myReader.ts rename to test-packages/test-cspell-eslint-plugin/fixtures/myReader.ts diff --git a/test-packages/test-cspell-eslint-plugin/src/reader.ts b/test-packages/test-cspell-eslint-plugin/fixtures/reader.ts similarity index 100% rename from test-packages/test-cspell-eslint-plugin/src/reader.ts rename to test-packages/test-cspell-eslint-plugin/fixtures/reader.ts diff --git a/test-packages/test-cspell-eslint-plugin/package.json b/test-packages/test-cspell-eslint-plugin/package.json index 17a3750770ec..d9a5cbcbccb6 100644 --- a/test-packages/test-cspell-eslint-plugin/package.json +++ b/test-packages/test-cspell-eslint-plugin/package.json @@ -4,10 +4,7 @@ "description": "Pure testing package for @cspell/eslint-plugin-cspell.", "private": true, "scripts": { - "clean": "rimraf dist .tsbuildinfo", - "build": "tsc -p .", - "eslint": "eslint -c .eslintrc.js", - "clean-build": "npm run clean && npm run build", + "eslint": "eslint -c .eslintrc.js .", "test": "npm run eslint" }, "engines": { diff --git a/test-packages/test-cspell-eslint-plugin/tsconfig.json b/test-packages/test-cspell-eslint-plugin/tsconfig.json index d11ecf39ed1f..447234decf1e 100644 --- a/test-packages/test-cspell-eslint-plugin/tsconfig.json +++ b/test-packages/test-cspell-eslint-plugin/tsconfig.json @@ -4,6 +4,6 @@ "outDir": "dist" }, "include": [ - "src" + "fixtures" ] }