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
 | 
						|
        };
 | 
						|
 | 
						|
    }
 | 
						|
};
 |