| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107 | /** * @fileoverview Rule to disallow unused labels. * @author Toru Nagashima */"use strict";//------------------------------------------------------------------------------// Rule Definition//------------------------------------------------------------------------------module.exports = {    meta: {        docs: {            description: "disallow unused labels",            category: "Best Practices",            recommended: true,            url: "https://eslint.org/docs/rules/no-unused-labels"        },        schema: [],        fixable: "code"    },    create(context) {        const sourceCode = context.getSourceCode();        let scopeInfo = null;        /**         * Adds a scope info to the stack.         *         * @param {ASTNode} node - A node to add. This is a LabeledStatement.         * @returns {void}         */        function enterLabeledScope(node) {            scopeInfo = {                label: node.label.name,                used: false,                upper: scopeInfo            };        }        /**         * Removes the top of the stack.         * At the same time, this reports the label if it's never used.         *         * @param {ASTNode} node - A node to report. This is a LabeledStatement.         * @returns {void}         */        function exitLabeledScope(node) {            if (!scopeInfo.used) {                context.report({                    node: node.label,                    message: "'{{name}}:' is defined but never used.",                    data: node.label,                    fix(fixer) {                        /*                         * Only perform a fix if there are no comments between the label and the body. This will be the case                         * when there is exactly one token/comment (the ":") between the label and the body.                         */                        if (sourceCode.getTokenAfter(node.label, { includeComments: true }) ===                                sourceCode.getTokenBefore(node.body, { includeComments: true })) {                            return fixer.removeRange([node.range[0], node.body.range[0]]);                        }                        return null;                    }                });            }            scopeInfo = scopeInfo.upper;        }        /**         * Marks the label of a given node as used.         *         * @param {ASTNode} node - A node to mark. This is a BreakStatement or         *      ContinueStatement.         * @returns {void}         */        function markAsUsed(node) {            if (!node.label) {                return;            }            const label = node.label.name;            let info = scopeInfo;            while (info) {                if (info.label === label) {                    info.used = true;                    break;                }                info = info.upper;            }        }        return {            LabeledStatement: enterLabeledScope,            "LabeledStatement:exit": exitLabeledScope,            BreakStatement: markAsUsed,            ContinueStatement: markAsUsed        };    }};
 |