Every new change
This commit is contained in:
172
node_modules/eslint/lib/rules/no-redeclare.js
generated
vendored
Normal file
172
node_modules/eslint/lib/rules/no-redeclare.js
generated
vendored
Normal file
@ -0,0 +1,172 @@
|
||||
/**
|
||||
* @fileoverview Rule to flag when the same variable is declared more then once.
|
||||
* @author Ilya Volodin
|
||||
*/
|
||||
|
||||
"use strict";
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Requirements
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
const astUtils = require("./utils/ast-utils");
|
||||
|
||||
//------------------------------------------------------------------------------
|
||||
// Rule Definition
|
||||
//------------------------------------------------------------------------------
|
||||
|
||||
module.exports = {
|
||||
meta: {
|
||||
type: "suggestion",
|
||||
|
||||
docs: {
|
||||
description: "disallow variable redeclaration",
|
||||
category: "Best Practices",
|
||||
recommended: true,
|
||||
url: "https://eslint.org/docs/rules/no-redeclare"
|
||||
},
|
||||
|
||||
messages: {
|
||||
redeclared: "'{{id}}' is already defined.",
|
||||
redeclaredAsBuiltin: "'{{id}}' is already defined as a built-in global variable.",
|
||||
redeclaredBySyntax: "'{{id}}' is already defined by a variable declaration."
|
||||
},
|
||||
|
||||
schema: [
|
||||
{
|
||||
type: "object",
|
||||
properties: {
|
||||
builtinGlobals: { type: "boolean", default: true }
|
||||
},
|
||||
additionalProperties: false
|
||||
}
|
||||
]
|
||||
},
|
||||
|
||||
create(context) {
|
||||
const options = {
|
||||
builtinGlobals: Boolean(
|
||||
context.options.length === 0 ||
|
||||
context.options[0].builtinGlobals
|
||||
)
|
||||
};
|
||||
const sourceCode = context.getSourceCode();
|
||||
|
||||
/**
|
||||
* Iterate declarations of a given variable.
|
||||
* @param {escope.variable} variable The variable object to iterate declarations.
|
||||
* @returns {IterableIterator<{type:string,node:ASTNode,loc:SourceLocation}>} The declarations.
|
||||
*/
|
||||
function *iterateDeclarations(variable) {
|
||||
if (options.builtinGlobals && (
|
||||
variable.eslintImplicitGlobalSetting === "readonly" ||
|
||||
variable.eslintImplicitGlobalSetting === "writable"
|
||||
)) {
|
||||
yield { type: "builtin" };
|
||||
}
|
||||
|
||||
for (const id of variable.identifiers) {
|
||||
yield { type: "syntax", node: id, loc: id.loc };
|
||||
}
|
||||
|
||||
if (variable.eslintExplicitGlobalComments) {
|
||||
for (const comment of variable.eslintExplicitGlobalComments) {
|
||||
yield {
|
||||
type: "comment",
|
||||
node: comment,
|
||||
loc: astUtils.getNameLocationInGlobalDirectiveComment(
|
||||
sourceCode,
|
||||
comment,
|
||||
variable.name
|
||||
)
|
||||
};
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Find variables in a given scope and flag redeclared ones.
|
||||
* @param {Scope} scope An eslint-scope scope object.
|
||||
* @returns {void}
|
||||
* @private
|
||||
*/
|
||||
function findVariablesInScope(scope) {
|
||||
for (const variable of scope.variables) {
|
||||
const [
|
||||
declaration,
|
||||
...extraDeclarations
|
||||
] = iterateDeclarations(variable);
|
||||
|
||||
if (extraDeclarations.length === 0) {
|
||||
continue;
|
||||
}
|
||||
|
||||
/*
|
||||
* If the type of a declaration is different from the type of
|
||||
* the first declaration, it shows the location of the first
|
||||
* declaration.
|
||||
*/
|
||||
const detailMessageId = declaration.type === "builtin"
|
||||
? "redeclaredAsBuiltin"
|
||||
: "redeclaredBySyntax";
|
||||
const data = { id: variable.name };
|
||||
|
||||
// Report extra declarations.
|
||||
for (const { type, node, loc } of extraDeclarations) {
|
||||
const messageId = type === declaration.type
|
||||
? "redeclared"
|
||||
: detailMessageId;
|
||||
|
||||
context.report({ node, loc, messageId, data });
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Find variables in the current scope.
|
||||
* @param {ASTNode} node The node of the current scope.
|
||||
* @returns {void}
|
||||
* @private
|
||||
*/
|
||||
function checkForBlock(node) {
|
||||
const scope = context.getScope();
|
||||
|
||||
/*
|
||||
* In ES5, some node type such as `BlockStatement` doesn't have that scope.
|
||||
* `scope.block` is a different node in such a case.
|
||||
*/
|
||||
if (scope.block === node) {
|
||||
findVariablesInScope(scope);
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
Program() {
|
||||
const scope = context.getScope();
|
||||
|
||||
findVariablesInScope(scope);
|
||||
|
||||
// Node.js or ES modules has a special scope.
|
||||
if (
|
||||
scope.type === "global" &&
|
||||
scope.childScopes[0] &&
|
||||
|
||||
// The special scope's block is the Program node.
|
||||
scope.block === scope.childScopes[0].block
|
||||
) {
|
||||
findVariablesInScope(scope.childScopes[0]);
|
||||
}
|
||||
},
|
||||
|
||||
FunctionDeclaration: checkForBlock,
|
||||
FunctionExpression: checkForBlock,
|
||||
ArrowFunctionExpression: checkForBlock,
|
||||
|
||||
BlockStatement: checkForBlock,
|
||||
ForStatement: checkForBlock,
|
||||
ForInStatement: checkForBlock,
|
||||
ForOfStatement: checkForBlock,
|
||||
SwitchStatement: checkForBlock
|
||||
};
|
||||
}
|
||||
};
|
Reference in New Issue
Block a user