.
This commit is contained in:
		
							
								
								
									
										178
									
								
								qwen/nodejs/node_modules/eslint/lib/rules/prefer-named-capture-group.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										178
									
								
								qwen/nodejs/node_modules/eslint/lib/rules/prefer-named-capture-group.js
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							@@ -0,0 +1,178 @@
 | 
			
		||||
/**
 | 
			
		||||
 * @fileoverview Rule to enforce requiring named capture groups in regular expression.
 | 
			
		||||
 * @author Pig Fang <https://github.com/g-plane>
 | 
			
		||||
 */
 | 
			
		||||
 | 
			
		||||
"use strict";
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// Requirements
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
const {
 | 
			
		||||
    CALL,
 | 
			
		||||
    CONSTRUCT,
 | 
			
		||||
    ReferenceTracker,
 | 
			
		||||
    getStringIfConstant
 | 
			
		||||
} = require("@eslint-community/eslint-utils");
 | 
			
		||||
const regexpp = require("@eslint-community/regexpp");
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// Helpers
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
const parser = new regexpp.RegExpParser();
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * Creates fixer suggestions for the regex, if statically determinable.
 | 
			
		||||
 * @param {number} groupStart Starting index of the regex group.
 | 
			
		||||
 * @param {string} pattern The regular expression pattern to be checked.
 | 
			
		||||
 * @param {string} rawText Source text of the regexNode.
 | 
			
		||||
 * @param {ASTNode} regexNode AST node which contains the regular expression.
 | 
			
		||||
 * @returns {Array<SuggestionResult>} Fixer suggestions for the regex, if statically determinable.
 | 
			
		||||
 */
 | 
			
		||||
