/** * @fileoverview Rule to check for "block scoped" variables by binding context * @author Matt DuVall */ "use strict"; //------------------------------------------------------------------------------ // Rule Definition //------------------------------------------------------------------------------ module.exports = { meta: { type: "suggestion", docs: { description: "enforce the use of variables within the scope they are defined", category: "Best Practices", recommended: false, url: "https://eslint.org/docs/rules/block-scoped-var" }, schema: [], messages: { outOfScope: "'{{name}}' used outside of binding context." } }, create(context) { let stack = []; /** * Makes a block scope. * @param {ASTNode} node A node of a scope. * @returns {void} */ function enterScope(node) { stack.push(node.range); } /** * Pops the last block scope. * @returns {void} */ function exitScope() { stack.pop(); } /** * Reports a given reference. * @param {eslint-scope.Reference} reference A reference to report. * @returns {void} */ function report(reference) { const identifier = reference.identifier; context.report({ node: identifier, messageId: "outOfScope", data: { name: identifier.name } }); } /** * Finds and reports references which are outside of valid scopes. * @param {ASTNode} node A node to get variables. * @returns {void} */ function checkForVariables(node) { if (node.kind !== "var") { return; } // Defines a predicate to check whether or not a given reference is outside of valid scope. const scopeRange = stack[stack.length - 1]; /** * Check if a reference is out of scope * @param {ASTNode} reference node to examine * @returns {boolean} True is its outside the scope * @private */ function isOutsideOfScope(reference) { const idRange = reference.identifier.range; return idRange[0] < scopeRange[0] || idRange[1] > scopeRange[1]; } // Gets declared variables, and checks its references. const variables = context.getDeclaredVariables(node); for (let i = 0; i < variables.length; ++i) { // Reports. variables[i] .references .filter(isOutsideOfScope) .forEach(report); } } return { Program(node) { stack = [node.range]; }, // Manages scopes. BlockStatement: enterScope, "BlockStatement:exit": exitScope, ForStatement: enterScope, "ForStatement:exit": exitScope, ForInStatement: enterScope, "ForInStatement:exit": exitScope, ForOfStatement: enterScope, "ForOfStatement:exit": exitScope, SwitchStatement: enterScope, "SwitchStatement:exit": exitScope, CatchClause: enterScope, "CatchClause:exit": exitScope, // Finds and reports references which are outside of valid scope. VariableDeclaration: checkForVariables }; } };