UNPKG

concurrently

Version:
108 lines (107 loc) 4.53 kB
"use strict"; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.ExpandWildcard = void 0; const fs_1 = __importDefault(require("fs")); const lodash_1 = __importDefault(require("lodash")); const jsonc_1 = __importDefault(require("../jsonc")); // Matches a negative filter surrounded by '(!' and ')'. const OMISSION = /\(!([^)]+)\)/; /** * Finds wildcards in 'npm/yarn/pnpm/bun run', 'node --run' and 'deno task' * commands and replaces them with all matching scripts in the NodeJS and Deno * configuration files of the current directory. */ class ExpandWildcard { readDeno; readPackage; static readDeno() { try { let json = '{}'; if (fs_1.default.existsSync('deno.json')) { json = fs_1.default.readFileSync('deno.json', { encoding: 'utf-8' }); } else if (fs_1.default.existsSync('deno.jsonc')) { json = fs_1.default.readFileSync('deno.jsonc', { encoding: 'utf-8' }); } return jsonc_1.default.parse(json); } catch (e) { return {}; } } static readPackage() { try { const json = fs_1.default.readFileSync('package.json', { encoding: 'utf-8' }); return JSON.parse(json); } catch (e) { return {}; } } packageScripts; denoTasks; constructor(readDeno = ExpandWildcard.readDeno, readPackage = ExpandWildcard.readPackage) { this.readDeno = readDeno; this.readPackage = readPackage; } relevantScripts(command) { if (!this.packageScripts) { this.packageScripts = Object.keys(this.readPackage().scripts || {}); } if (command === 'deno task') { if (!this.denoTasks) { // If Deno tries to run a task that doesn't exist, // it can fall back to running a script with the same name. // Therefore, the actual list of tasks is the union of the tasks and scripts. this.denoTasks = [ ...Object.keys(this.readDeno().tasks || {}), ...this.packageScripts, ]; } return this.denoTasks; } return this.packageScripts; } parse(commandInfo) { // We expect one of the following patterns: // - <npm|yarn|pnpm|bun> run <script> [args] // - node --run <script> [args] // - deno task <script> [args] const [, command, scriptGlob, args] = /((?:npm|yarn|pnpm|bun) (?:run)|node --run|deno task) (\S+)([^&]*)/.exec(commandInfo.command) || []; const wildcardPosition = (scriptGlob || '').indexOf('*'); // If the regex didn't match an npm script, or it has no wildcard, // then we have nothing to do here if (wildcardPosition === -1) { return commandInfo; } const [, omission] = OMISSION.exec(scriptGlob) || []; const scriptGlobSansOmission = scriptGlob.replace(OMISSION, ''); const preWildcard = lodash_1.default.escapeRegExp(scriptGlobSansOmission.slice(0, wildcardPosition)); const postWildcard = lodash_1.default.escapeRegExp(scriptGlobSansOmission.slice(wildcardPosition + 1)); const wildcardRegex = new RegExp(`^${preWildcard}(.*?)${postWildcard}$`); // If 'commandInfo.name' doesn't match 'scriptGlob', this means a custom name // has been specified and thus becomes the prefix (as described in the README). const prefix = commandInfo.name !== scriptGlob ? commandInfo.name : ''; return this.relevantScripts(command) .map((script) => { if (omission && RegExp(omission).test(script)) { return; } const [, match] = wildcardRegex.exec(script) || []; if (match !== undefined) { return { ...commandInfo, command: `${command} ${script}${args}`, // Will use an empty command name if no prefix has been specified and // the wildcard match is empty, e.g. if `npm:watch-*` matches `npm run watch-`. name: prefix + match, }; } }) .filter((commandInfo) => !!commandInfo); } } exports.ExpandWildcard = ExpandWildcard;