| const EventEmitter = require('node:events').EventEmitter; |
| const childProcess = require('node:child_process'); |
| const path = require('node:path'); |
| const fs = require('node:fs'); |
| const process = require('node:process'); |
|
|
| const { Argument, humanReadableArgName } = require('./argument.js'); |
| const { CommanderError } = require('./error.js'); |
| const { Help, stripColor } = require('./help.js'); |
| const { Option, DualOptions } = require('./option.js'); |
| const { suggestSimilar } = require('./suggestSimilar'); |
|
|
| class Command extends EventEmitter { |
| |
| |
| |
| |
| |
|
|
| constructor(name) { |
| super(); |
| |
| this.commands = []; |
| |
| this.options = []; |
| this.parent = null; |
| this._allowUnknownOption = false; |
| this._allowExcessArguments = false; |
| |
| this.registeredArguments = []; |
| this._args = this.registeredArguments; |
| |
| this.args = []; |
| this.rawArgs = []; |
| this.processedArgs = []; |
| this._scriptPath = null; |
| this._name = name || ''; |
| this._optionValues = {}; |
| this._optionValueSources = {}; |
| this._storeOptionsAsProperties = false; |
| this._actionHandler = null; |
| this._executableHandler = false; |
| this._executableFile = null; |
| this._executableDir = null; |
| this._defaultCommandName = null; |
| this._exitCallback = null; |
| this._aliases = []; |
| this._combineFlagAndOptionalValue = true; |
| this._description = ''; |
| this._summary = ''; |
| this._argsDescription = undefined; |
| this._enablePositionalOptions = false; |
| this._passThroughOptions = false; |
| this._lifeCycleHooks = {}; |
| |
| this._showHelpAfterError = false; |
| this._showSuggestionAfterError = true; |
| this._savedState = null; |
|
|
| |
| this._outputConfiguration = { |
| writeOut: (str) => process.stdout.write(str), |
| writeErr: (str) => process.stderr.write(str), |
| outputError: (str, write) => write(str), |
| getOutHelpWidth: () => |
| process.stdout.isTTY ? process.stdout.columns : undefined, |
| getErrHelpWidth: () => |
| process.stderr.isTTY ? process.stderr.columns : undefined, |
| getOutHasColors: () => |
| useColor() ?? (process.stdout.isTTY && process.stdout.hasColors?.()), |
| getErrHasColors: () => |
| useColor() ?? (process.stderr.isTTY && process.stderr.hasColors?.()), |
| stripColor: (str) => stripColor(str), |
| }; |
|
|
| this._hidden = false; |
| |
| this._helpOption = undefined; |
| this._addImplicitHelpCommand = undefined; |
| |
| this._helpCommand = undefined; |
| this._helpConfiguration = {}; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| copyInheritedSettings(sourceCommand) { |
| this._outputConfiguration = sourceCommand._outputConfiguration; |
| this._helpOption = sourceCommand._helpOption; |
| this._helpCommand = sourceCommand._helpCommand; |
| this._helpConfiguration = sourceCommand._helpConfiguration; |
| this._exitCallback = sourceCommand._exitCallback; |
| this._storeOptionsAsProperties = sourceCommand._storeOptionsAsProperties; |
| this._combineFlagAndOptionalValue = |
| sourceCommand._combineFlagAndOptionalValue; |
| this._allowExcessArguments = sourceCommand._allowExcessArguments; |
| this._enablePositionalOptions = sourceCommand._enablePositionalOptions; |
| this._showHelpAfterError = sourceCommand._showHelpAfterError; |
| this._showSuggestionAfterError = sourceCommand._showSuggestionAfterError; |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
|
|
| _getCommandAndAncestors() { |
| const result = []; |
| |
| for (let command = this; command; command = command.parent) { |
| result.push(command); |
| } |
| return result; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| command(nameAndArgs, actionOptsOrExecDesc, execOpts) { |
| let desc = actionOptsOrExecDesc; |
| let opts = execOpts; |
| if (typeof desc === 'object' && desc !== null) { |
| opts = desc; |
| desc = null; |
| } |
| opts = opts || {}; |
| const [, name, args] = nameAndArgs.match(/([^ ]+) *(.*)/); |
|
|
| const cmd = this.createCommand(name); |
| if (desc) { |
| cmd.description(desc); |
| cmd._executableHandler = true; |
| } |
| if (opts.isDefault) this._defaultCommandName = cmd._name; |
| cmd._hidden = !!(opts.noHelp || opts.hidden); |
| cmd._executableFile = opts.executableFile || null; |
| if (args) cmd.arguments(args); |
| this._registerCommand(cmd); |
| cmd.parent = this; |
| cmd.copyInheritedSettings(this); |
|
|
| if (desc) return this; |
| return cmd; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| createCommand(name) { |
| return new Command(name); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| createHelp() { |
| return Object.assign(new Help(), this.configureHelp()); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| configureHelp(configuration) { |
| if (configuration === undefined) return this._helpConfiguration; |
|
|
| this._helpConfiguration = configuration; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| configureOutput(configuration) { |
| if (configuration === undefined) return this._outputConfiguration; |
|
|
| Object.assign(this._outputConfiguration, configuration); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| showHelpAfterError(displayHelp = true) { |
| if (typeof displayHelp !== 'string') displayHelp = !!displayHelp; |
| this._showHelpAfterError = displayHelp; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| showSuggestionAfterError(displaySuggestion = true) { |
| this._showSuggestionAfterError = !!displaySuggestion; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| addCommand(cmd, opts) { |
| if (!cmd._name) { |
| throw new Error(`Command passed to .addCommand() must have a name |
| - specify the name in Command constructor or using .name()`); |
| } |
|
|
| opts = opts || {}; |
| if (opts.isDefault) this._defaultCommandName = cmd._name; |
| if (opts.noHelp || opts.hidden) cmd._hidden = true; |
|
|
| this._registerCommand(cmd); |
| cmd.parent = this; |
| cmd._checkForBrokenPassThrough(); |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| createArgument(name, description) { |
| return new Argument(name, description); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| argument(name, description, fn, defaultValue) { |
| const argument = this.createArgument(name, description); |
| if (typeof fn === 'function') { |
| argument.default(defaultValue).argParser(fn); |
| } else { |
| argument.default(fn); |
| } |
| this.addArgument(argument); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| arguments(names) { |
| names |
| .trim() |
| .split(/ +/) |
| .forEach((detail) => { |
| this.argument(detail); |
| }); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| addArgument(argument) { |
| const previousArgument = this.registeredArguments.slice(-1)[0]; |
| if (previousArgument && previousArgument.variadic) { |
| throw new Error( |
| `only the last argument can be variadic '${previousArgument.name()}'`, |
| ); |
| } |
| if ( |
| argument.required && |
| argument.defaultValue !== undefined && |
| argument.parseArg === undefined |
| ) { |
| throw new Error( |
| `a default value for a required argument is never used: '${argument.name()}'`, |
| ); |
| } |
| this.registeredArguments.push(argument); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| helpCommand(enableOrNameAndArgs, description) { |
| if (typeof enableOrNameAndArgs === 'boolean') { |
| this._addImplicitHelpCommand = enableOrNameAndArgs; |
| return this; |
| } |
|
|
| enableOrNameAndArgs = enableOrNameAndArgs ?? 'help [command]'; |
| const [, helpName, helpArgs] = enableOrNameAndArgs.match(/([^ ]+) *(.*)/); |
| const helpDescription = description ?? 'display help for command'; |
|
|
| const helpCommand = this.createCommand(helpName); |
| helpCommand.helpOption(false); |
| if (helpArgs) helpCommand.arguments(helpArgs); |
| if (helpDescription) helpCommand.description(helpDescription); |
|
|
| this._addImplicitHelpCommand = true; |
| this._helpCommand = helpCommand; |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| addHelpCommand(helpCommand, deprecatedDescription) { |
| |
| |
| if (typeof helpCommand !== 'object') { |
| this.helpCommand(helpCommand, deprecatedDescription); |
| return this; |
| } |
|
|
| this._addImplicitHelpCommand = true; |
| this._helpCommand = helpCommand; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| _getHelpCommand() { |
| const hasImplicitHelpCommand = |
| this._addImplicitHelpCommand ?? |
| (this.commands.length && |
| !this._actionHandler && |
| !this._findCommand('help')); |
|
|
| if (hasImplicitHelpCommand) { |
| if (this._helpCommand === undefined) { |
| this.helpCommand(undefined, undefined); |
| } |
| return this._helpCommand; |
| } |
| return null; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| hook(event, listener) { |
| const allowedValues = ['preSubcommand', 'preAction', 'postAction']; |
| if (!allowedValues.includes(event)) { |
| throw new Error(`Unexpected value for event passed to hook : '${event}'. |
| Expecting one of '${allowedValues.join("', '")}'`); |
| } |
| if (this._lifeCycleHooks[event]) { |
| this._lifeCycleHooks[event].push(listener); |
| } else { |
| this._lifeCycleHooks[event] = [listener]; |
| } |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| exitOverride(fn) { |
| if (fn) { |
| this._exitCallback = fn; |
| } else { |
| this._exitCallback = (err) => { |
| if (err.code !== 'commander.executeSubCommandAsync') { |
| throw err; |
| } else { |
| |
| } |
| }; |
| } |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| _exit(exitCode, code, message) { |
| if (this._exitCallback) { |
| this._exitCallback(new CommanderError(exitCode, code, message)); |
| |
| } |
| process.exit(exitCode); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| action(fn) { |
| const listener = (args) => { |
| |
| const expectedArgsCount = this.registeredArguments.length; |
| const actionArgs = args.slice(0, expectedArgsCount); |
| if (this._storeOptionsAsProperties) { |
| actionArgs[expectedArgsCount] = this; |
| } else { |
| actionArgs[expectedArgsCount] = this.opts(); |
| } |
| actionArgs.push(this); |
|
|
| return fn.apply(this, actionArgs); |
| }; |
| this._actionHandler = listener; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| createOption(flags, description) { |
| return new Option(flags, description); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| _callParseArg(target, value, previous, invalidArgumentMessage) { |
| try { |
| return target.parseArg(value, previous); |
| } catch (err) { |
| if (err.code === 'commander.invalidArgument') { |
| const message = `${invalidArgumentMessage} ${err.message}`; |
| this.error(message, { exitCode: err.exitCode, code: err.code }); |
| } |
| throw err; |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| _registerOption(option) { |
| const matchingOption = |
| (option.short && this._findOption(option.short)) || |
| (option.long && this._findOption(option.long)); |
| if (matchingOption) { |
| const matchingFlag = |
| option.long && this._findOption(option.long) |
| ? option.long |
| : option.short; |
| throw new Error(`Cannot add option '${option.flags}'${this._name && ` to command '${this._name}'`} due to conflicting flag '${matchingFlag}' |
| - already used by option '${matchingOption.flags}'`); |
| } |
|
|
| this.options.push(option); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| _registerCommand(command) { |
| const knownBy = (cmd) => { |
| return [cmd.name()].concat(cmd.aliases()); |
| }; |
|
|
| const alreadyUsed = knownBy(command).find((name) => |
| this._findCommand(name), |
| ); |
| if (alreadyUsed) { |
| const existingCmd = knownBy(this._findCommand(alreadyUsed)).join('|'); |
| const newCmd = knownBy(command).join('|'); |
| throw new Error( |
| `cannot add command '${newCmd}' as already have command '${existingCmd}'`, |
| ); |
| } |
|
|
| this.commands.push(command); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| addOption(option) { |
| this._registerOption(option); |
|
|
| const oname = option.name(); |
| const name = option.attributeName(); |
|
|
| |
| if (option.negate) { |
| |
| const positiveLongFlag = option.long.replace(/^--no-/, '--'); |
| if (!this._findOption(positiveLongFlag)) { |
| this.setOptionValueWithSource( |
| name, |
| option.defaultValue === undefined ? true : option.defaultValue, |
| 'default', |
| ); |
| } |
| } else if (option.defaultValue !== undefined) { |
| this.setOptionValueWithSource(name, option.defaultValue, 'default'); |
| } |
|
|
| |
| const handleOptionValue = (val, invalidValueMessage, valueSource) => { |
| |
| |
| if (val == null && option.presetArg !== undefined) { |
| val = option.presetArg; |
| } |
|
|
| |
| const oldValue = this.getOptionValue(name); |
| if (val !== null && option.parseArg) { |
| val = this._callParseArg(option, val, oldValue, invalidValueMessage); |
| } else if (val !== null && option.variadic) { |
| val = option._concatValue(val, oldValue); |
| } |
|
|
| |
| if (val == null) { |
| if (option.negate) { |
| val = false; |
| } else if (option.isBoolean() || option.optional) { |
| val = true; |
| } else { |
| val = ''; |
| } |
| } |
| this.setOptionValueWithSource(name, val, valueSource); |
| }; |
|
|
| this.on('option:' + oname, (val) => { |
| const invalidValueMessage = `error: option '${option.flags}' argument '${val}' is invalid.`; |
| handleOptionValue(val, invalidValueMessage, 'cli'); |
| }); |
|
|
| if (option.envVar) { |
| this.on('optionEnv:' + oname, (val) => { |
| const invalidValueMessage = `error: option '${option.flags}' value '${val}' from env '${option.envVar}' is invalid.`; |
| handleOptionValue(val, invalidValueMessage, 'env'); |
| }); |
| } |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| _optionEx(config, flags, description, fn, defaultValue) { |
| if (typeof flags === 'object' && flags instanceof Option) { |
| throw new Error( |
| 'To add an Option object use addOption() instead of option() or requiredOption()', |
| ); |
| } |
| const option = this.createOption(flags, description); |
| option.makeOptionMandatory(!!config.mandatory); |
| if (typeof fn === 'function') { |
| option.default(defaultValue).argParser(fn); |
| } else if (fn instanceof RegExp) { |
| |
| const regex = fn; |
| fn = (val, def) => { |
| const m = regex.exec(val); |
| return m ? m[0] : def; |
| }; |
| option.default(defaultValue).argParser(fn); |
| } else { |
| option.default(fn); |
| } |
|
|
| return this.addOption(option); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| option(flags, description, parseArg, defaultValue) { |
| return this._optionEx({}, flags, description, parseArg, defaultValue); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| requiredOption(flags, description, parseArg, defaultValue) { |
| return this._optionEx( |
| { mandatory: true }, |
| flags, |
| description, |
| parseArg, |
| defaultValue, |
| ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| combineFlagAndOptionalValue(combine = true) { |
| this._combineFlagAndOptionalValue = !!combine; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| allowUnknownOption(allowUnknown = true) { |
| this._allowUnknownOption = !!allowUnknown; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| allowExcessArguments(allowExcess = true) { |
| this._allowExcessArguments = !!allowExcess; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| enablePositionalOptions(positional = true) { |
| this._enablePositionalOptions = !!positional; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| passThroughOptions(passThrough = true) { |
| this._passThroughOptions = !!passThrough; |
| this._checkForBrokenPassThrough(); |
| return this; |
| } |
|
|
| |
| |
| |
|
|
| _checkForBrokenPassThrough() { |
| if ( |
| this.parent && |
| this._passThroughOptions && |
| !this.parent._enablePositionalOptions |
| ) { |
| throw new Error( |
| `passThroughOptions cannot be used for '${this._name}' without turning on enablePositionalOptions for parent command(s)`, |
| ); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| storeOptionsAsProperties(storeAsProperties = true) { |
| if (this.options.length) { |
| throw new Error('call .storeOptionsAsProperties() before adding options'); |
| } |
| if (Object.keys(this._optionValues).length) { |
| throw new Error( |
| 'call .storeOptionsAsProperties() before setting option values', |
| ); |
| } |
| this._storeOptionsAsProperties = !!storeAsProperties; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| getOptionValue(key) { |
| if (this._storeOptionsAsProperties) { |
| return this[key]; |
| } |
| return this._optionValues[key]; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| setOptionValue(key, value) { |
| return this.setOptionValueWithSource(key, value, undefined); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| setOptionValueWithSource(key, value, source) { |
| if (this._storeOptionsAsProperties) { |
| this[key] = value; |
| } else { |
| this._optionValues[key] = value; |
| } |
| this._optionValueSources[key] = source; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| getOptionValueSource(key) { |
| return this._optionValueSources[key]; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| getOptionValueSourceWithGlobals(key) { |
| |
| let source; |
| this._getCommandAndAncestors().forEach((cmd) => { |
| if (cmd.getOptionValueSource(key) !== undefined) { |
| source = cmd.getOptionValueSource(key); |
| } |
| }); |
| return source; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _prepareUserArgs(argv, parseOptions) { |
| if (argv !== undefined && !Array.isArray(argv)) { |
| throw new Error('first parameter to parse must be array or undefined'); |
| } |
| parseOptions = parseOptions || {}; |
|
|
| |
| if (argv === undefined && parseOptions.from === undefined) { |
| if (process.versions?.electron) { |
| parseOptions.from = 'electron'; |
| } |
| |
| const execArgv = process.execArgv ?? []; |
| if ( |
| execArgv.includes('-e') || |
| execArgv.includes('--eval') || |
| execArgv.includes('-p') || |
| execArgv.includes('--print') |
| ) { |
| parseOptions.from = 'eval'; |
| } |
| } |
|
|
| |
| if (argv === undefined) { |
| argv = process.argv; |
| } |
| this.rawArgs = argv.slice(); |
|
|
| |
| let userArgs; |
| switch (parseOptions.from) { |
| case undefined: |
| case 'node': |
| this._scriptPath = argv[1]; |
| userArgs = argv.slice(2); |
| break; |
| case 'electron': |
| |
| if (process.defaultApp) { |
| this._scriptPath = argv[1]; |
| userArgs = argv.slice(2); |
| } else { |
| userArgs = argv.slice(1); |
| } |
| break; |
| case 'user': |
| userArgs = argv.slice(0); |
| break; |
| case 'eval': |
| userArgs = argv.slice(1); |
| break; |
| default: |
| throw new Error( |
| `unexpected parse option { from: '${parseOptions.from}' }`, |
| ); |
| } |
|
|
| |
| if (!this._name && this._scriptPath) |
| this.nameFromFilename(this._scriptPath); |
| this._name = this._name || 'program'; |
|
|
| return userArgs; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| parse(argv, parseOptions) { |
| this._prepareForParse(); |
| const userArgs = this._prepareUserArgs(argv, parseOptions); |
| this._parseCommand([], userArgs); |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| async parseAsync(argv, parseOptions) { |
| this._prepareForParse(); |
| const userArgs = this._prepareUserArgs(argv, parseOptions); |
| await this._parseCommand([], userArgs); |
|
|
| return this; |
| } |
|
|
| _prepareForParse() { |
| if (this._savedState === null) { |
| this.saveStateBeforeParse(); |
| } else { |
| this.restoreStateBeforeParse(); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| saveStateBeforeParse() { |
| this._savedState = { |
| |
| _name: this._name, |
| |
| |
| _optionValues: { ...this._optionValues }, |
| _optionValueSources: { ...this._optionValueSources }, |
| }; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| restoreStateBeforeParse() { |
| if (this._storeOptionsAsProperties) |
| throw new Error(`Can not call parse again when storeOptionsAsProperties is true. |
| - either make a new Command for each call to parse, or stop storing options as properties`); |
|
|
| |
| this._name = this._savedState._name; |
| this._scriptPath = null; |
| this.rawArgs = []; |
| |
| this._optionValues = { ...this._savedState._optionValues }; |
| this._optionValueSources = { ...this._savedState._optionValueSources }; |
| |
| this.args = []; |
| |
| this.processedArgs = []; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| _checkForMissingExecutable(executableFile, executableDir, subcommandName) { |
| if (fs.existsSync(executableFile)) return; |
|
|
| const executableDirMessage = executableDir |
| ? `searched for local subcommand relative to directory '${executableDir}'` |
| : 'no directory for search for local subcommand, use .executableDir() to supply a custom directory'; |
| const executableMissing = `'${executableFile}' does not exist |
| - if '${subcommandName}' is not meant to be an executable command, remove description parameter from '.command()' and use '.description()' instead |
| - if the default executable name is not suitable, use the executableFile option to supply a custom name or path |
| - ${executableDirMessage}`; |
| throw new Error(executableMissing); |
| } |
|
|
| |
| |
| |
| |
| |
|
|
| _executeSubCommand(subcommand, args) { |
| args = args.slice(); |
| let launchWithNode = false; |
| const sourceExt = ['.js', '.ts', '.tsx', '.mjs', '.cjs']; |
|
|
| function findFile(baseDir, baseName) { |
| |
| const localBin = path.resolve(baseDir, baseName); |
| if (fs.existsSync(localBin)) return localBin; |
|
|
| |
| if (sourceExt.includes(path.extname(baseName))) return undefined; |
|
|
| |
| const foundExt = sourceExt.find((ext) => |
| fs.existsSync(`${localBin}${ext}`), |
| ); |
| if (foundExt) return `${localBin}${foundExt}`; |
|
|
| return undefined; |
| } |
|
|
| |
| this._checkForMissingMandatoryOptions(); |
| this._checkForConflictingOptions(); |
|
|
| |
| let executableFile = |
| subcommand._executableFile || `${this._name}-${subcommand._name}`; |
| let executableDir = this._executableDir || ''; |
| if (this._scriptPath) { |
| let resolvedScriptPath; |
| try { |
| resolvedScriptPath = fs.realpathSync(this._scriptPath); |
| } catch { |
| resolvedScriptPath = this._scriptPath; |
| } |
| executableDir = path.resolve( |
| path.dirname(resolvedScriptPath), |
| executableDir, |
| ); |
| } |
|
|
| |
| if (executableDir) { |
| let localFile = findFile(executableDir, executableFile); |
|
|
| |
| if (!localFile && !subcommand._executableFile && this._scriptPath) { |
| const legacyName = path.basename( |
| this._scriptPath, |
| path.extname(this._scriptPath), |
| ); |
| if (legacyName !== this._name) { |
| localFile = findFile( |
| executableDir, |
| `${legacyName}-${subcommand._name}`, |
| ); |
| } |
| } |
| executableFile = localFile || executableFile; |
| } |
|
|
| launchWithNode = sourceExt.includes(path.extname(executableFile)); |
|
|
| let proc; |
| if (process.platform !== 'win32') { |
| if (launchWithNode) { |
| args.unshift(executableFile); |
| |
| args = incrementNodeInspectorPort(process.execArgv).concat(args); |
|
|
| proc = childProcess.spawn(process.argv[0], args, { stdio: 'inherit' }); |
| } else { |
| proc = childProcess.spawn(executableFile, args, { stdio: 'inherit' }); |
| } |
| } else { |
| this._checkForMissingExecutable( |
| executableFile, |
| executableDir, |
| subcommand._name, |
| ); |
| args.unshift(executableFile); |
| |
| args = incrementNodeInspectorPort(process.execArgv).concat(args); |
| proc = childProcess.spawn(process.execPath, args, { stdio: 'inherit' }); |
| } |
|
|
| if (!proc.killed) { |
| |
| const signals = ['SIGUSR1', 'SIGUSR2', 'SIGTERM', 'SIGINT', 'SIGHUP']; |
| signals.forEach((signal) => { |
| process.on(signal, () => { |
| if (proc.killed === false && proc.exitCode === null) { |
| |
| proc.kill(signal); |
| } |
| }); |
| }); |
| } |
|
|
| |
| const exitCallback = this._exitCallback; |
| proc.on('close', (code) => { |
| code = code ?? 1; |
| if (!exitCallback) { |
| process.exit(code); |
| } else { |
| exitCallback( |
| new CommanderError( |
| code, |
| 'commander.executeSubCommandAsync', |
| '(close)', |
| ), |
| ); |
| } |
| }); |
| proc.on('error', (err) => { |
| |
| if (err.code === 'ENOENT') { |
| this._checkForMissingExecutable( |
| executableFile, |
| executableDir, |
| subcommand._name, |
| ); |
| |
| } else if (err.code === 'EACCES') { |
| throw new Error(`'${executableFile}' not executable`); |
| } |
| if (!exitCallback) { |
| process.exit(1); |
| } else { |
| const wrappedError = new CommanderError( |
| 1, |
| 'commander.executeSubCommandAsync', |
| '(error)', |
| ); |
| wrappedError.nestedError = err; |
| exitCallback(wrappedError); |
| } |
| }); |
|
|
| |
| this.runningCommand = proc; |
| } |
|
|
| |
| |
| |
|
|
| _dispatchSubcommand(commandName, operands, unknown) { |
| const subCommand = this._findCommand(commandName); |
| if (!subCommand) this.help({ error: true }); |
|
|
| subCommand._prepareForParse(); |
| let promiseChain; |
| promiseChain = this._chainOrCallSubCommandHook( |
| promiseChain, |
| subCommand, |
| 'preSubcommand', |
| ); |
| promiseChain = this._chainOrCall(promiseChain, () => { |
| if (subCommand._executableHandler) { |
| this._executeSubCommand(subCommand, operands.concat(unknown)); |
| } else { |
| return subCommand._parseCommand(operands, unknown); |
| } |
| }); |
| return promiseChain; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _dispatchHelpCommand(subcommandName) { |
| if (!subcommandName) { |
| this.help(); |
| } |
| const subCommand = this._findCommand(subcommandName); |
| if (subCommand && !subCommand._executableHandler) { |
| subCommand.help(); |
| } |
|
|
| |
| return this._dispatchSubcommand( |
| subcommandName, |
| [], |
| [this._getHelpOption()?.long ?? this._getHelpOption()?.short ?? '--help'], |
| ); |
| } |
|
|
| |
| |
| |
| |
| |
|
|
| _checkNumberOfArguments() { |
| |
| this.registeredArguments.forEach((arg, i) => { |
| if (arg.required && this.args[i] == null) { |
| this.missingArgument(arg.name()); |
| } |
| }); |
| |
| if ( |
| this.registeredArguments.length > 0 && |
| this.registeredArguments[this.registeredArguments.length - 1].variadic |
| ) { |
| return; |
| } |
| if (this.args.length > this.registeredArguments.length) { |
| this._excessArguments(this.args); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
|
|
| _processArguments() { |
| const myParseArg = (argument, value, previous) => { |
| |
| let parsedValue = value; |
| if (value !== null && argument.parseArg) { |
| const invalidValueMessage = `error: command-argument value '${value}' is invalid for argument '${argument.name()}'.`; |
| parsedValue = this._callParseArg( |
| argument, |
| value, |
| previous, |
| invalidValueMessage, |
| ); |
| } |
| return parsedValue; |
| }; |
|
|
| this._checkNumberOfArguments(); |
|
|
| const processedArgs = []; |
| this.registeredArguments.forEach((declaredArg, index) => { |
| let value = declaredArg.defaultValue; |
| if (declaredArg.variadic) { |
| |
| if (index < this.args.length) { |
| value = this.args.slice(index); |
| if (declaredArg.parseArg) { |
| value = value.reduce((processed, v) => { |
| return myParseArg(declaredArg, v, processed); |
| }, declaredArg.defaultValue); |
| } |
| } else if (value === undefined) { |
| value = []; |
| } |
| } else if (index < this.args.length) { |
| value = this.args[index]; |
| if (declaredArg.parseArg) { |
| value = myParseArg(declaredArg, value, declaredArg.defaultValue); |
| } |
| } |
| processedArgs[index] = value; |
| }); |
| this.processedArgs = processedArgs; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| _chainOrCall(promise, fn) { |
| |
| if (promise && promise.then && typeof promise.then === 'function') { |
| |
| return promise.then(() => fn()); |
| } |
| |
| return fn(); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| _chainOrCallHooks(promise, event) { |
| let result = promise; |
| const hooks = []; |
| this._getCommandAndAncestors() |
| .reverse() |
| .filter((cmd) => cmd._lifeCycleHooks[event] !== undefined) |
| .forEach((hookedCommand) => { |
| hookedCommand._lifeCycleHooks[event].forEach((callback) => { |
| hooks.push({ hookedCommand, callback }); |
| }); |
| }); |
| if (event === 'postAction') { |
| hooks.reverse(); |
| } |
|
|
| hooks.forEach((hookDetail) => { |
| result = this._chainOrCall(result, () => { |
| return hookDetail.callback(hookDetail.hookedCommand, this); |
| }); |
| }); |
| return result; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| _chainOrCallSubCommandHook(promise, subCommand, event) { |
| let result = promise; |
| if (this._lifeCycleHooks[event] !== undefined) { |
| this._lifeCycleHooks[event].forEach((hook) => { |
| result = this._chainOrCall(result, () => { |
| return hook(this, subCommand); |
| }); |
| }); |
| } |
| return result; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _parseCommand(operands, unknown) { |
| const parsed = this.parseOptions(unknown); |
| this._parseOptionsEnv(); |
| this._parseOptionsImplied(); |
| operands = operands.concat(parsed.operands); |
| unknown = parsed.unknown; |
| this.args = operands.concat(unknown); |
|
|
| if (operands && this._findCommand(operands[0])) { |
| return this._dispatchSubcommand(operands[0], operands.slice(1), unknown); |
| } |
| if ( |
| this._getHelpCommand() && |
| operands[0] === this._getHelpCommand().name() |
| ) { |
| return this._dispatchHelpCommand(operands[1]); |
| } |
| if (this._defaultCommandName) { |
| this._outputHelpIfRequested(unknown); |
| return this._dispatchSubcommand( |
| this._defaultCommandName, |
| operands, |
| unknown, |
| ); |
| } |
| if ( |
| this.commands.length && |
| this.args.length === 0 && |
| !this._actionHandler && |
| !this._defaultCommandName |
| ) { |
| |
| this.help({ error: true }); |
| } |
|
|
| this._outputHelpIfRequested(parsed.unknown); |
| this._checkForMissingMandatoryOptions(); |
| this._checkForConflictingOptions(); |
|
|
| |
| const checkForUnknownOptions = () => { |
| if (parsed.unknown.length > 0) { |
| this.unknownOption(parsed.unknown[0]); |
| } |
| }; |
|
|
| const commandEvent = `command:${this.name()}`; |
| if (this._actionHandler) { |
| checkForUnknownOptions(); |
| this._processArguments(); |
|
|
| let promiseChain; |
| promiseChain = this._chainOrCallHooks(promiseChain, 'preAction'); |
| promiseChain = this._chainOrCall(promiseChain, () => |
| this._actionHandler(this.processedArgs), |
| ); |
| if (this.parent) { |
| promiseChain = this._chainOrCall(promiseChain, () => { |
| this.parent.emit(commandEvent, operands, unknown); |
| }); |
| } |
| promiseChain = this._chainOrCallHooks(promiseChain, 'postAction'); |
| return promiseChain; |
| } |
| if (this.parent && this.parent.listenerCount(commandEvent)) { |
| checkForUnknownOptions(); |
| this._processArguments(); |
| this.parent.emit(commandEvent, operands, unknown); |
| } else if (operands.length) { |
| if (this._findCommand('*')) { |
| |
| return this._dispatchSubcommand('*', operands, unknown); |
| } |
| if (this.listenerCount('command:*')) { |
| |
| this.emit('command:*', operands, unknown); |
| } else if (this.commands.length) { |
| this.unknownCommand(); |
| } else { |
| checkForUnknownOptions(); |
| this._processArguments(); |
| } |
| } else if (this.commands.length) { |
| checkForUnknownOptions(); |
| |
| this.help({ error: true }); |
| } else { |
| checkForUnknownOptions(); |
| this._processArguments(); |
| |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| _findCommand(name) { |
| if (!name) return undefined; |
| return this.commands.find( |
| (cmd) => cmd._name === name || cmd._aliases.includes(name), |
| ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| _findOption(arg) { |
| return this.options.find((option) => option.is(arg)); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _checkForMissingMandatoryOptions() { |
| |
| this._getCommandAndAncestors().forEach((cmd) => { |
| cmd.options.forEach((anOption) => { |
| if ( |
| anOption.mandatory && |
| cmd.getOptionValue(anOption.attributeName()) === undefined |
| ) { |
| cmd.missingMandatoryOptionValue(anOption); |
| } |
| }); |
| }); |
| } |
|
|
| |
| |
| |
| |
| |
| _checkForConflictingLocalOptions() { |
| const definedNonDefaultOptions = this.options.filter((option) => { |
| const optionKey = option.attributeName(); |
| if (this.getOptionValue(optionKey) === undefined) { |
| return false; |
| } |
| return this.getOptionValueSource(optionKey) !== 'default'; |
| }); |
|
|
| const optionsWithConflicting = definedNonDefaultOptions.filter( |
| (option) => option.conflictsWith.length > 0, |
| ); |
|
|
| optionsWithConflicting.forEach((option) => { |
| const conflictingAndDefined = definedNonDefaultOptions.find((defined) => |
| option.conflictsWith.includes(defined.attributeName()), |
| ); |
| if (conflictingAndDefined) { |
| this._conflictingOption(option, conflictingAndDefined); |
| } |
| }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| _checkForConflictingOptions() { |
| |
| this._getCommandAndAncestors().forEach((cmd) => { |
| cmd._checkForConflictingLocalOptions(); |
| }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| parseOptions(argv) { |
| const operands = []; |
| const unknown = []; |
| let dest = operands; |
| const args = argv.slice(); |
|
|
| function maybeOption(arg) { |
| return arg.length > 1 && arg[0] === '-'; |
| } |
|
|
| |
| let activeVariadicOption = null; |
| while (args.length) { |
| const arg = args.shift(); |
|
|
| |
| if (arg === '--') { |
| if (dest === unknown) dest.push(arg); |
| dest.push(...args); |
| break; |
| } |
|
|
| if (activeVariadicOption && !maybeOption(arg)) { |
| this.emit(`option:${activeVariadicOption.name()}`, arg); |
| continue; |
| } |
| activeVariadicOption = null; |
|
|
| if (maybeOption(arg)) { |
| const option = this._findOption(arg); |
| |
| if (option) { |
| if (option.required) { |
| const value = args.shift(); |
| if (value === undefined) this.optionMissingArgument(option); |
| this.emit(`option:${option.name()}`, value); |
| } else if (option.optional) { |
| let value = null; |
| |
| if (args.length > 0 && !maybeOption(args[0])) { |
| value = args.shift(); |
| } |
| this.emit(`option:${option.name()}`, value); |
| } else { |
| |
| this.emit(`option:${option.name()}`); |
| } |
| activeVariadicOption = option.variadic ? option : null; |
| continue; |
| } |
| } |
|
|
| |
| if (arg.length > 2 && arg[0] === '-' && arg[1] !== '-') { |
| const option = this._findOption(`-${arg[1]}`); |
| if (option) { |
| if ( |
| option.required || |
| (option.optional && this._combineFlagAndOptionalValue) |
| ) { |
| |
| this.emit(`option:${option.name()}`, arg.slice(2)); |
| } else { |
| |
| this.emit(`option:${option.name()}`); |
| args.unshift(`-${arg.slice(2)}`); |
| } |
| continue; |
| } |
| } |
|
|
| |
| if (/^--[^=]+=/.test(arg)) { |
| const index = arg.indexOf('='); |
| const option = this._findOption(arg.slice(0, index)); |
| if (option && (option.required || option.optional)) { |
| this.emit(`option:${option.name()}`, arg.slice(index + 1)); |
| continue; |
| } |
| } |
|
|
| |
| |
|
|
| |
| if (maybeOption(arg)) { |
| dest = unknown; |
| } |
|
|
| |
| if ( |
| (this._enablePositionalOptions || this._passThroughOptions) && |
| operands.length === 0 && |
| unknown.length === 0 |
| ) { |
| if (this._findCommand(arg)) { |
| operands.push(arg); |
| if (args.length > 0) unknown.push(...args); |
| break; |
| } else if ( |
| this._getHelpCommand() && |
| arg === this._getHelpCommand().name() |
| ) { |
| operands.push(arg); |
| if (args.length > 0) operands.push(...args); |
| break; |
| } else if (this._defaultCommandName) { |
| unknown.push(arg); |
| if (args.length > 0) unknown.push(...args); |
| break; |
| } |
| } |
|
|
| |
| if (this._passThroughOptions) { |
| dest.push(arg); |
| if (args.length > 0) dest.push(...args); |
| break; |
| } |
|
|
| |
| dest.push(arg); |
| } |
|
|
| return { operands, unknown }; |
| } |
|
|
| |
| |
| |
| |
| |
| opts() { |
| if (this._storeOptionsAsProperties) { |
| |
| const result = {}; |
| const len = this.options.length; |
|
|
| for (let i = 0; i < len; i++) { |
| const key = this.options[i].attributeName(); |
| result[key] = |
| key === this._versionOptionName ? this._version : this[key]; |
| } |
| return result; |
| } |
|
|
| return this._optionValues; |
| } |
|
|
| |
| |
| |
| |
| |
| optsWithGlobals() { |
| |
| return this._getCommandAndAncestors().reduce( |
| (combinedOptions, cmd) => Object.assign(combinedOptions, cmd.opts()), |
| {}, |
| ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| error(message, errorOptions) { |
| |
| this._outputConfiguration.outputError( |
| `${message}\n`, |
| this._outputConfiguration.writeErr, |
| ); |
| if (typeof this._showHelpAfterError === 'string') { |
| this._outputConfiguration.writeErr(`${this._showHelpAfterError}\n`); |
| } else if (this._showHelpAfterError) { |
| this._outputConfiguration.writeErr('\n'); |
| this.outputHelp({ error: true }); |
| } |
|
|
| |
| const config = errorOptions || {}; |
| const exitCode = config.exitCode || 1; |
| const code = config.code || 'commander.error'; |
| this._exit(exitCode, code, message); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| _parseOptionsEnv() { |
| this.options.forEach((option) => { |
| if (option.envVar && option.envVar in process.env) { |
| const optionKey = option.attributeName(); |
| |
| if ( |
| this.getOptionValue(optionKey) === undefined || |
| ['default', 'config', 'env'].includes( |
| this.getOptionValueSource(optionKey), |
| ) |
| ) { |
| if (option.required || option.optional) { |
| |
| |
| this.emit(`optionEnv:${option.name()}`, process.env[option.envVar]); |
| } else { |
| |
| |
| this.emit(`optionEnv:${option.name()}`); |
| } |
| } |
| } |
| }); |
| } |
|
|
| |
| |
| |
| |
| |
| _parseOptionsImplied() { |
| const dualHelper = new DualOptions(this.options); |
| const hasCustomOptionValue = (optionKey) => { |
| return ( |
| this.getOptionValue(optionKey) !== undefined && |
| !['default', 'implied'].includes(this.getOptionValueSource(optionKey)) |
| ); |
| }; |
| this.options |
| .filter( |
| (option) => |
| option.implied !== undefined && |
| hasCustomOptionValue(option.attributeName()) && |
| dualHelper.valueFromOption( |
| this.getOptionValue(option.attributeName()), |
| option, |
| ), |
| ) |
| .forEach((option) => { |
| Object.keys(option.implied) |
| .filter((impliedKey) => !hasCustomOptionValue(impliedKey)) |
| .forEach((impliedKey) => { |
| this.setOptionValueWithSource( |
| impliedKey, |
| option.implied[impliedKey], |
| 'implied', |
| ); |
| }); |
| }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| missingArgument(name) { |
| const message = `error: missing required argument '${name}'`; |
| this.error(message, { code: 'commander.missingArgument' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| optionMissingArgument(option) { |
| const message = `error: option '${option.flags}' argument missing`; |
| this.error(message, { code: 'commander.optionMissingArgument' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| missingMandatoryOptionValue(option) { |
| const message = `error: required option '${option.flags}' not specified`; |
| this.error(message, { code: 'commander.missingMandatoryOptionValue' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| _conflictingOption(option, conflictingOption) { |
| |
| |
| const findBestOptionFromValue = (option) => { |
| const optionKey = option.attributeName(); |
| const optionValue = this.getOptionValue(optionKey); |
| const negativeOption = this.options.find( |
| (target) => target.negate && optionKey === target.attributeName(), |
| ); |
| const positiveOption = this.options.find( |
| (target) => !target.negate && optionKey === target.attributeName(), |
| ); |
| if ( |
| negativeOption && |
| ((negativeOption.presetArg === undefined && optionValue === false) || |
| (negativeOption.presetArg !== undefined && |
| optionValue === negativeOption.presetArg)) |
| ) { |
| return negativeOption; |
| } |
| return positiveOption || option; |
| }; |
|
|
| const getErrorMessage = (option) => { |
| const bestOption = findBestOptionFromValue(option); |
| const optionKey = bestOption.attributeName(); |
| const source = this.getOptionValueSource(optionKey); |
| if (source === 'env') { |
| return `environment variable '${bestOption.envVar}'`; |
| } |
| return `option '${bestOption.flags}'`; |
| }; |
|
|
| const message = `error: ${getErrorMessage(option)} cannot be used with ${getErrorMessage(conflictingOption)}`; |
| this.error(message, { code: 'commander.conflictingOption' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| unknownOption(flag) { |
| if (this._allowUnknownOption) return; |
| let suggestion = ''; |
|
|
| if (flag.startsWith('--') && this._showSuggestionAfterError) { |
| |
| let candidateFlags = []; |
| |
| let command = this; |
| do { |
| const moreFlags = command |
| .createHelp() |
| .visibleOptions(command) |
| .filter((option) => option.long) |
| .map((option) => option.long); |
| candidateFlags = candidateFlags.concat(moreFlags); |
| command = command.parent; |
| } while (command && !command._enablePositionalOptions); |
| suggestion = suggestSimilar(flag, candidateFlags); |
| } |
|
|
| const message = `error: unknown option '${flag}'${suggestion}`; |
| this.error(message, { code: 'commander.unknownOption' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _excessArguments(receivedArgs) { |
| if (this._allowExcessArguments) return; |
|
|
| const expected = this.registeredArguments.length; |
| const s = expected === 1 ? '' : 's'; |
| const forSubcommand = this.parent ? ` for '${this.name()}'` : ''; |
| const message = `error: too many arguments${forSubcommand}. Expected ${expected} argument${s} but got ${receivedArgs.length}.`; |
| this.error(message, { code: 'commander.excessArguments' }); |
| } |
|
|
| |
| |
| |
| |
| |
|
|
| unknownCommand() { |
| const unknownName = this.args[0]; |
| let suggestion = ''; |
|
|
| if (this._showSuggestionAfterError) { |
| const candidateNames = []; |
| this.createHelp() |
| .visibleCommands(this) |
| .forEach((command) => { |
| candidateNames.push(command.name()); |
| |
| if (command.alias()) candidateNames.push(command.alias()); |
| }); |
| suggestion = suggestSimilar(unknownName, candidateNames); |
| } |
|
|
| const message = `error: unknown command '${unknownName}'${suggestion}`; |
| this.error(message, { code: 'commander.unknownCommand' }); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| version(str, flags, description) { |
| if (str === undefined) return this._version; |
| this._version = str; |
| flags = flags || '-V, --version'; |
| description = description || 'output the version number'; |
| const versionOption = this.createOption(flags, description); |
| this._versionOptionName = versionOption.attributeName(); |
| this._registerOption(versionOption); |
|
|
| this.on('option:' + versionOption.name(), () => { |
| this._outputConfiguration.writeOut(`${str}\n`); |
| this._exit(0, 'commander.version', str); |
| }); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| description(str, argsDescription) { |
| if (str === undefined && argsDescription === undefined) |
| return this._description; |
| this._description = str; |
| if (argsDescription) { |
| this._argsDescription = argsDescription; |
| } |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| summary(str) { |
| if (str === undefined) return this._summary; |
| this._summary = str; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| alias(alias) { |
| if (alias === undefined) return this._aliases[0]; |
|
|
| |
| |
| let command = this; |
| if ( |
| this.commands.length !== 0 && |
| this.commands[this.commands.length - 1]._executableHandler |
| ) { |
| |
| command = this.commands[this.commands.length - 1]; |
| } |
|
|
| if (alias === command._name) |
| throw new Error("Command alias can't be the same as its name"); |
| const matchingCommand = this.parent?._findCommand(alias); |
| if (matchingCommand) { |
| |
| const existingCmd = [matchingCommand.name()] |
| .concat(matchingCommand.aliases()) |
| .join('|'); |
| throw new Error( |
| `cannot add alias '${alias}' to command '${this.name()}' as already have command '${existingCmd}'`, |
| ); |
| } |
|
|
| command._aliases.push(alias); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| aliases(aliases) { |
| |
| if (aliases === undefined) return this._aliases; |
|
|
| aliases.forEach((alias) => this.alias(alias)); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| usage(str) { |
| if (str === undefined) { |
| if (this._usage) return this._usage; |
|
|
| const args = this.registeredArguments.map((arg) => { |
| return humanReadableArgName(arg); |
| }); |
| return [] |
| .concat( |
| this.options.length || this._helpOption !== null ? '[options]' : [], |
| this.commands.length ? '[command]' : [], |
| this.registeredArguments.length ? args : [], |
| ) |
| .join(' '); |
| } |
|
|
| this._usage = str; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| name(str) { |
| if (str === undefined) return this._name; |
| this._name = str; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| nameFromFilename(filename) { |
| this._name = path.basename(filename, path.extname(filename)); |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| executableDir(path) { |
| if (path === undefined) return this._executableDir; |
| this._executableDir = path; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| helpInformation(contextOptions) { |
| const helper = this.createHelp(); |
| const context = this._getOutputContext(contextOptions); |
| helper.prepareContext({ |
| error: context.error, |
| helpWidth: context.helpWidth, |
| outputHasColors: context.hasColors, |
| }); |
| const text = helper.formatHelp(this, helper); |
| if (context.hasColors) return text; |
| return this._outputConfiguration.stripColor(text); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| _getOutputContext(contextOptions) { |
| contextOptions = contextOptions || {}; |
| const error = !!contextOptions.error; |
| let baseWrite; |
| let hasColors; |
| let helpWidth; |
| if (error) { |
| baseWrite = (str) => this._outputConfiguration.writeErr(str); |
| hasColors = this._outputConfiguration.getErrHasColors(); |
| helpWidth = this._outputConfiguration.getErrHelpWidth(); |
| } else { |
| baseWrite = (str) => this._outputConfiguration.writeOut(str); |
| hasColors = this._outputConfiguration.getOutHasColors(); |
| helpWidth = this._outputConfiguration.getOutHelpWidth(); |
| } |
| const write = (str) => { |
| if (!hasColors) str = this._outputConfiguration.stripColor(str); |
| return baseWrite(str); |
| }; |
| return { error, write, hasColors, helpWidth }; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| outputHelp(contextOptions) { |
| let deprecatedCallback; |
| if (typeof contextOptions === 'function') { |
| deprecatedCallback = contextOptions; |
| contextOptions = undefined; |
| } |
|
|
| const outputContext = this._getOutputContext(contextOptions); |
| |
| const eventContext = { |
| error: outputContext.error, |
| write: outputContext.write, |
| command: this, |
| }; |
|
|
| this._getCommandAndAncestors() |
| .reverse() |
| .forEach((command) => command.emit('beforeAllHelp', eventContext)); |
| this.emit('beforeHelp', eventContext); |
|
|
| let helpInformation = this.helpInformation({ error: outputContext.error }); |
| if (deprecatedCallback) { |
| helpInformation = deprecatedCallback(helpInformation); |
| if ( |
| typeof helpInformation !== 'string' && |
| !Buffer.isBuffer(helpInformation) |
| ) { |
| throw new Error('outputHelp callback must return a string or a Buffer'); |
| } |
| } |
| outputContext.write(helpInformation); |
|
|
| if (this._getHelpOption()?.long) { |
| this.emit(this._getHelpOption().long); |
| } |
| this.emit('afterHelp', eventContext); |
| this._getCommandAndAncestors().forEach((command) => |
| command.emit('afterAllHelp', eventContext), |
| ); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| helpOption(flags, description) { |
| |
| if (typeof flags === 'boolean') { |
| |
| |
| if (flags) { |
| this._helpOption = this._helpOption ?? undefined; |
| } else { |
| this._helpOption = null; |
| } |
| return this; |
| } |
|
|
| |
| flags = flags ?? '-h, --help'; |
| description = description ?? 'display help for command'; |
| this._helpOption = this.createOption(flags, description); |
|
|
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| _getHelpOption() { |
| |
| if (this._helpOption === undefined) { |
| this.helpOption(undefined, undefined); |
| } |
| return this._helpOption; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| addHelpOption(option) { |
| this._helpOption = option; |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| help(contextOptions) { |
| this.outputHelp(contextOptions); |
| let exitCode = Number(process.exitCode ?? 0); |
| if ( |
| exitCode === 0 && |
| contextOptions && |
| typeof contextOptions !== 'function' && |
| contextOptions.error |
| ) { |
| exitCode = 1; |
| } |
| |
| this._exit(exitCode, 'commander.help', '(outputHelp)'); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| addHelpText(position, text) { |
| const allowedValues = ['beforeAll', 'before', 'after', 'afterAll']; |
| if (!allowedValues.includes(position)) { |
| throw new Error(`Unexpected value for position to addHelpText. |
| Expecting one of '${allowedValues.join("', '")}'`); |
| } |
|
|
| const helpEvent = `${position}Help`; |
| this.on(helpEvent, ( context) => { |
| let helpStr; |
| if (typeof text === 'function') { |
| helpStr = text({ error: context.error, command: context.command }); |
| } else { |
| helpStr = text; |
| } |
| |
| if (helpStr) { |
| context.write(`${helpStr}\n`); |
| } |
| }); |
| return this; |
| } |
|
|
| |
| |
| |
| |
| |
| |
|
|
| _outputHelpIfRequested(args) { |
| const helpOption = this._getHelpOption(); |
| const helpRequested = helpOption && args.find((arg) => helpOption.is(arg)); |
| if (helpRequested) { |
| this.outputHelp(); |
| |
| this._exit(0, 'commander.helpDisplayed', '(outputHelp)'); |
| } |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
|
|
| function incrementNodeInspectorPort(args) { |
| |
| |
| |
| |
| return args.map((arg) => { |
| if (!arg.startsWith('--inspect')) { |
| return arg; |
| } |
| let debugOption; |
| let debugHost = '127.0.0.1'; |
| let debugPort = '9229'; |
| let match; |
| if ((match = arg.match(/^(--inspect(-brk)?)$/)) !== null) { |
| |
| debugOption = match[1]; |
| } else if ( |
| (match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+)$/)) !== null |
| ) { |
| debugOption = match[1]; |
| if (/^\d+$/.test(match[3])) { |
| |
| debugPort = match[3]; |
| } else { |
| |
| debugHost = match[3]; |
| } |
| } else if ( |
| (match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+):(\d+)$/)) !== null |
| ) { |
| |
| debugOption = match[1]; |
| debugHost = match[3]; |
| debugPort = match[4]; |
| } |
|
|
| if (debugOption && debugPort !== '0') { |
| return `${debugOption}=${debugHost}:${parseInt(debugPort) + 1}`; |
| } |
| return arg; |
| }); |
| } |
|
|
| |
| |
| |
| |
| function useColor() { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| if ( |
| process.env.NO_COLOR || |
| process.env.FORCE_COLOR === '0' || |
| process.env.FORCE_COLOR === 'false' |
| ) |
| return false; |
| if (process.env.FORCE_COLOR || process.env.CLICOLOR_FORCE !== undefined) |
| return true; |
| return undefined; |
| } |
|
|
| exports.Command = Command; |
| exports.useColor = useColor; |
|
|