-
-
Notifications
You must be signed in to change notification settings - Fork 2.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(eslint-plugin): add suggestion to
require-await
to remove `asy…
…nc` keyword (#9718) * Add suggestion to require-await to remove async keyword Fixes #9621 * Include suggestion when there is a return type annotation. * Restructure suggestion range calculation. * Suggestion will replace return type of `Promise<T>` with `T`. * use noFormat instead of suppressing lint errors. * Remove unnecessary ecmaVersion from tests and correct test that had auto-formatting applied. * Corrected comment. * Updated the copy of ESLint test cases. * Added better support for async generators. * Used nullThrows to avoid unnecessary conditionals. * Added additional changes to end of array instead of inserting at the start. * Use removeRange instead of replaceTextRange with empty replacement. * Change typeArguments null check. * Replaced incorrect type guard.
- Loading branch information
Showing
5 changed files
with
850 additions
and
12 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
22 changes: 22 additions & 0 deletions
22
packages/eslint-plugin/src/util/isStartOfExpressionStatement.ts
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,22 @@ | ||
import type { TSESTree } from '@typescript-eslint/utils'; | ||
import { AST_NODE_TYPES } from '@typescript-eslint/utils'; | ||
|
||
// The following is copied from `eslint`'s source code. | ||
// https://github.com/eslint/eslint/blob/3a4eaf921543b1cd5d1df4ea9dec02fab396af2a/lib/rules/utils/ast-utils.js#L1026-L1041 | ||
// Could be export { isStartOfExpressionStatement } from 'eslint/lib/rules/utils/ast-utils' | ||
/** | ||
* Tests if a node appears at the beginning of an ancestor ExpressionStatement node. | ||
* @param node The node to check. | ||
* @returns Whether the node appears at the beginning of an ancestor ExpressionStatement node. | ||
*/ | ||
export function isStartOfExpressionStatement(node: TSESTree.Node): boolean { | ||
const start = node.range[0]; | ||
let ancestor: TSESTree.Node | undefined = node; | ||
|
||
while ((ancestor = ancestor.parent) && ancestor.range[0] === start) { | ||
if (ancestor.type === AST_NODE_TYPES.ExpressionStatement) { | ||
return true; | ||
} | ||
} | ||
return false; | ||
} |
125 changes: 125 additions & 0 deletions
125
packages/eslint-plugin/src/util/needsPrecedingSemiColon.ts
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,125 @@ | ||
import type { TSESTree } from '@typescript-eslint/utils'; | ||
import { AST_NODE_TYPES, AST_TOKEN_TYPES } from '@typescript-eslint/utils'; | ||
import { | ||
isClosingBraceToken, | ||
isClosingParenToken, | ||
} from '@typescript-eslint/utils/ast-utils'; | ||
import type { SourceCode } from '@typescript-eslint/utils/ts-eslint'; | ||
|
||
// The following is adapted from `eslint`'s source code. | ||
// https://github.com/eslint/eslint/blob/3a4eaf921543b1cd5d1df4ea9dec02fab396af2a/lib/rules/utils/ast-utils.js#L1043-L1132 | ||
// Could be export { isStartOfExpressionStatement } from 'eslint/lib/rules/utils/ast-utils' | ||
|
||
const BREAK_OR_CONTINUE = new Set([ | ||
AST_NODE_TYPES.BreakStatement, | ||
AST_NODE_TYPES.ContinueStatement, | ||
]); | ||
|
||
// Declaration types that must contain a string Literal node at the end. | ||
const DECLARATIONS = new Set([ | ||
AST_NODE_TYPES.ExportAllDeclaration, | ||
AST_NODE_TYPES.ExportNamedDeclaration, | ||
AST_NODE_TYPES.ImportDeclaration, | ||
]); | ||
|
||
const IDENTIFIER_OR_KEYWORD = new Set([ | ||
AST_NODE_TYPES.Identifier, | ||
AST_TOKEN_TYPES.Keyword, | ||
]); | ||
|
||
// Keywords that can immediately precede an ExpressionStatement node, mapped to the their node types. | ||
const NODE_TYPES_BY_KEYWORD: Record<string, TSESTree.AST_NODE_TYPES | null> = { | ||
__proto__: null, | ||
break: AST_NODE_TYPES.BreakStatement, | ||
continue: AST_NODE_TYPES.ContinueStatement, | ||
debugger: AST_NODE_TYPES.DebuggerStatement, | ||
do: AST_NODE_TYPES.DoWhileStatement, | ||
else: AST_NODE_TYPES.IfStatement, | ||
return: AST_NODE_TYPES.ReturnStatement, | ||
yield: AST_NODE_TYPES.YieldExpression, | ||
}; | ||
|
||
/* | ||
* Before an opening parenthesis, postfix `++` and `--` always trigger ASI; | ||
* the tokens `:`, `;`, `{` and `=>` don't expect a semicolon, as that would count as an empty statement. | ||
*/ | ||
const PUNCTUATORS = new Set([':', ';', '{', '=>', '++', '--']); | ||
|
||
/* | ||
* Statements that can contain an `ExpressionStatement` after a closing parenthesis. | ||
* DoWhileStatement is an exception in that it always triggers ASI after the closing parenthesis. | ||
*/ | ||
const STATEMENTS = new Set([ | ||
AST_NODE_TYPES.DoWhileStatement, | ||
AST_NODE_TYPES.ForInStatement, | ||
AST_NODE_TYPES.ForOfStatement, | ||
AST_NODE_TYPES.ForStatement, | ||
AST_NODE_TYPES.IfStatement, | ||
AST_NODE_TYPES.WhileStatement, | ||
AST_NODE_TYPES.WithStatement, | ||
]); | ||
|
||
/** | ||
* Determines whether an opening parenthesis `(`, bracket `[` or backtick ``` ` ``` needs to be preceded by a semicolon. | ||
* This opening parenthesis or bracket should be at the start of an `ExpressionStatement`, a `MethodDefinition` or at | ||
* the start of the body of an `ArrowFunctionExpression`. | ||
* @param sourceCode The source code object. | ||
* @param node A node at the position where an opening parenthesis or bracket will be inserted. | ||
* @returns Whether a semicolon is required before the opening parenthesis or bracket. | ||
*/ | ||
export function needsPrecedingSemicolon( | ||
sourceCode: SourceCode, | ||
node: TSESTree.Node, | ||
): boolean { | ||
const prevToken = sourceCode.getTokenBefore(node); | ||
|
||
if ( | ||
!prevToken || | ||
(prevToken.type === AST_TOKEN_TYPES.Punctuator && | ||
PUNCTUATORS.has(prevToken.value)) | ||
) { | ||
return false; | ||
} | ||
|
||
const prevNode = sourceCode.getNodeByRangeIndex(prevToken.range[0]); | ||
|
||
if (!prevNode) { | ||
return false; | ||
} | ||
|
||
if (isClosingParenToken(prevToken)) { | ||
return !STATEMENTS.has(prevNode.type); | ||
} | ||
|
||
if (isClosingBraceToken(prevToken)) { | ||
return ( | ||
(prevNode.type === AST_NODE_TYPES.BlockStatement && | ||
prevNode.parent.type === AST_NODE_TYPES.FunctionExpression && | ||
prevNode.parent.parent.type !== AST_NODE_TYPES.MethodDefinition) || | ||
(prevNode.type === AST_NODE_TYPES.ClassBody && | ||
prevNode.parent.type === AST_NODE_TYPES.ClassExpression) || | ||
prevNode.type === AST_NODE_TYPES.ObjectExpression | ||
); | ||
} | ||
|
||
if (!prevNode.parent) { | ||
return false; | ||
} | ||
|
||
if (IDENTIFIER_OR_KEYWORD.has(prevToken.type)) { | ||
if (BREAK_OR_CONTINUE.has(prevNode.parent.type)) { | ||
return false; | ||
} | ||
|
||
const keyword = prevToken.value; | ||
const nodeType = NODE_TYPES_BY_KEYWORD[keyword]; | ||
|
||
return prevNode.type !== nodeType; | ||
} | ||
|
||
if (prevToken.type === AST_TOKEN_TYPES.String) { | ||
return !DECLARATIONS.has(prevNode.parent.type); | ||
} | ||
|
||
return true; | ||
} |
Oops, something went wrong.