/**
							 | 
						|
								 * @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
							 | 
						|
								        };
							 | 
						|
								    }
							 | 
						|
								};
							 |