function suggestIfPossible(groupStart, pattern, rawText, regexNode) {
 | 
			
		||||
    switch (regexNode.type) {
 | 
			
		||||
        case "Literal":
 | 
			
		||||
            if (typeof regexNode.value === "string" && rawText.includes("\\")) {
 | 
			
		||||
                return null;
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
        case "TemplateLiteral":
 | 
			
		||||
            if (regexNode.expressions.length || rawText.slice(1, -1) !== pattern) {
 | 
			
		||||
                return null;
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
        default:
 | 
			
		||||
            return null;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    const start = regexNode.range[0] + groupStart + 2;
 | 
			
		||||
 | 
			
		||||
    return [
 | 
			
		||||
        {
 | 
			
		||||
            fix(fixer) {
 | 
			
		||||
                const existingTemps = pattern.match(/temp\d+/gu) || [];
 | 
			
		||||
                const highestTempCount = existingTemps.reduce(
 | 
			
		||||
                    (previous, next) =>
 | 
			
		||||
                        Math.max(previous, Number(next.slice("temp".length))),
 | 
			
		||||
                    0
 | 
			
		||||
                );
 | 
			
		||||
 | 
			
		||||
                return fixer.insertTextBeforeRange(
 | 
			
		||||
                    [start, start],
 | 
			
		||||
                    `?<temp${highestTempCount + 1}>`
 | 
			
		||||
                );
 | 
			
		||||
            },
 | 
			
		||||
            messageId: "addGroupName"
 | 
			
		||||
        },
 | 
			
		||||
        {
 | 
			
		||||
            fix(fixer) {
 | 
			
		||||
                return fixer.insertTextBeforeRange(
 | 
			
		||||
                    [start, start],
 | 
			
		||||
                    "?:"
 | 
			
		||||
                );
 | 
			
		||||
            },
 | 
			
		||||
            messageId: "addNonCapture"
 | 
			
		||||
        }
 | 
			
		||||
    ];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// Rule Definition
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
/** @type {import('../shared/types').Rule} */
 | 
			
		||||
module.exports = {
 | 
			
		||||
    meta: {
 | 
			
		||||
        type: "suggestion",
 | 
			
		||||
 | 
			
		||||
        docs: {
 | 
			
		||||
            description: "Enforce using named capture group in regular expression",
 | 
			
		||||
            recommended: false,
 | 
			
		||||
            url: "https://eslint.org/docs/latest/rules/prefer-named-capture-group"
 | 
			
		||||
        },
 | 
			
		||||
 | 
			
		||||
        hasSuggestions: true,
 | 
			
		||||
 | 
			
		||||
        schema: [],
 | 
			
		||||
 | 
			
		||||
        messages: {
 | 
			
		||||
            addGroupName: "Add name to capture group.",
 | 
			
		||||
            addNonCapture: "Convert group to non-capturing.",
 | 
			
		||||
            required: "Capture group '{{group}}' should be converted to a named or non-capturing group."
 | 
			
		||||
        }
 | 
			
		||||
    },
 | 
			
		||||
 | 
			
		||||
    create(context) {
 | 
			
		||||
        const sourceCode = context.sourceCode;
 | 
			
		||||
 | 
			
		||||
        /**
 | 
			
		||||
         * Function to check regular expression.
 | 
			
		||||
         * @param {string} pattern The regular expression pattern to be checked.
 | 
			
		||||
         * @param {ASTNode} node AST node which contains the regular expression or a call/new expression.
 | 
			
		||||
         * @param {ASTNode} regexNode AST node which contains the regular expression.
 | 
			
		||||
         * @param {string|null} flags The regular expression flags to be checked.
 | 
			
		||||
         * @returns {void}
 | 
			
		||||
         */
 | 
			
		||||
        function checkRegex(pattern, node, regexNode, flags) {
 | 
			
		||||
            let ast;
 | 
			
		||||
 | 
			
		||||
            try {
 | 
			
		||||
                ast = parser.parsePattern(pattern, 0, pattern.length, {
 | 
			
		||||
                    unicode: Boolean(flags && flags.includes("u")),
 | 
			
		||||
                    unicodeSets: Boolean(flags && flags.includes("v"))
 | 
			
		||||
                });
 | 
			
		||||
            } catch {
 | 
			
		||||
 | 
			
		||||
                // ignore regex syntax errors
 | 
			
		||||
                return;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            regexpp.visitRegExpAST(ast, {
 | 
			
		||||
                onCapturingGroupEnter(group) {
 | 
			
		||||
                    if (!group.name) {
 | 
			
		||||
                        const rawText = sourceCode.getText(regexNode);
 | 
			
		||||
                        const suggest = suggestIfPossible(group.start, pattern, rawText, regexNode);
 | 
			
		||||
 | 
			
		||||
                        context.report({
 | 
			
		||||
                            node,
 | 
			
		||||
                            messageId: "required",
 | 
			
		||||
                            data: {
 | 
			
		||||
                                group: group.raw
 | 
			
		||||
                            },
 | 
			
		||||
                            suggest
 | 
			
		||||
                        });
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        return {
 | 
			
		||||
            Literal(node) {
 | 
			
		||||
                if (node.regex) {
 | 
			
		||||
                    checkRegex(node.regex.pattern, node, node, node.regex.flags);
 | 
			
		||||
                }
 | 
			
		||||
            },
 | 
			
		||||
            Program(node) {
 | 
			
		||||
                const scope = sourceCode.getScope(node);
 | 
			
		||||
                const tracker = new ReferenceTracker(scope);
 | 
			
		||||
                const traceMap = {
 | 
			
		||||
                    RegExp: {
 | 
			
		||||
                        [CALL]: true,
 | 
			
		||||
                        [CONSTRUCT]: true
 | 
			
		||||
                    }
 | 
			
		||||
                };
 | 
			
		||||
 | 
			
		||||
                for (const { node: refNode } of tracker.iterateGlobalReferences(traceMap)) {
 | 
			
		||||
                    const regex = getStringIfConstant(refNode.arguments[0]);
 | 
			
		||||
                    const flags = getStringIfConstant(refNode.arguments[1]);
 | 
			
		||||
 | 
			
		||||
                    if (regex) {
 | 
			
		||||
                        checkRegex(regex, refNode, refNode.arguments[0], flags);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
};
 | 
			
		||||
		Reference in New Issue
	
	Block a user