568 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			568 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /**
 | |
|  * @fileoverview A rule to control the use of single variable declarations.
 | |
|  * @author Ian Christian Myers
 | |
|  */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| //------------------------------------------------------------------------------
 | |
| // Requirements
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| const astUtils = require("./utils/ast-utils");
 | |
| 
 | |
| //------------------------------------------------------------------------------
 | |
| // Helpers
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| /**
 | |
|  * Determines whether the given node is in a statement list.
 | |
|  * @param {ASTNode} node node to check
 | |
|  * @returns {boolean} `true` if the given node is in a statement list
 | |
|  */
 | |
| function isInStatementList(node) {
 | |
|     return astUtils.STATEMENT_LIST_PARENTS.has(node.parent.type);
 | |
| }
 | |
| 
 | |
| //------------------------------------------------------------------------------
 | |
| // Rule Definition
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| /** @type {import('../shared/types').Rule} */
 | |
| module.exports = {
 | |
|     meta: {
 | |
|         type: "suggestion",
 | |
| 
 | |
|         docs: {
 | |
|             description: "Enforce variables to be declared either together or separately in functions",
 | |
|             recommended: false,
 | |
|             url: "https://eslint.org/docs/latest/rules/one-var"
 | |
|         },
 | |
| 
 | |
|         fixable: "code",
 | |
| 
 | |
|         schema: [
 | |
|             {
 | |
|                 oneOf: [
 | |
|                     {
 | |
|                         enum: ["always", "never", "consecutive"]
 | |
|                     },
 | |
|                     {
 | |
|                         type: "object",
 | |
|                         properties: {
 | |
|                             separateRequires: {
 | |
|                                 type: "boolean"
 | |
|                             },
 | |
|                             var: {
 | |
|                                 enum: ["always", "never", "consecutive"]
 | |
|                             },
 | |
|                             let: {
 | |
|                                 enum: ["always", "never", "consecutive"]
 | |
|                             },
 | |
|                             const: {
 | |
|                                 enum: ["always", "never", "consecutive"]
 | |
|                             }
 | |
|                         },
 | |
|                         additionalProperties: false
 | |
|                     },
 | |
|                     {
 | |
|                         type: "object",
 | |
|                         properties: {
 | |
|                             initialized: {
 | |
|                                 enum: ["always", "never", "consecutive"]
 | |
|                             },
 | |
|                             uninitialized: {
 | |
|                                 enum: ["always", "never", "consecutive"]
 | |
|                             }
 | |
|                         },
 | |
|                         additionalProperties: false
 | |
|                     }
 | |
|                 ]
 | |
|             }
 | |
|         ],
 | |
| 
 | |
|         messages: {
 | |
|             combineUninitialized: "Combine this with the previous '{{type}}' statement with uninitialized variables.",
 | |
|             combineInitialized: "Combine this with the previous '{{type}}' statement with initialized variables.",
 | |
|             splitUninitialized: "Split uninitialized '{{type}}' declarations into multiple statements.",
 | |
|             splitInitialized: "Split initialized '{{type}}' declarations into multiple statements.",
 | |
|             splitRequires: "Split requires to be separated into a single block.",
 | |
|             combine: "Combine this with the previous '{{type}}' statement.",
 | |
|             split: "Split '{{type}}' declarations into multiple statements."
 | |
|         }
 | |
|     },
 | |
| 
 | |
|     create(context) {
 | |
|         const MODE_ALWAYS = "always";
 | |
|         const MODE_NEVER = "never";
 | |
|         const MODE_CONSECUTIVE = "consecutive";
 | |
|         const mode = context.options[0] || MODE_ALWAYS;
 | |
| 
 | |
|         const options = {};
 | |
| 
 | |
|         if (typeof mode === "string") { // simple options configuration with just a string
 | |
|             options.var = { uninitialized: mode, initialized: mode };
 | |
|             options.let = { uninitialized: mode, initialized: mode };
 | |
|             options.const = { uninitialized: mode, initialized: mode };
 | |
|         } else if (typeof mode === "object") { // options configuration is an object
 | |
|             options.separateRequires = !!mode.separateRequires;
 | |
|             options.var = { uninitialized: mode.var, initialized: mode.var };
 | |
|             options.let = { uninitialized: mode.let, initialized: mode.let };
 | |
|             options.const = { uninitialized: mode.const, initialized: mode.const };
 | |
|             if (Object.prototype.hasOwnProperty.call(mode, "uninitialized")) {
 | |
|                 options.var.uninitialized = mode.uninitialized;
 | |
|                 options.let.uninitialized = mode.uninitialized;
 | |
|                 options.const.uninitialized = mode.uninitialized;
 | |
|             }
 | |
|             if (Object.prototype.hasOwnProperty.call(mode, "initialized")) {
 | |
|                 options.var.initialized = mode.initialized;
 | |
|                 options.let.initialized = mode.initialized;
 | |
|                 options.const.initialized = mode.initialized;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         const sourceCode = context.sourceCode;
 | |
| 
 | |
|         //--------------------------------------------------------------------------
 | |
|         // Helpers
 | |
|         //--------------------------------------------------------------------------
 | |
| 
 | |
|         const functionStack = [];
 | |
|         const blockStack = [];
 | |
| 
 | |
|         /**
 | |
|          * Increments the blockStack counter.
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function startBlock() {
 | |
|             blockStack.push({
 | |
|                 let: { initialized: false, uninitialized: false },
 | |
|                 const: { initialized: false, uninitialized: false }
 | |
|             });
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Increments the functionStack counter.
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function startFunction() {
 | |
|             functionStack.push({ initialized: false, uninitialized: false });
 | |
|             startBlock();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Decrements the blockStack counter.
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function endBlock() {
 | |
|             blockStack.pop();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Decrements the functionStack counter.
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function endFunction() {
 | |
|             functionStack.pop();
 | |
|             endBlock();
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Check if a variable declaration is a require.
 | |
|          * @param {ASTNode} decl variable declaration Node
 | |
|          * @returns {bool} if decl is a require, return true; else return false.
 | |
|          * @private
 | |
|          */
 | |
|         function isRequire(decl) {
 | |
|             return decl.init && decl.init.type === "CallExpression" && decl.init.callee.name === "require";
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Records whether initialized/uninitialized/required variables are defined in current scope.
 | |
|          * @param {string} statementType node.kind, one of: "var", "let", or "const"
 | |
|          * @param {ASTNode[]} declarations List of declarations
 | |
|          * @param {Object} currentScope The scope being investigated
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function recordTypes(statementType, declarations, currentScope) {
 | |
|             for (let i = 0; i < declarations.length; i++) {
 | |
|                 if (declarations[i].init === null) {
 | |
|                     if (options[statementType] && options[statementType].uninitialized === MODE_ALWAYS) {
 | |
|                         currentScope.uninitialized = true;
 | |
|                     }
 | |
|                 } else {
 | |
|                     if (options[statementType] && options[statementType].initialized === MODE_ALWAYS) {
 | |
|                         if (options.separateRequires && isRequire(declarations[i])) {
 | |
|                             currentScope.required = true;
 | |
|                         } else {
 | |
|                             currentScope.initialized = true;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Determines the current scope (function or block)
 | |
|          * @param {string} statementType node.kind, one of: "var", "let", or "const"
 | |
|          * @returns {Object} The scope associated with statementType
 | |
|          */
 | |
|         function getCurrentScope(statementType) {
 | |
|             let currentScope;
 | |
| 
 | |
|             if (statementType === "var") {
 | |
|                 currentScope = functionStack[functionStack.length - 1];
 | |
|             } else if (statementType === "let") {
 | |
|                 currentScope = blockStack[blockStack.length - 1].let;
 | |
|             } else if (statementType === "const") {
 | |
|                 currentScope = blockStack[blockStack.length - 1].const;
 | |
|             }
 | |
|             return currentScope;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Counts the number of initialized and uninitialized declarations in a list of declarations
 | |
|          * @param {ASTNode[]} declarations List of declarations
 | |
|          * @returns {Object} Counts of 'uninitialized' and 'initialized' declarations
 | |
|          * @private
 | |
|          */
 | |
|         function countDeclarations(declarations) {
 | |
|             const counts = { uninitialized: 0, initialized: 0 };
 | |
| 
 | |
|             for (let i = 0; i < declarations.length; i++) {
 | |
|                 if (declarations[i].init === null) {
 | |
|                     counts.uninitialized++;
 | |
|                 } else {
 | |
|                     counts.initialized++;
 | |
|                 }
 | |
|             }
 | |
|             return counts;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Determines if there is more than one var statement in the current scope.
 | |
|          * @param {string} statementType node.kind, one of: "var", "let", or "const"
 | |
|          * @param {ASTNode[]} declarations List of declarations
 | |
|          * @returns {boolean} Returns true if it is the first var declaration, false if not.
 | |
|          * @private
 | |
|          */
 | |
|         function hasOnlyOneStatement(statementType, declarations) {
 | |
| 
 | |
|             const declarationCounts = countDeclarations(declarations);
 | |
|             const currentOptions = options[statementType] || {};
 | |
|             const currentScope = getCurrentScope(statementType);
 | |
|             const hasRequires = declarations.some(isRequire);
 | |
| 
 | |
|             if (currentOptions.uninitialized === MODE_ALWAYS && currentOptions.initialized === MODE_ALWAYS) {
 | |
|                 if (currentScope.uninitialized || currentScope.initialized) {
 | |
|                     if (!hasRequires) {
 | |
|                         return false;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (declarationCounts.uninitialized > 0) {
 | |
|                 if (currentOptions.uninitialized === MODE_ALWAYS && currentScope.uninitialized) {
 | |
|                     return false;
 | |
|                 }
 | |
|             }
 | |
|             if (declarationCounts.initialized > 0) {
 | |
|                 if (currentOptions.initialized === MODE_ALWAYS && currentScope.initialized) {
 | |
|                     if (!hasRequires) {
 | |
|                         return false;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             if (currentScope.required && hasRequires) {
 | |
|                 return false;
 | |
|             }
 | |
|             recordTypes(statementType, declarations, currentScope);
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Fixer to join VariableDeclaration's into a single declaration
 | |
|          * @param {VariableDeclarator[]} declarations The `VariableDeclaration` to join
 | |
|          * @returns {Function} The fixer function
 | |
|          */
 | |
|         function joinDeclarations(declarations) {
 | |
|             const declaration = declarations[0];
 | |
|             const body = Array.isArray(declaration.parent.parent.body) ? declaration.parent.parent.body : [];
 | |
|             const currentIndex = body.findIndex(node => node.range[0] === declaration.parent.range[0]);
 | |
|             const previousNode = body[currentIndex - 1];
 | |
| 
 | |
|             return fixer => {
 | |
|                 const type = sourceCode.getTokenBefore(declaration);
 | |
|                 const prevSemi = sourceCode.getTokenBefore(type);
 | |
|                 const res = [];
 | |
| 
 | |
|                 if (previousNode && previousNode.kind === sourceCode.getText(type)) {
 | |
|                     if (prevSemi.value === ";") {
 | |
|                         res.push(fixer.replaceText(prevSemi, ","));
 | |
|                     } else {
 | |
|                         res.push(fixer.insertTextAfter(prevSemi, ","));
 | |
|                     }
 | |
|                     res.push(fixer.replaceText(type, ""));
 | |
|                 }
 | |
| 
 | |
|                 return res;
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Fixer to split a VariableDeclaration into individual declarations
 | |
|          * @param {VariableDeclaration} declaration The `VariableDeclaration` to split
 | |
|          * @returns {Function|null} The fixer function
 | |
|          */
 | |
|         function splitDeclarations(declaration) {
 | |
|             const { parent } = declaration;
 | |
| 
 | |
|             // don't autofix code such as: if (foo) var x, y;
 | |
|             if (!isInStatementList(parent.type === "ExportNamedDeclaration" ? parent : declaration)) {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             return fixer => declaration.declarations.map(declarator => {
 | |
|                 const tokenAfterDeclarator = sourceCode.getTokenAfter(declarator);
 | |
| 
 | |
|                 if (tokenAfterDeclarator === null) {
 | |
|                     return null;
 | |
|                 }
 | |
| 
 | |
|                 const afterComma = sourceCode.getTokenAfter(tokenAfterDeclarator, { includeComments: true });
 | |
| 
 | |
|                 if (tokenAfterDeclarator.value !== ",") {
 | |
|                     return null;
 | |
|                 }
 | |
| 
 | |
|                 const exportPlacement = declaration.parent.type === "ExportNamedDeclaration" ? "export " : "";
 | |
| 
 | |
|                 /*
 | |
|                  * `var x,y`
 | |
|                  * tokenAfterDeclarator ^^ afterComma
 | |
|                  */
 | |
|                 if (afterComma.range[0] === tokenAfterDeclarator.range[1]) {
 | |
|                     return fixer.replaceText(tokenAfterDeclarator, `; ${exportPlacement}${declaration.kind} `);
 | |
|                 }
 | |
| 
 | |
|                 /*
 | |
|                  * `var x,
 | |
|                  * tokenAfterDeclarator ^
 | |
|                  *      y`
 | |
|                  *      ^ afterComma
 | |
|                  */
 | |
|                 if (
 | |
|                     afterComma.loc.start.line > tokenAfterDeclarator.loc.end.line ||
 | |
|                     afterComma.type === "Line" ||
 | |
|                     afterComma.type === "Block"
 | |
|                 ) {
 | |
|                     let lastComment = afterComma;
 | |
| 
 | |
|                     while (lastComment.type === "Line" || lastComment.type === "Block") {
 | |
|                         lastComment = sourceCode.getTokenAfter(lastComment, { includeComments: true });
 | |
|                     }
 | |
| 
 | |
|                     return fixer.replaceTextRange(
 | |
|                         [tokenAfterDeclarator.range[0], lastComment.range[0]],
 | |
|                         `;${sourceCode.text.slice(tokenAfterDeclarator.range[1], lastComment.range[0])}${exportPlacement}${declaration.kind} `
 | |
|                     );
 | |
|                 }
 | |
| 
 | |
|                 return fixer.replaceText(tokenAfterDeclarator, `; ${exportPlacement}${declaration.kind}`);
 | |
|             }).filter(x => x);
 | |
|         }
 | |
| 
 | |
|         /**
 | |
|          * Checks a given VariableDeclaration node for errors.
 | |
|          * @param {ASTNode} node The VariableDeclaration node to check
 | |
|          * @returns {void}
 | |
|          * @private
 | |
|          */
 | |
|         function checkVariableDeclaration(node) {
 | |
|             const parent = node.parent;
 | |
|             const type = node.kind;
 | |
| 
 | |
|             if (!options[type]) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             const declarations = node.declarations;
 | |
|             const declarationCounts = countDeclarations(declarations);
 | |
|             const mixedRequires = declarations.some(isRequire) && !declarations.every(isRequire);
 | |
| 
 | |
|             if (options[type].initialized === MODE_ALWAYS) {
 | |
|                 if (options.separateRequires && mixedRequires) {
 | |
|                     context.report({
 | |
|                         node,
 | |
|                         messageId: "splitRequires"
 | |
|                     });
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // consecutive
 | |
|             const nodeIndex = (parent.body && parent.body.length > 0 && parent.body.indexOf(node)) || 0;
 | |
| 
 | |
|             if (nodeIndex > 0) {
 | |
|                 const previousNode = parent.body[nodeIndex - 1];
 | |
|                 const isPreviousNodeDeclaration = previousNode.type === "VariableDeclaration";
 | |
|                 const declarationsWithPrevious = declarations.concat(previousNode.declarations || []);
 | |
| 
 | |
|                 if (
 | |
|                     isPreviousNodeDeclaration &&
 | |
|                     previousNode.kind === type &&
 | |
|                     !(declarationsWithPrevious.some(isRequire) && !declarationsWithPrevious.every(isRequire))
 | |
|                 ) {
 | |
|                     const previousDeclCounts = countDeclarations(previousNode.declarations);
 | |
| 
 | |
|                     if (options[type].initialized === MODE_CONSECUTIVE && options[type].uninitialized === MODE_CONSECUTIVE) {
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "combine",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: joinDeclarations(declarations)
 | |
|                         });
 | |
|                     } else if (options[type].initialized === MODE_CONSECUTIVE && declarationCounts.initialized > 0 && previousDeclCounts.initialized > 0) {
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "combineInitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: joinDeclarations(declarations)
 | |
|                         });
 | |
|                     } else if (options[type].uninitialized === MODE_CONSECUTIVE &&
 | |
|                             declarationCounts.uninitialized > 0 &&
 | |
|                             previousDeclCounts.uninitialized > 0) {
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "combineUninitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: joinDeclarations(declarations)
 | |
|                         });
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // always
 | |
|             if (!hasOnlyOneStatement(type, declarations)) {
 | |
|                 if (options[type].initialized === MODE_ALWAYS && options[type].uninitialized === MODE_ALWAYS) {
 | |
|                     context.report({
 | |
|                         node,
 | |
|                         messageId: "combine",
 | |
|                         data: {
 | |
|                             type
 | |
|                         },
 | |
|                         fix: joinDeclarations(declarations)
 | |
|                     });
 | |
|                 } else {
 | |
|                     if (options[type].initialized === MODE_ALWAYS && declarationCounts.initialized > 0) {
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "combineInitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: joinDeclarations(declarations)
 | |
|                         });
 | |
|                     }
 | |
|                     if (options[type].uninitialized === MODE_ALWAYS && declarationCounts.uninitialized > 0) {
 | |
|                         if (node.parent.left === node && (node.parent.type === "ForInStatement" || node.parent.type === "ForOfStatement")) {
 | |
|                             return;
 | |
|                         }
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "combineUninitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: joinDeclarations(declarations)
 | |
|                         });
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             // never
 | |
|             if (parent.type !== "ForStatement" || parent.init !== node) {
 | |
|                 const totalDeclarations = declarationCounts.uninitialized + declarationCounts.initialized;
 | |
| 
 | |
|                 if (totalDeclarations > 1) {
 | |
|                     if (options[type].initialized === MODE_NEVER && options[type].uninitialized === MODE_NEVER) {
 | |
| 
 | |
|                         // both initialized and uninitialized
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "split",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: splitDeclarations(node)
 | |
|                         });
 | |
|                     } else if (options[type].initialized === MODE_NEVER && declarationCounts.initialized > 0) {
 | |
| 
 | |
|                         // initialized
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "splitInitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: splitDeclarations(node)
 | |
|                         });
 | |
|                     } else if (options[type].uninitialized === MODE_NEVER && declarationCounts.uninitialized > 0) {
 | |
| 
 | |
|                         // uninitialized
 | |
|                         context.report({
 | |
|                             node,
 | |
|                             messageId: "splitUninitialized",
 | |
|                             data: {
 | |
|                                 type
 | |
|                             },
 | |
|                             fix: splitDeclarations(node)
 | |
|                         });
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         //--------------------------------------------------------------------------
 | |
|         // Public API
 | |
|         //--------------------------------------------------------------------------
 | |
| 
 | |
|         return {
 | |
|             Program: startFunction,
 | |
|             FunctionDeclaration: startFunction,
 | |
|             FunctionExpression: startFunction,
 | |
|             ArrowFunctionExpression: startFunction,
 | |
|             StaticBlock: startFunction, // StaticBlock creates a new scope for `var` variables
 | |
| 
 | |
|             BlockStatement: startBlock,
 | |
|             ForStatement: startBlock,
 | |
|             ForInStatement: startBlock,
 | |
|             ForOfStatement: startBlock,
 | |
|             SwitchStatement: startBlock,
 | |
|             VariableDeclaration: checkVariableDeclaration,
 | |
|             "ForStatement:exit": endBlock,
 | |
|             "ForOfStatement:exit": endBlock,
 | |
|             "ForInStatement:exit": endBlock,
 | |
|             "SwitchStatement:exit": endBlock,
 | |
|             "BlockStatement:exit": endBlock,
 | |
| 
 | |
|             "Program:exit": endFunction,
 | |
|             "FunctionDeclaration:exit": endFunction,
 | |
|             "FunctionExpression:exit": endFunction,
 | |
|             "ArrowFunctionExpression:exit": endFunction,
 | |
|             "StaticBlock:exit": endFunction
 | |
|         };
 | |
| 
 | |
|     }
 | |
| };
 |