@typescript-eslint/eslint-plugin
Version:
TypeScript plugin for ESLint
631 lines (605 loc) • 28.8 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const scope_manager_1 = require("@typescript-eslint/scope-manager");
const utils_1 = require("@typescript-eslint/utils");
const util_1 = require("../util");
const referenceContainsTypeQuery_1 = require("../util/referenceContainsTypeQuery");
exports.default = (0, util_1.createRule)({
name: 'no-unused-vars',
meta: {
type: 'problem',
docs: {
description: 'Disallow unused variables',
recommended: 'recommended',
extendsBaseRule: true,
},
schema: [
{
oneOf: [
{
type: 'string',
enum: ['all', 'local'],
},
{
type: 'object',
properties: {
vars: {
description: 'Whether to check all variables or only locally-declared variables.',
type: 'string',
enum: ['all', 'local'],
},
varsIgnorePattern: {
description: 'Regular expressions of variable names to not check for usage.',
type: 'string',
},
args: {
description: 'Whether to check all, some, or no arguments.',
type: 'string',
enum: ['all', 'after-used', 'none'],
},
argsIgnorePattern: {
description: 'Regular expressions of argument names to not check for usage.',
type: 'string',
},
caughtErrors: {
description: 'Whether to check catch block arguments.',
type: 'string',
enum: ['all', 'none'],
},
caughtErrorsIgnorePattern: {
description: 'Regular expressions of catch block argument names to not check for usage.',
type: 'string',
},
destructuredArrayIgnorePattern: {
description: 'Regular expressions of destructured array variable names to not check for usage.',
type: 'string',
},
ignoreClassWithStaticInitBlock: {
description: 'Whether to ignore classes with at least one static initialization block.',
type: 'boolean',
},
ignoreRestSiblings: {
description: 'Whether to ignore sibling properties in `...` destructurings.',
type: 'boolean',
},
reportUsedIgnorePattern: {
description: 'Whether to report variables that match any of the valid ignore pattern options if they have been used.',
type: 'boolean',
},
},
additionalProperties: false,
},
],
},
],
messages: {
unusedVar: "'{{varName}}' is {{action}} but never used{{additional}}.",
usedIgnoredVar: "'{{varName}}' is marked as ignored but is used{{additional}}.",
usedOnlyAsType: "'{{varName}}' is {{action}} but only used as a type{{additional}}.",
},
},
defaultOptions: [{}],
create(context, [firstOption]) {
const MODULE_DECL_CACHE = new Map();
const options = (() => {
const options = {
vars: 'all',
args: 'after-used',
ignoreRestSiblings: false,
caughtErrors: 'all',
ignoreClassWithStaticInitBlock: false,
reportUsedIgnorePattern: false,
};
if (typeof firstOption === 'string') {
options.vars = firstOption;
}
else {
options.vars = firstOption.vars ?? options.vars;
options.args = firstOption.args ?? options.args;
options.ignoreRestSiblings =
firstOption.ignoreRestSiblings ?? options.ignoreRestSiblings;
options.caughtErrors = firstOption.caughtErrors ?? options.caughtErrors;
options.ignoreClassWithStaticInitBlock =
firstOption.ignoreClassWithStaticInitBlock ??
options.ignoreClassWithStaticInitBlock;
options.reportUsedIgnorePattern =
firstOption.reportUsedIgnorePattern ??
options.reportUsedIgnorePattern;
if (firstOption.varsIgnorePattern) {
options.varsIgnorePattern = new RegExp(firstOption.varsIgnorePattern, 'u');
}
if (firstOption.argsIgnorePattern) {
options.argsIgnorePattern = new RegExp(firstOption.argsIgnorePattern, 'u');
}
if (firstOption.caughtErrorsIgnorePattern) {
options.caughtErrorsIgnorePattern = new RegExp(firstOption.caughtErrorsIgnorePattern, 'u');
}
if (firstOption.destructuredArrayIgnorePattern) {
options.destructuredArrayIgnorePattern = new RegExp(firstOption.destructuredArrayIgnorePattern, 'u');
}
}
return options;
})();
/**
* Determines what variable type a def is.
* @param def the declaration to check
* @returns a simple name for the types of variables that this rule supports
*/
function defToVariableType(def) {
/*
* This `destructuredArrayIgnorePattern` error report works differently from the catch
* clause and parameter error reports. _Both_ the `varsIgnorePattern` and the
* `destructuredArrayIgnorePattern` will be checked for array destructuring. However,
* for the purposes of the report, the currently defined behavior is to only inform the
* user of the `destructuredArrayIgnorePattern` if it's present (regardless of the fact
* that the `varsIgnorePattern` would also apply). If it's not present, the user will be
* informed of the `varsIgnorePattern`, assuming that's present.
*/
if (options.destructuredArrayIgnorePattern &&
def.name.parent.type === utils_1.AST_NODE_TYPES.ArrayPattern) {
return 'array-destructure';
}
switch (def.type) {
case scope_manager_1.DefinitionType.CatchClause:
return 'catch-clause';
case scope_manager_1.DefinitionType.Parameter:
return 'parameter';
default:
return 'variable';
}
}
/**
* Gets a given variable's description and configured ignore pattern
* based on the provided variableType
* @param variableType a simple name for the types of variables that this rule supports
* @returns the given variable's description and
* ignore pattern
*/
function getVariableDescription(variableType) {
switch (variableType) {
case 'array-destructure':
return {
pattern: options.destructuredArrayIgnorePattern?.toString(),
variableDescription: 'elements of array destructuring',
};
case 'catch-clause':
return {
pattern: options.caughtErrorsIgnorePattern?.toString(),
variableDescription: 'caught errors',
};
case 'parameter':
return {
pattern: options.argsIgnorePattern?.toString(),
variableDescription: 'args',
};
case 'variable':
return {
pattern: options.varsIgnorePattern?.toString(),
variableDescription: 'vars',
};
}
}
/**
* Generates the message data about the variable being defined and unused,
* including the ignore pattern if configured.
* @param unusedVar eslint-scope variable object.
* @returns The message data to be used with this unused variable.
*/
function getDefinedMessageData(unusedVar) {
const def = unusedVar.defs.at(0);
let additionalMessageData = '';
if (def) {
const { variableDescription, pattern } = getVariableDescription(defToVariableType(def));
if (pattern && variableDescription) {
additionalMessageData = `. Allowed unused ${variableDescription} must match ${pattern}`;
}
}
return {
varName: unusedVar.name,
action: 'defined',
additional: additionalMessageData,
};
}
/**
* Generate the warning message about the variable being
* assigned and unused, including the ignore pattern if configured.
* @param unusedVar eslint-scope variable object.
* @returns The message data to be used with this unused variable.
*/
function getAssignedMessageData(unusedVar) {
const def = unusedVar.defs.at(0);
let additionalMessageData = '';
if (def) {
const { variableDescription, pattern } = getVariableDescription(defToVariableType(def));
if (pattern && variableDescription) {
additionalMessageData = `. Allowed unused ${variableDescription} must match ${pattern}`;
}
}
return {
varName: unusedVar.name,
action: 'assigned a value',
additional: additionalMessageData,
};
}
/**
* Generate the warning message about a variable being used even though
* it is marked as being ignored.
* @param variable eslint-scope variable object
* @param variableType a simple name for the types of variables that this rule supports
* @returns The message data to be used with this used ignored variable.
*/
function getUsedIgnoredMessageData(variable, variableType) {
const { variableDescription, pattern } = getVariableDescription(variableType);
let additionalMessageData = '';
if (pattern && variableDescription) {
additionalMessageData = `. Used ${variableDescription} must not match ${pattern}`;
}
return {
varName: variable.name,
additional: additionalMessageData,
};
}
function collectUnusedVariables() {
/**
* Checks whether a node is a sibling of the rest property or not.
* @param node a node to check
* @returns True if the node is a sibling of the rest property, otherwise false.
*/
function hasRestSibling(node) {
return (node.type === utils_1.AST_NODE_TYPES.Property &&
node.parent.type === utils_1.AST_NODE_TYPES.ObjectPattern &&
node.parent.properties[node.parent.properties.length - 1].type ===
utils_1.AST_NODE_TYPES.RestElement);
}
/**
* Determines if a variable has a sibling rest property
* @param variable eslint-scope variable object.
* @returns True if the variable is exported, false if not.
*/
function hasRestSpreadSibling(variable) {
if (options.ignoreRestSiblings) {
const hasRestSiblingDefinition = variable.defs.some(def => hasRestSibling(def.name.parent));
const hasRestSiblingReference = variable.references.some(ref => hasRestSibling(ref.identifier.parent));
return hasRestSiblingDefinition || hasRestSiblingReference;
}
return false;
}
/**
* Checks whether the given variable is after the last used parameter.
* @param variable The variable to check.
* @returns `true` if the variable is defined after the last used parameter.
*/
function isAfterLastUsedArg(variable) {
const def = variable.defs[0];
const params = context.sourceCode.getDeclaredVariables(def.node);
const posteriorParams = params.slice(params.indexOf(variable) + 1);
// If any used parameters occur after this parameter, do not report.
return !posteriorParams.some(v => v.references.length > 0 || v.eslintUsed);
}
const analysisResults = (0, util_1.collectVariables)(context);
const variables = [
...Array.from(analysisResults.unusedVariables, variable => ({
used: false,
variable,
})),
...Array.from(analysisResults.usedVariables, variable => ({
used: true,
variable,
})),
];
const unusedVariablesReturn = [];
for (const { used, variable } of variables) {
// explicit global variables don't have definitions.
if (variable.defs.length === 0) {
if (!used) {
unusedVariablesReturn.push(variable);
}
continue;
}
const def = variable.defs[0];
if (variable.scope.type === utils_1.TSESLint.Scope.ScopeType.global &&
options.vars === 'local') {
// skip variables in the global scope if configured to
continue;
}
const refUsedInArrayPatterns = variable.references.some(ref => ref.identifier.parent.type === utils_1.AST_NODE_TYPES.ArrayPattern);
// skip elements of array destructuring patterns
if ((def.name.parent.type === utils_1.AST_NODE_TYPES.ArrayPattern ||
refUsedInArrayPatterns) &&
def.name.type === utils_1.AST_NODE_TYPES.Identifier &&
options.destructuredArrayIgnorePattern?.test(def.name.name)) {
if (options.reportUsedIgnorePattern && used) {
context.report({
node: def.name,
messageId: 'usedIgnoredVar',
data: getUsedIgnoredMessageData(variable, 'array-destructure'),
});
}
continue;
}
if (def.type === utils_1.TSESLint.Scope.DefinitionType.ClassName) {
const hasStaticBlock = def.node.body.body.some(node => node.type === utils_1.AST_NODE_TYPES.StaticBlock);
if (options.ignoreClassWithStaticInitBlock && hasStaticBlock) {
continue;
}
}
// skip catch variables
if (def.type === utils_1.TSESLint.Scope.DefinitionType.CatchClause) {
if (options.caughtErrors === 'none') {
continue;
}
// skip ignored parameters
if (def.name.type === utils_1.AST_NODE_TYPES.Identifier &&
options.caughtErrorsIgnorePattern?.test(def.name.name)) {
if (options.reportUsedIgnorePattern && used) {
context.report({
node: def.name,
messageId: 'usedIgnoredVar',
data: getUsedIgnoredMessageData(variable, 'catch-clause'),
});
}
continue;
}
}
else if (def.type === utils_1.TSESLint.Scope.DefinitionType.Parameter) {
// if "args" option is "none", skip any parameter
if (options.args === 'none') {
continue;
}
// skip ignored parameters
if (def.name.type === utils_1.AST_NODE_TYPES.Identifier &&
options.argsIgnorePattern?.test(def.name.name)) {
if (options.reportUsedIgnorePattern && used) {
context.report({
node: def.name,
messageId: 'usedIgnoredVar',
data: getUsedIgnoredMessageData(variable, 'parameter'),
});
}
continue;
}
// if "args" option is "after-used", skip used variables
if (options.args === 'after-used' &&
(0, util_1.isFunction)(def.name.parent) &&
!isAfterLastUsedArg(variable)) {
continue;
}
}
// skip ignored variables
else if (def.name.type === utils_1.AST_NODE_TYPES.Identifier &&
options.varsIgnorePattern?.test(def.name.name)) {
if (options.reportUsedIgnorePattern && used) {
context.report({
node: def.name,
messageId: 'usedIgnoredVar',
data: getUsedIgnoredMessageData(variable, 'variable'),
});
}
continue;
}
if (hasRestSpreadSibling(variable)) {
continue;
}
// in case another rule has run and used the collectUnusedVariables,
// we want to ensure our selectors that marked variables as used are respected
if (variable.eslintUsed) {
continue;
}
if (!used) {
unusedVariablesReturn.push(variable);
}
}
return unusedVariablesReturn;
}
return {
// declaration file handling
[ambientDeclarationSelector(utils_1.AST_NODE_TYPES.Program, true)](node) {
if (!(0, util_1.isDefinitionFile)(context.filename)) {
return;
}
markDeclarationChildAsUsed(node);
},
// children of a namespace that is a child of a declared namespace are auto-exported
[ambientDeclarationSelector('TSModuleDeclaration[declare = true] > TSModuleBlock TSModuleDeclaration > TSModuleBlock', false)](node) {
markDeclarationChildAsUsed(node);
},
// declared namespace handling
[ambientDeclarationSelector('TSModuleDeclaration[declare = true] > TSModuleBlock', false)](node) {
const moduleDecl = (0, util_1.nullThrows)(node.parent.parent, util_1.NullThrowsReasons.MissingParent);
// declared ambient modules with an `export =` statement will only export that one thing
// all other statements are not automatically exported in this case
if (moduleDecl.id.type === utils_1.AST_NODE_TYPES.Literal &&
checkModuleDeclForExportEquals(moduleDecl)) {
return;
}
markDeclarationChildAsUsed(node);
},
// collect
'Program:exit'(programNode) {
const unusedVars = collectUnusedVariables();
for (const unusedVar of unusedVars) {
// Report the first declaration.
if (unusedVar.defs.length > 0) {
const usedOnlyAsType = unusedVar.references.some(ref => (0, referenceContainsTypeQuery_1.referenceContainsTypeQuery)(ref.identifier));
const isImportUsedOnlyAsType = usedOnlyAsType &&
unusedVar.defs.some(def => def.type === scope_manager_1.DefinitionType.ImportBinding);
if (isImportUsedOnlyAsType) {
continue;
}
const writeReferences = unusedVar.references.filter(ref => ref.isWrite() &&
ref.from.variableScope === unusedVar.scope.variableScope);
const id = writeReferences.length
? writeReferences[writeReferences.length - 1].identifier
: unusedVar.identifiers[0];
const messageId = usedOnlyAsType ? 'usedOnlyAsType' : 'unusedVar';
const { start } = id.loc;
const idLength = id.name.length;
const loc = {
start,
end: {
line: start.line,
column: start.column + idLength,
},
};
context.report({
loc,
messageId,
data: unusedVar.references.some(ref => ref.isWrite())
? getAssignedMessageData(unusedVar)
: getDefinedMessageData(unusedVar),
});
// If there are no regular declaration, report the first `/*globals*/` comment directive.
}
else if ('eslintExplicitGlobalComments' in unusedVar &&
unusedVar.eslintExplicitGlobalComments) {
const directiveComment = unusedVar.eslintExplicitGlobalComments[0];
context.report({
node: programNode,
loc: (0, util_1.getNameLocationInGlobalDirectiveComment)(context.sourceCode, directiveComment, unusedVar.name),
messageId: 'unusedVar',
data: getDefinedMessageData(unusedVar),
});
}
}
},
};
function checkModuleDeclForExportEquals(node) {
const cached = MODULE_DECL_CACHE.get(node);
if (cached != null) {
return cached;
}
if (node.body) {
for (const statement of node.body.body) {
if (statement.type === utils_1.AST_NODE_TYPES.TSExportAssignment) {
MODULE_DECL_CACHE.set(node, true);
return true;
}
}
}
MODULE_DECL_CACHE.set(node, false);
return false;
}
function ambientDeclarationSelector(parent, childDeclare) {
return [
// Types are ambiently exported
`${parent} > :matches(${[
utils_1.AST_NODE_TYPES.TSInterfaceDeclaration,
utils_1.AST_NODE_TYPES.TSTypeAliasDeclaration,
].join(', ')})`,
// Value things are ambiently exported if they are "declare"d
`${parent} > :matches(${[
utils_1.AST_NODE_TYPES.ClassDeclaration,
utils_1.AST_NODE_TYPES.TSDeclareFunction,
utils_1.AST_NODE_TYPES.TSEnumDeclaration,
utils_1.AST_NODE_TYPES.TSModuleDeclaration,
utils_1.AST_NODE_TYPES.VariableDeclaration,
].join(', ')})${childDeclare ? '[declare = true]' : ''}`,
].join(', ');
}
function markDeclarationChildAsUsed(node) {
const identifiers = [];
switch (node.type) {
case utils_1.AST_NODE_TYPES.TSInterfaceDeclaration:
case utils_1.AST_NODE_TYPES.TSTypeAliasDeclaration:
case utils_1.AST_NODE_TYPES.ClassDeclaration:
case utils_1.AST_NODE_TYPES.FunctionDeclaration:
case utils_1.AST_NODE_TYPES.TSDeclareFunction:
case utils_1.AST_NODE_TYPES.TSEnumDeclaration:
case utils_1.AST_NODE_TYPES.TSModuleDeclaration:
if (node.id?.type === utils_1.AST_NODE_TYPES.Identifier) {
identifiers.push(node.id);
}
break;
case utils_1.AST_NODE_TYPES.VariableDeclaration:
for (const declaration of node.declarations) {
visitPattern(declaration, pattern => {
identifiers.push(pattern);
});
}
break;
}
let scope = context.sourceCode.getScope(node);
const shouldUseUpperScope = [
utils_1.AST_NODE_TYPES.TSModuleDeclaration,
utils_1.AST_NODE_TYPES.TSDeclareFunction,
].includes(node.type);
if (scope.variableScope !== scope) {
scope = scope.variableScope;
}
else if (shouldUseUpperScope && scope.upper) {
scope = scope.upper;
}
for (const id of identifiers) {
const superVar = scope.set.get(id.name);
if (superVar) {
superVar.eslintUsed = true;
}
}
}
function visitPattern(node, cb) {
const visitor = new scope_manager_1.PatternVisitor({}, node, cb);
visitor.visit(node);
}
},
});
/*
###### TODO ######
Edge cases that aren't currently handled due to laziness and them being super edgy edge cases
--- function params referenced in typeof type refs in the function declaration ---
--- NOTE - TS gets these cases wrong
function _foo(
arg: number // arg should be unused
): typeof arg {
return 1 as any;
}
function _bar(
arg: number, // arg should be unused
_arg2: typeof arg,
) {}
--- function names referenced in typeof type refs in the function declaration ---
--- NOTE - TS gets these cases right
function foo( // foo should be unused
): typeof foo {
return 1 as any;
}
function bar( // bar should be unused
_arg: typeof bar
) {}
--- if an interface is merged into a namespace ---
--- NOTE - TS gets these cases wrong
namespace Test {
interface Foo { // Foo should be unused here
a: string;
}
export namespace Foo {
export type T = 'b';
}
}
type T = Test.Foo; // Error: Namespace 'Test' has no exported member 'Foo'.
namespace Test {
export interface Foo {
a: string;
}
namespace Foo { // Foo should be unused here
export type T = 'b';
}
}
type T = Test.Foo.T; // Error: Namespace 'Test' has no exported member 'Foo'.
---
These cases are mishandled because the base rule assumes that each variable has one def, but type-value shadowing
creates a variable with two defs
--- type-only or value-only references to type/value shadowed variables ---
--- NOTE - TS gets these cases wrong
type T = 1;
const T = 2; // this T should be unused
type U = T; // this U should be unused
const U = 3;
const _V = U;
--- partially exported type/value shadowed variables ---
--- NOTE - TS gets these cases wrong
export interface Foo {}
const Foo = 1; // this Foo should be unused
interface Bar {} // this Bar should be unused
export const Bar = 1;
*/
//# sourceMappingURL=no-unused-vars.js.map