-
Notifications
You must be signed in to change notification settings - Fork 609
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[eslint-plugin] Add new import-related rules to the ESLint plugin (#4889
) * Add new eslint rules * Consume new eslint rules in Rushstack * Update to newer eslint * Fix lint issues * Disable fixes when running in production mode * Rush change
- Loading branch information
Showing
39 changed files
with
1,219 additions
and
32 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
10 changes: 10 additions & 0 deletions
10
common/changes/@microsoft/api-extractor-model/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@microsoft/api-extractor-model", | ||
"comment": "", | ||
"type": "none" | ||
} | ||
], | ||
"packageName": "@microsoft/api-extractor-model" | ||
} |
10 changes: 10 additions & 0 deletions
10
common/changes/@microsoft/load-themed-styles/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@microsoft/load-themed-styles", | ||
"comment": "", | ||
"type": "none" | ||
} | ||
], | ||
"packageName": "@microsoft/load-themed-styles" | ||
} |
10 changes: 10 additions & 0 deletions
10
...changes/@microsoft/loader-load-themed-styles/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@microsoft/loader-load-themed-styles", | ||
"comment": "", | ||
"type": "none" | ||
} | ||
], | ||
"packageName": "@microsoft/loader-load-themed-styles" | ||
} |
10 changes: 10 additions & 0 deletions
10
common/changes/@microsoft/rush/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@microsoft/rush", | ||
"comment": "", | ||
"type": "none" | ||
} | ||
], | ||
"packageName": "@microsoft/rush" | ||
} |
10 changes: 10 additions & 0 deletions
10
common/changes/@rushstack/eslint-plugin/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@rushstack/eslint-plugin", | ||
"comment": "Add 4 new ESLint rules: \"@rushstack/no-backslash-imports\", used to prevent backslashes in import and require statements; \"@rushstack/no-external-local-imports\", used to prevent referencing external depedencies in import and require statements; \"@rushstack/no-transitive-dependency-imports\", used to prevent referencing transitive dependencies (ie. dependencies of dependencies) in import and require statements; and \"@rushstack/normalized-imports\", used to ensure that the most direct path to a dependency is provided in import and require statements", | ||
"type": "minor" | ||
} | ||
], | ||
"packageName": "@rushstack/eslint-plugin" | ||
} |
10 changes: 10 additions & 0 deletions
10
common/changes/@rushstack/heft-lint-plugin/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@rushstack/heft-lint-plugin", | ||
"comment": "Unintrusively disable \"--fix\" mode when running in \"--production\" mode", | ||
"type": "patch" | ||
} | ||
], | ||
"packageName": "@rushstack/heft-lint-plugin" | ||
} |
10 changes: 10 additions & 0 deletions
10
common/changes/@rushstack/heft/user-danade-LintTweaks_2024-08-14-21-07.json
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,10 @@ | ||
{ | ||
"changes": [ | ||
{ | ||
"packageName": "@rushstack/heft", | ||
"comment": "", | ||
"type": "none" | ||
} | ||
], | ||
"packageName": "@rushstack/heft" | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license. | ||
// See LICENSE in the project root for license information. | ||
|
||
import * as path from 'path'; | ||
import { ESLintUtils, TSESTree, type TSESLint } from '@typescript-eslint/utils'; | ||
import type { Program } from 'typescript'; | ||
|
||
export interface IParsedImportSpecifier { | ||
loader?: string; | ||
importTarget: string; | ||
loaderOptions?: string; | ||
} | ||
|
||
// Regex to parse out the import target from the specifier. Expected formats are: | ||
// - '<target>' | ||
// - '<loader>!<target>' | ||
// - '<target>?<loader-options>' | ||
// - '<loader>!<target>?<loader-options>' | ||
const LOADER_CAPTURE_GROUP: 'loader' = 'loader'; | ||
const IMPORT_TARGET_CAPTURE_GROUP: 'importTarget' = 'importTarget'; | ||
const LOADER_OPTIONS_CAPTURE_GROUP: 'loaderOptions' = 'loaderOptions'; | ||
// eslint-disable-next-line @rushstack/security/no-unsafe-regexp | ||
const SPECIFIER_REGEX: RegExp = new RegExp( | ||
`^((?<${LOADER_CAPTURE_GROUP}>(!|-!|!!).+)!)?` + | ||
`(?<${IMPORT_TARGET_CAPTURE_GROUP}>[^!?]+)` + | ||
`(\\?(?<${LOADER_OPTIONS_CAPTURE_GROUP}>.*))?$` | ||
); | ||
|
||
export function getFilePathFromContext(context: TSESLint.RuleContext<string, unknown[]>): string { | ||
return context.physicalFilename || context.filename; | ||
} | ||
|
||
export function getRootDirectoryFromContext( | ||
context: TSESLint.RuleContext<string, unknown[]> | ||
): string | undefined { | ||
let rootDirectory: string | undefined; | ||
try { | ||
// First attempt to get the root directory from the tsconfig baseUrl, then the program current directory | ||
const program: Program | null | undefined = ( | ||
context.sourceCode?.parserServices ?? ESLintUtils.getParserServices(context) | ||
).program; | ||
rootDirectory = program?.getCompilerOptions().baseUrl ?? program?.getCurrentDirectory(); | ||
} catch { | ||
// Ignore the error if we cannot retrieve a TS program | ||
} | ||
|
||
// Fall back to the parserOptions.tsconfigRootDir if available, otherwise the eslint working directory | ||
if (!rootDirectory) { | ||
rootDirectory = context.parserOptions?.tsconfigRootDir ?? context.getCwd?.(); | ||
} | ||
|
||
return rootDirectory; | ||
} | ||
|
||
export function parseImportSpecifierFromExpression( | ||
importExpression: TSESTree.Expression | ||
): IParsedImportSpecifier | undefined { | ||
if ( | ||
!importExpression || | ||
importExpression.type !== TSESTree.AST_NODE_TYPES.Literal || | ||
typeof importExpression.value !== 'string' | ||
) { | ||
// Can't determine the path of the import target, return | ||
return undefined; | ||
} | ||
|
||
// Extract the target of the import, stripping out webpack loaders and query strings. The regex will | ||
// also ensure that the import target is a relative path. | ||
const specifierMatch: RegExpMatchArray | null = importExpression.value.match(SPECIFIER_REGEX); | ||
if (!specifierMatch?.groups) { | ||
// Can't determine the path of the import target, return | ||
return undefined; | ||
} | ||
|
||
const loader: string | undefined = specifierMatch.groups[LOADER_CAPTURE_GROUP]; | ||
const importTarget: string = specifierMatch.groups[IMPORT_TARGET_CAPTURE_GROUP]; | ||
const loaderOptions: string | undefined = specifierMatch.groups[LOADER_OPTIONS_CAPTURE_GROUP]; | ||
return { loader, importTarget, loaderOptions }; | ||
} | ||
|
||
export function serializeImportSpecifier(parsedImportPath: IParsedImportSpecifier): string { | ||
const { loader, importTarget, loaderOptions } = parsedImportPath; | ||
return `${loader ? `${loader}!` : ''}${importTarget}${loaderOptions ? `?${loaderOptions}` : ''}`; | ||
} | ||
|
||
export function getImportPathFromExpression( | ||
importExpression: TSESTree.Expression, | ||
relativeImportsOnly: boolean = true | ||
): string | undefined { | ||
const parsedImportSpecifier: IParsedImportSpecifier | undefined = | ||
parseImportSpecifierFromExpression(importExpression); | ||
if ( | ||
!parsedImportSpecifier || | ||
(relativeImportsOnly && !parsedImportSpecifier.importTarget.startsWith('.')) | ||
) { | ||
// The import target isn't a path, return | ||
return undefined; | ||
} | ||
return parsedImportSpecifier?.importTarget; | ||
} | ||
|
||
export function getImportAbsolutePathFromExpression( | ||
context: TSESLint.RuleContext<string, unknown[]>, | ||
importExpression: TSESTree.Expression, | ||
relativeImportsOnly: boolean = true | ||
): string | undefined { | ||
const importPath: string | undefined = getImportPathFromExpression(importExpression, relativeImportsOnly); | ||
if (importPath === undefined) { | ||
// Can't determine the absolute path of the import target, return | ||
return undefined; | ||
} | ||
|
||
const filePath: string = getFilePathFromContext(context); | ||
const fileDirectory: string = path.dirname(filePath); | ||
|
||
// Combine the import path with the absolute path of the file parent directory to get the | ||
// absolute path of the import target | ||
return path.resolve(fileDirectory, importPath); | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
// Copyright (c) Microsoft Corporation. All rights reserved. Licensed under the MIT license. | ||
// See LICENSE in the project root for license information. | ||
|
||
import type { TSESTree, TSESLint } from '@typescript-eslint/utils'; | ||
import { | ||
parseImportSpecifierFromExpression, | ||
serializeImportSpecifier, | ||
type IParsedImportSpecifier | ||
} from './LintUtilities'; | ||
|
||
export const MESSAGE_ID: 'no-backslash-imports' = 'no-backslash-imports'; | ||
type RuleModule = TSESLint.RuleModule<typeof MESSAGE_ID, []>; | ||
type RuleContext = TSESLint.RuleContext<typeof MESSAGE_ID, []>; | ||
|
||
export const noBackslashImportsRule: RuleModule = { | ||
defaultOptions: [], | ||
meta: { | ||
type: 'problem', | ||
messages: { | ||
[MESSAGE_ID]: 'The specified import target path contains backslashes.' | ||
}, | ||
schema: [], | ||
docs: { | ||
description: 'Prevents imports using paths that use backslashes', | ||
url: 'https://www.npmjs.com/package/@rushstack/eslint-plugin' | ||
}, | ||
fixable: 'code' | ||
}, | ||
create: (context: RuleContext) => { | ||
const checkImportExpression: (importExpression: TSESTree.Expression | null) => void = ( | ||
importExpression: TSESTree.Expression | null | ||
) => { | ||
if (!importExpression) { | ||
// Can't validate, return | ||
return; | ||
} | ||
|
||
// Determine the target file path and find the most direct relative path from the source file | ||
const importSpecifier: IParsedImportSpecifier | undefined = | ||
parseImportSpecifierFromExpression(importExpression); | ||
if (importSpecifier === undefined) { | ||
// Can't validate, return | ||
return; | ||
} | ||
|
||
// Check if the import path contains backslashes. If it does, suggest a fix to replace them with forward | ||
// slashes. | ||
const { importTarget } = importSpecifier; | ||
if (importTarget.includes('\\')) { | ||
context.report({ | ||
node: importExpression, | ||
messageId: MESSAGE_ID, | ||
fix: (fixer: TSESLint.RuleFixer) => { | ||
const normalizedSpecifier: IParsedImportSpecifier = { | ||
...importSpecifier, | ||
importTarget: importTarget.replace(/\\/g, '/') | ||
}; | ||
return fixer.replaceText(importExpression, `'${serializeImportSpecifier(normalizedSpecifier)}'`); | ||
} | ||
}); | ||
} | ||
}; | ||
|
||
return { | ||
ImportDeclaration: (node: TSESTree.ImportDeclaration) => checkImportExpression(node.source), | ||
ImportExpression: (node: TSESTree.ImportExpression) => checkImportExpression(node.source), | ||
ExportAllDeclaration: (node: TSESTree.ExportAllDeclaration) => checkImportExpression(node.source), | ||
ExportNamedDeclaration: (node: TSESTree.ExportNamedDeclaration) => checkImportExpression(node.source) | ||
}; | ||
} | ||
}; |
Oops, something went wrong.