|                                                                                                                                               |  | /** * @fileoverview Rule to disallow unused labels. * @author Toru Nagashima */
"use strict";
//------------------------------------------------------------------------------
// Requirements
//------------------------------------------------------------------------------
const astUtils = require("./utils/ast-utils");
//------------------------------------------------------------------------------
// Rule Definition
//------------------------------------------------------------------------------
/** @type {import('../shared/types').Rule} */module.exports = {    meta: {        type: "suggestion",
        docs: {            description: "Disallow unused labels",            recommended: true,            url: "https://eslint.org/docs/latest/rules/no-unused-labels"        },
        schema: [],
        fixable: "code",
        messages: {            unused: "'{{name}}:' is defined but never used."        }    },
    create(context) {        const sourceCode = context.sourceCode;        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            };        }
        /**         * Checks if a `LabeledStatement` node is fixable.         * For a node to be fixable, there must be no comments between the label and the body.         * Furthermore, is must be possible to remove the label without turning the body statement into a         * directive after other fixes are applied.         * @param {ASTNode} node The node to evaluate.         * @returns {boolean} Whether or not the node is fixable.         */        function isFixable(node) {
            /*             * 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 false;            }
            // Looking for the node's deepest ancestor which is not a `LabeledStatement`.
            let ancestor = node.parent;
            while (ancestor.type === "LabeledStatement") {                ancestor = ancestor.parent;            }
            if (ancestor.type === "Program" ||                (ancestor.type === "BlockStatement" && astUtils.isFunction(ancestor.parent))) {                const { body } = node;
                if (body.type === "ExpressionStatement" &&                    ((body.expression.type === "Literal" && typeof body.expression.value === "string") ||                    astUtils.isStaticTemplateLiteral(body.expression))) {                    return false; // potential directive
                }            }            return true;        }
        /**         * 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,                    messageId: "unused",                    data: node.label,                    fix: isFixable(node) ? fixer => fixer.removeRange([node.range[0], node.body.range[0]]) : 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        };    }};
 |