You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					1757 lines
				
				50 KiB
			
		
		
			
		
	
	
					1757 lines
				
				50 KiB
			| 
											3 years ago
										 | /** | ||
|  |  * Module dependencies. | ||
|  |  */ | ||
|  | 
 | ||
|  | const EventEmitter = require('events').EventEmitter; | ||
|  | const spawn = require('child_process').spawn; | ||
|  | const path = require('path'); | ||
|  | const fs = require('fs'); | ||
|  | 
 | ||
|  | // @ts-check
 | ||
|  | 
 | ||
|  | class Option { | ||
|  |   /** | ||
|  |    * Initialize a new `Option` with the given `flags` and `description`. | ||
|  |    * | ||
|  |    * @param {string} flags | ||
|  |    * @param {string} description | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   constructor(flags, description) { | ||
|  |     this.flags = flags; | ||
|  |     this.required = flags.indexOf('<') >= 0; // A value must be supplied when the option is specified.
 | ||
|  |     this.optional = flags.indexOf('[') >= 0; // A value is optional when the option is specified.
 | ||
|  |     this.mandatory = false; // The option must have a value after parsing, which usually means it must be specified on command line.
 | ||
|  |     this.negate = flags.indexOf('-no-') !== -1; | ||
|  |     const flagParts = flags.split(/[ ,|]+/); | ||
|  |     if (flagParts.length > 1 && !/^[[<]/.test(flagParts[1])) this.short = flagParts.shift(); | ||
|  |     this.long = flagParts.shift(); | ||
|  |     this.description = description || ''; | ||
|  |     this.defaultValue = undefined; | ||
|  |   } | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return option name. | ||
|  |    * | ||
|  |    * @return {string} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   name() { | ||
|  |     return this.long.replace(/^--/, ''); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return option name, in a camelcase format that can be used | ||
|  |    * as a object attribute key. | ||
|  |    * | ||
|  |    * @return {string} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   attributeName() { | ||
|  |     return camelcase(this.name().replace(/^no-/, '')); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Check if `arg` matches the short or long flag. | ||
|  |    * | ||
|  |    * @param {string} arg | ||
|  |    * @return {boolean} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   is(arg) { | ||
|  |     return this.short === arg || this.long === arg; | ||
|  |   }; | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * CommanderError class | ||
|  |  * @class | ||
|  |  */ | ||
|  | class CommanderError extends Error { | ||
|  |   /** | ||
|  |    * Constructs the CommanderError class | ||
|  |    * @param {number} exitCode suggested exit code which could be used with process.exit | ||
|  |    * @param {string} code an id string representing the error | ||
|  |    * @param {string} message human-readable description of the error | ||
|  |    * @constructor | ||
|  |    */ | ||
|  |   constructor(exitCode, code, message) { | ||
|  |     super(message); | ||
|  |     // properly capture stack trace in Node.js
 | ||
|  |     Error.captureStackTrace(this, this.constructor); | ||
|  |     this.name = this.constructor.name; | ||
|  |     this.code = code; | ||
|  |     this.exitCode = exitCode; | ||
|  |     this.nestedError = undefined; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | class Command extends EventEmitter { | ||
|  |   /** | ||
|  |    * Initialize a new `Command`. | ||
|  |    * | ||
|  |    * @param {string} [name] | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   constructor(name) { | ||
|  |     super(); | ||
|  |     this.commands = []; | ||
|  |     this.options = []; | ||
|  |     this.parent = null; | ||
|  |     this._allowUnknownOption = false; | ||
|  |     this._args = []; | ||
|  |     this.rawArgs = null; | ||
|  |     this._scriptPath = null; | ||
|  |     this._name = name || ''; | ||
|  |     this._optionValues = {}; | ||
|  |     this._storeOptionsAsProperties = true; // backwards compatible by default
 | ||
|  |     this._passCommandToAction = true; // backwards compatible by default
 | ||
|  |     this._actionResults = []; | ||
|  |     this._actionHandler = null; | ||
|  |     this._executableHandler = false; | ||
|  |     this._executableFile = null; // custom name for executable
 | ||
|  |     this._defaultCommandName = null; | ||
|  |     this._exitCallback = null; | ||
|  |     this._aliases = []; | ||
|  | 
 | ||
|  |     this._hidden = false; | ||
|  |     this._helpFlags = '-h, --help'; | ||
|  |     this._helpDescription = 'display help for command'; | ||
|  |     this._helpShortFlag = '-h'; | ||
|  |     this._helpLongFlag = '--help'; | ||
|  |     this._hasImplicitHelpCommand = undefined; // Deliberately undefined, not decided whether true or false
 | ||
|  |     this._helpCommandName = 'help'; | ||
|  |     this._helpCommandnameAndArgs = 'help [command]'; | ||
|  |     this._helpCommandDescription = 'display help for command'; | ||
|  |   } | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Define a command. | ||
|  |    * | ||
|  |    * There are two styles of command: pay attention to where to put the description. | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *      // Command implemented using action handler (description is supplied separately to `.command`)
 | ||
|  |    *      program | ||
|  |    *        .command('clone <source> [destination]') | ||
|  |    *        .description('clone a repository into a newly created directory') | ||
|  |    *        .action((source, destination) => { | ||
|  |    *          console.log('clone command called'); | ||
|  |    *        }); | ||
|  |    * | ||
|  |    *      // Command implemented using separate executable file (description is second parameter to `.command`)
 | ||
|  |    *      program | ||
|  |    *        .command('start <service>', 'start named service') | ||
|  |    *        .command('stop [service]', 'stop named service, or all if no name supplied'); | ||
|  |    * | ||
|  |    * @param {string} nameAndArgs - command name and arguments, args are `<required>` or `[optional]` and last may also be `variadic...` | ||
|  |    * @param {Object|string} [actionOptsOrExecDesc] - configuration options (for action), or description (for executable) | ||
|  |    * @param {Object} [execOpts] - configuration options (for executable) | ||
|  |    * @return {Command} returns new command for action handler, or `this` for executable command | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   command(nameAndArgs, actionOptsOrExecDesc, execOpts) { | ||
|  |     let desc = actionOptsOrExecDesc; | ||
|  |     let opts = execOpts; | ||
|  |     if (typeof desc === 'object' && desc !== null) { | ||
|  |       opts = desc; | ||
|  |       desc = null; | ||
|  |     } | ||
|  |     opts = opts || {}; | ||
|  |     const args = nameAndArgs.split(/ +/); | ||
|  |     const cmd = this.createCommand(args.shift()); | ||
|  | 
 | ||
|  |     if (desc) { | ||
|  |       cmd.description(desc); | ||
|  |       cmd._executableHandler = true; | ||
|  |     } | ||
|  |     if (opts.isDefault) this._defaultCommandName = cmd._name; | ||
|  | 
 | ||
|  |     cmd._hidden = !!(opts.noHelp || opts.hidden); | ||
|  |     cmd._helpFlags = this._helpFlags; | ||
|  |     cmd._helpDescription = this._helpDescription; | ||
|  |     cmd._helpShortFlag = this._helpShortFlag; | ||
|  |     cmd._helpLongFlag = this._helpLongFlag; | ||
|  |     cmd._helpCommandName = this._helpCommandName; | ||
|  |     cmd._helpCommandnameAndArgs = this._helpCommandnameAndArgs; | ||
|  |     cmd._helpCommandDescription = this._helpCommandDescription; | ||
|  |     cmd._exitCallback = this._exitCallback; | ||
|  |     cmd._storeOptionsAsProperties = this._storeOptionsAsProperties; | ||
|  |     cmd._passCommandToAction = this._passCommandToAction; | ||
|  | 
 | ||
|  |     cmd._executableFile = opts.executableFile || null; // Custom name for executable file, set missing to null to match constructor
 | ||
|  |     this.commands.push(cmd); | ||
|  |     cmd._parseExpectedArgs(args); | ||
|  |     cmd.parent = this; | ||
|  | 
 | ||
|  |     if (desc) return this; | ||
|  |     return cmd; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Factory routine to create a new unattached command. | ||
|  |    * | ||
|  |    * See .command() for creating an attached subcommand, which uses this routine to | ||
|  |    * create the command. You can override createCommand to customise subcommands. | ||
|  |    * | ||
|  |    * @param {string} [name] | ||
|  |    * @return {Command} new command | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   createCommand(name) { | ||
|  |     return new Command(name); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Add a prepared subcommand. | ||
|  |    * | ||
|  |    * See .command() for creating an attached subcommand which inherits settings from its parent. | ||
|  |    * | ||
|  |    * @param {Command} cmd - new subcommand | ||
|  |    * @param {Object} [opts] - configuration options | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   addCommand(cmd, opts) { | ||
|  |     if (!cmd._name) throw new Error('Command passed to .addCommand() must have a name'); | ||
|  | 
 | ||
|  |     // To keep things simple, block automatic name generation for deeply nested executables.
 | ||
|  |     // Fail fast and detect when adding rather than later when parsing.
 | ||
|  |     function checkExplicitNames(commandArray) { | ||
|  |       commandArray.forEach((cmd) => { | ||
|  |         if (cmd._executableHandler && !cmd._executableFile) { | ||
|  |           throw new Error(`Must specify executableFile for deeply nested executable: ${cmd.name()}`); | ||
|  |         } | ||
|  |         checkExplicitNames(cmd.commands); | ||
|  |       }); | ||
|  |     } | ||
|  |     checkExplicitNames(cmd.commands); | ||
|  | 
 | ||
|  |     opts = opts || {}; | ||
|  |     if (opts.isDefault) this._defaultCommandName = cmd._name; | ||
|  |     if (opts.noHelp || opts.hidden) cmd._hidden = true; // modifying passed command due to existing implementation
 | ||
|  | 
 | ||
|  |     this.commands.push(cmd); | ||
|  |     cmd.parent = this; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Define argument syntax for the command. | ||
|  |    * | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   arguments(desc) { | ||
|  |     return this._parseExpectedArgs(desc.split(/ +/)); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Override default decision whether to add implicit help command. | ||
|  |    * | ||
|  |    *    addHelpCommand() // force on
 | ||
|  |    *    addHelpCommand(false); // force off
 | ||
|  |    *    addHelpCommand('help [cmd]', 'display help for [cmd]'); // force on with custom detais
 | ||
|  |    * | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   addHelpCommand(enableOrNameAndArgs, description) { | ||
|  |     if (enableOrNameAndArgs === false) { | ||
|  |       this._hasImplicitHelpCommand = false; | ||
|  |     } else { | ||
|  |       this._hasImplicitHelpCommand = true; | ||
|  |       if (typeof enableOrNameAndArgs === 'string') { | ||
|  |         this._helpCommandName = enableOrNameAndArgs.split(' ')[0]; | ||
|  |         this._helpCommandnameAndArgs = enableOrNameAndArgs; | ||
|  |       } | ||
|  |       this._helpCommandDescription = description || this._helpCommandDescription; | ||
|  |     } | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * @return {boolean} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _lazyHasImplicitHelpCommand() { | ||
|  |     if (this._hasImplicitHelpCommand === undefined) { | ||
|  |       this._hasImplicitHelpCommand = this.commands.length && !this._actionHandler && !this._findCommand('help'); | ||
|  |     } | ||
|  |     return this._hasImplicitHelpCommand; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Parse expected `args`. | ||
|  |    * | ||
|  |    * For example `["[type]"]` becomes `[{ required: false, name: 'type' }]`. | ||
|  |    * | ||
|  |    * @param {Array} args | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _parseExpectedArgs(args) { | ||
|  |     if (!args.length) return; | ||
|  |     args.forEach((arg) => { | ||
|  |       const argDetails = { | ||
|  |         required: false, | ||
|  |         name: '', | ||
|  |         variadic: false | ||
|  |       }; | ||
|  | 
 | ||
|  |       switch (arg[0]) { | ||
|  |         case '<': | ||
|  |           argDetails.required = true; | ||
|  |           argDetails.name = arg.slice(1, -1); | ||
|  |           break; | ||
|  |         case '[': | ||
|  |           argDetails.name = arg.slice(1, -1); | ||
|  |           break; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (argDetails.name.length > 3 && argDetails.name.slice(-3) === '...') { | ||
|  |         argDetails.variadic = true; | ||
|  |         argDetails.name = argDetails.name.slice(0, -3); | ||
|  |       } | ||
|  |       if (argDetails.name) { | ||
|  |         this._args.push(argDetails); | ||
|  |       } | ||
|  |     }); | ||
|  |     this._args.forEach((arg, i) => { | ||
|  |       if (arg.variadic && i < this._args.length - 1) { | ||
|  |         throw new Error(`only the last argument can be variadic '${arg.name}'`); | ||
|  |       } | ||
|  |     }); | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Register callback to use as replacement for calling process.exit. | ||
|  |    * | ||
|  |    * @param {Function} [fn] optional callback which will be passed a CommanderError, defaults to throwing | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   exitOverride(fn) { | ||
|  |     if (fn) { | ||
|  |       this._exitCallback = fn; | ||
|  |     } else { | ||
|  |       this._exitCallback = (err) => { | ||
|  |         if (err.code !== 'commander.executeSubCommandAsync') { | ||
|  |           throw err; | ||
|  |         } else { | ||
|  |           // Async callback from spawn events, not useful to throw.
 | ||
|  |         } | ||
|  |       }; | ||
|  |     } | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Call process.exit, and _exitCallback if defined. | ||
|  |    * | ||
|  |    * @param {number} exitCode exit code for using with process.exit | ||
|  |    * @param {string} code an id string representing the error | ||
|  |    * @param {string} message human-readable description of the error | ||
|  |    * @return never | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _exit(exitCode, code, message) { | ||
|  |     if (this._exitCallback) { | ||
|  |       this._exitCallback(new CommanderError(exitCode, code, message)); | ||
|  |       // Expecting this line is not reached.
 | ||
|  |     } | ||
|  |     process.exit(exitCode); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Register callback `fn` for the command. | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *      program | ||
|  |    *        .command('help') | ||
|  |    *        .description('display verbose help') | ||
|  |    *        .action(function() { | ||
|  |    *           // output help here
 | ||
|  |    *        }); | ||
|  |    * | ||
|  |    * @param {Function} fn | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   action(fn) { | ||
|  |     const listener = (args) => { | ||
|  |       // The .action callback takes an extra parameter which is the command or options.
 | ||
|  |       const expectedArgsCount = this._args.length; | ||
|  |       const actionArgs = args.slice(0, expectedArgsCount); | ||
|  |       if (this._passCommandToAction) { | ||
|  |         actionArgs[expectedArgsCount] = this; | ||
|  |       } else { | ||
|  |         actionArgs[expectedArgsCount] = this.opts(); | ||
|  |       } | ||
|  |       // Add the extra arguments so available too.
 | ||
|  |       if (args.length > expectedArgsCount) { | ||
|  |         actionArgs.push(args.slice(expectedArgsCount)); | ||
|  |       } | ||
|  | 
 | ||
|  |       const actionResult = fn.apply(this, actionArgs); | ||
|  |       // Remember result in case it is async. Assume parseAsync getting called on root.
 | ||
|  |       let rootCommand = this; | ||
|  |       while (rootCommand.parent) { | ||
|  |         rootCommand = rootCommand.parent; | ||
|  |       } | ||
|  |       rootCommand._actionResults.push(actionResult); | ||
|  |     }; | ||
|  |     this._actionHandler = listener; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Internal implementation shared by .option() and .requiredOption() | ||
|  |    * | ||
|  |    * @param {Object} config | ||
|  |    * @param {string} flags | ||
|  |    * @param {string} description | ||
|  |    * @param {Function|*} [fn] - custom option processing function or default vaue | ||
|  |    * @param {*} [defaultValue] | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _optionEx(config, flags, description, fn, defaultValue) { | ||
|  |     const option = new Option(flags, description); | ||
|  |     const oname = option.name(); | ||
|  |     const name = option.attributeName(); | ||
|  |     option.mandatory = !!config.mandatory; | ||
|  | 
 | ||
|  |     // default as 3rd arg
 | ||
|  |     if (typeof fn !== 'function') { | ||
|  |       if (fn instanceof RegExp) { | ||
|  |         // This is a bit simplistic (especially no error messages), and probably better handled by caller using custom option processing.
 | ||
|  |         // No longer documented in README, but still present for backwards compatibility.
 | ||
|  |         const regex = fn; | ||
|  |         fn = (val, def) => { | ||
|  |           const m = regex.exec(val); | ||
|  |           return m ? m[0] : def; | ||
|  |         }; | ||
|  |       } else { | ||
|  |         defaultValue = fn; | ||
|  |         fn = null; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // preassign default value for --no-*, [optional], <required>, or plain flag if boolean value
 | ||
|  |     if (option.negate || option.optional || option.required || typeof defaultValue === 'boolean') { | ||
|  |       // when --no-foo we make sure default is true, unless a --foo option is already defined
 | ||
|  |       if (option.negate) { | ||
|  |         const positiveLongFlag = option.long.replace(/^--no-/, '--'); | ||
|  |         defaultValue = this._findOption(positiveLongFlag) ? this._getOptionValue(name) : true; | ||
|  |       } | ||
|  |       // preassign only if we have a default
 | ||
|  |       if (defaultValue !== undefined) { | ||
|  |         this._setOptionValue(name, defaultValue); | ||
|  |         option.defaultValue = defaultValue; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     // register the option
 | ||
|  |     this.options.push(option); | ||
|  | 
 | ||
|  |     // when it's passed assign the value
 | ||
|  |     // and conditionally invoke the callback
 | ||
|  |     this.on('option:' + oname, (val) => { | ||
|  |       // coercion
 | ||
|  |       if (val !== null && fn) { | ||
|  |         val = fn(val, this._getOptionValue(name) === undefined ? defaultValue : this._getOptionValue(name)); | ||
|  |       } | ||
|  | 
 | ||
|  |       // unassigned or boolean value
 | ||
|  |       if (typeof this._getOptionValue(name) === 'boolean' || typeof this._getOptionValue(name) === 'undefined') { | ||
|  |         // if no value, negate false, and we have a default, then use it!
 | ||
|  |         if (val == null) { | ||
|  |           this._setOptionValue(name, option.negate | ||
|  |             ? false | ||
|  |             : defaultValue || true); | ||
|  |         } else { | ||
|  |           this._setOptionValue(name, val); | ||
|  |         } | ||
|  |       } else if (val !== null) { | ||
|  |         // reassign
 | ||
|  |         this._setOptionValue(name, option.negate ? false : val); | ||
|  |       } | ||
|  |     }); | ||
|  | 
 | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Define option with `flags`, `description` and optional | ||
|  |    * coercion `fn`. | ||
|  |    * | ||
|  |    * The `flags` string should contain both the short and long flags, | ||
|  |    * separated by comma, a pipe or space. The following are all valid | ||
|  |    * all will output this way when `--help` is used. | ||
|  |    * | ||
|  |    *    "-p, --pepper" | ||
|  |    *    "-p|--pepper" | ||
|  |    *    "-p --pepper" | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *     // simple boolean defaulting to undefined
 | ||
|  |    *     program.option('-p, --pepper', 'add pepper'); | ||
|  |    * | ||
|  |    *     program.pepper | ||
|  |    *     // => undefined
 | ||
|  |    * | ||
|  |    *     --pepper | ||
|  |    *     program.pepper | ||
|  |    *     // => true
 | ||
|  |    * | ||
|  |    *     // simple boolean defaulting to true (unless non-negated option is also defined)
 | ||
|  |    *     program.option('-C, --no-cheese', 'remove cheese'); | ||
|  |    * | ||
|  |    *     program.cheese | ||
|  |    *     // => true
 | ||
|  |    * | ||
|  |    *     --no-cheese | ||
|  |    *     program.cheese | ||
|  |    *     // => false
 | ||
|  |    * | ||
|  |    *     // required argument
 | ||
|  |    *     program.option('-C, --chdir <path>', 'change the working directory'); | ||
|  |    * | ||
|  |    *     --chdir /tmp | ||
|  |    *     program.chdir | ||
|  |    *     // => "/tmp"
 | ||
|  |    * | ||
|  |    *     // optional argument
 | ||
|  |    *     program.option('-c, --cheese [type]', 'add cheese [marble]'); | ||
|  |    * | ||
|  |    * @param {string} flags | ||
|  |    * @param {string} description | ||
|  |    * @param {Function|*} [fn] - custom option processing function or default vaue | ||
|  |    * @param {*} [defaultValue] | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   option(flags, description, fn, defaultValue) { | ||
|  |     return this._optionEx({}, flags, description, fn, defaultValue); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /* | ||
|  |   * Add a required option which must have a value after parsing. This usually means | ||
|  |   * the option must be specified on the command line. (Otherwise the same as .option().) | ||
|  |   * | ||
|  |   * The `flags` string should contain both the short and long flags, separated by comma, a pipe or space. | ||
|  |   * | ||
|  |   * @param {string} flags | ||
|  |   * @param {string} description | ||
|  |   * @param {Function|*} [fn] - custom option processing function or default vaue | ||
|  |   * @param {*} [defaultValue] | ||
|  |   * @return {Command} `this` command for chaining | ||
|  |   * @api public | ||
|  |   */ | ||
|  | 
 | ||
|  |   requiredOption(flags, description, fn, defaultValue) { | ||
|  |     return this._optionEx({ mandatory: true }, flags, description, fn, defaultValue); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Allow unknown options on the command line. | ||
|  |    * | ||
|  |    * @param {Boolean} [arg] - if `true` or omitted, no error will be thrown | ||
|  |    * for unknown options. | ||
|  |    * @api public | ||
|  |    */ | ||
|  |   allowUnknownOption(arg) { | ||
|  |     this._allowUnknownOption = (arg === undefined) || arg; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |     * Whether to store option values as properties on command object, | ||
|  |     * or store separately (specify false). In both cases the option values can be accessed using .opts(). | ||
|  |     * | ||
|  |     * @param {boolean} value | ||
|  |     * @return {Command} `this` command for chaining | ||
|  |     * @api public | ||
|  |     */ | ||
|  | 
 | ||
|  |   storeOptionsAsProperties(value) { | ||
|  |     this._storeOptionsAsProperties = (value === undefined) || value; | ||
|  |     if (this.options.length) { | ||
|  |       throw new Error('call .storeOptionsAsProperties() before adding options'); | ||
|  |     } | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |     * Whether to pass command to action handler, | ||
|  |     * or just the options (specify false). | ||
|  |     * | ||
|  |     * @param {boolean} value | ||
|  |     * @return {Command} `this` command for chaining | ||
|  |     * @api public | ||
|  |     */ | ||
|  | 
 | ||
|  |   passCommandToAction(value) { | ||
|  |     this._passCommandToAction = (value === undefined) || value; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Store option value | ||
|  |    * | ||
|  |    * @param {string} key | ||
|  |    * @param {Object} value | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _setOptionValue(key, value) { | ||
|  |     if (this._storeOptionsAsProperties) { | ||
|  |       this[key] = value; | ||
|  |     } else { | ||
|  |       this._optionValues[key] = value; | ||
|  |     } | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Retrieve option value | ||
|  |    * | ||
|  |    * @param {string} key | ||
|  |    * @return {Object} value | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _getOptionValue(key) { | ||
|  |     if (this._storeOptionsAsProperties) { | ||
|  |       return this[key]; | ||
|  |     } | ||
|  |     return this._optionValues[key]; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Parse `argv`, setting options and invoking commands when defined. | ||
|  |    * | ||
|  |    * The default expectation is that the arguments are from node and have the application as argv[0] | ||
|  |    * and the script being run in argv[1], with user parameters after that. | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *      program.parse(process.argv); | ||
|  |    *      program.parse(); // implicitly use process.argv and auto-detect node vs electron conventions
 | ||
|  |    *      program.parse(my-args, { from: 'user' }); // just user supplied arguments, nothing special about argv[0]
 | ||
|  |    * | ||
|  |    * @param {string[]} [argv] - optional, defaults to process.argv | ||
|  |    * @param {Object} [parseOptions] - optionally specify style of options with from: node/user/electron | ||
|  |    * @param {string} [parseOptions.from] - where the args are from: 'node', 'user', 'electron' | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   parse(argv, parseOptions) { | ||
|  |     if (argv !== undefined && !Array.isArray(argv)) { | ||
|  |       throw new Error('first parameter to parse must be array or undefined'); | ||
|  |     } | ||
|  |     parseOptions = parseOptions || {}; | ||
|  | 
 | ||
|  |     // Default to using process.argv
 | ||
|  |     if (argv === undefined) { | ||
|  |       argv = process.argv; | ||
|  |       // @ts-ignore
 | ||
|  |       if (process.versions && process.versions.electron) { | ||
|  |         parseOptions.from = 'electron'; | ||
|  |       } | ||
|  |     } | ||
|  |     this.rawArgs = argv.slice(); | ||
|  | 
 | ||
|  |     // make it a little easier for callers by supporting various argv conventions
 | ||
|  |     let userArgs; | ||
|  |     switch (parseOptions.from) { | ||
|  |       case undefined: | ||
|  |       case 'node': | ||
|  |         this._scriptPath = argv[1]; | ||
|  |         userArgs = argv.slice(2); | ||
|  |         break; | ||
|  |       case 'electron': | ||
|  |         // @ts-ignore
 | ||
|  |         if (process.defaultApp) { | ||
|  |           this._scriptPath = argv[1]; | ||
|  |           userArgs = argv.slice(2); | ||
|  |         } else { | ||
|  |           userArgs = argv.slice(1); | ||
|  |         } | ||
|  |         break; | ||
|  |       case 'user': | ||
|  |         userArgs = argv.slice(0); | ||
|  |         break; | ||
|  |       default: | ||
|  |         throw new Error(`unexpected parse option { from: '${parseOptions.from}' }`); | ||
|  |     } | ||
|  |     if (!this._scriptPath && process.mainModule) { | ||
|  |       this._scriptPath = process.mainModule.filename; | ||
|  |     } | ||
|  | 
 | ||
|  |     // Guess name, used in usage in help.
 | ||
|  |     this._name = this._name || (this._scriptPath && path.basename(this._scriptPath, path.extname(this._scriptPath))); | ||
|  | 
 | ||
|  |     // Let's go!
 | ||
|  |     this._parseCommand([], userArgs); | ||
|  | 
 | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Parse `argv`, setting options and invoking commands when defined. | ||
|  |    * | ||
|  |    * Use parseAsync instead of parse if any of your action handlers are async. Returns a Promise. | ||
|  |    * | ||
|  |    * The default expectation is that the arguments are from node and have the application as argv[0] | ||
|  |    * and the script being run in argv[1], with user parameters after that. | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *      program.parseAsync(process.argv); | ||
|  |    *      program.parseAsync(); // implicitly use process.argv and auto-detect node vs electron conventions
 | ||
|  |    *      program.parseAsync(my-args, { from: 'user' }); // just user supplied arguments, nothing special about argv[0]
 | ||
|  |    * | ||
|  |    * @param {string[]} [argv] | ||
|  |    * @param {Object} [parseOptions] | ||
|  |    * @param {string} parseOptions.from - where the args are from: 'node', 'user', 'electron' | ||
|  |    * @return {Promise} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   parseAsync(argv, parseOptions) { | ||
|  |     this.parse(argv, parseOptions); | ||
|  |     return Promise.all(this._actionResults).then(() => this); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Execute a sub-command executable. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _executeSubCommand(subcommand, args) { | ||
|  |     args = args.slice(); | ||
|  |     let launchWithNode = false; // Use node for source targets so do not need to get permissions correct, and on Windows.
 | ||
|  |     const sourceExt = ['.js', '.ts', '.mjs']; | ||
|  | 
 | ||
|  |     // Not checking for help first. Unlikely to have mandatory and executable, and can't robustly test for help flags in external command.
 | ||
|  |     this._checkForMissingMandatoryOptions(); | ||
|  | 
 | ||
|  |     // Want the entry script as the reference for command name and directory for searching for other files.
 | ||
|  |     const scriptPath = this._scriptPath; | ||
|  | 
 | ||
|  |     let baseDir; | ||
|  |     try { | ||
|  |       const resolvedLink = fs.realpathSync(scriptPath); | ||
|  |       baseDir = path.dirname(resolvedLink); | ||
|  |     } catch (e) { | ||
|  |       baseDir = '.'; // dummy, probably not going to find executable!
 | ||
|  |     } | ||
|  | 
 | ||
|  |     // name of the subcommand, like `pm-install`
 | ||
|  |     let bin = path.basename(scriptPath, path.extname(scriptPath)) + '-' + subcommand._name; | ||
|  |     if (subcommand._executableFile) { | ||
|  |       bin = subcommand._executableFile; | ||
|  |     } | ||
|  | 
 | ||
|  |     const localBin = path.join(baseDir, bin); | ||
|  |     if (fs.existsSync(localBin)) { | ||
|  |       // prefer local `./<bin>` to bin in the $PATH
 | ||
|  |       bin = localBin; | ||
|  |     } else { | ||
|  |       // Look for source files.
 | ||
|  |       sourceExt.forEach((ext) => { | ||
|  |         if (fs.existsSync(`${localBin}${ext}`)) { | ||
|  |           bin = `${localBin}${ext}`; | ||
|  |         } | ||
|  |       }); | ||
|  |     } | ||
|  |     launchWithNode = sourceExt.includes(path.extname(bin)); | ||
|  | 
 | ||
|  |     let proc; | ||
|  |     if (process.platform !== 'win32') { | ||
|  |       if (launchWithNode) { | ||
|  |         args.unshift(bin); | ||
|  |         // add executable arguments to spawn
 | ||
|  |         args = incrementNodeInspectorPort(process.execArgv).concat(args); | ||
|  | 
 | ||
|  |         proc = spawn(process.argv[0], args, { stdio: 'inherit' }); | ||
|  |       } else { | ||
|  |         proc = spawn(bin, args, { stdio: 'inherit' }); | ||
|  |       } | ||
|  |     } else { | ||
|  |       args.unshift(bin); | ||
|  |       // add executable arguments to spawn
 | ||
|  |       args = incrementNodeInspectorPort(process.execArgv).concat(args); | ||
|  |       proc = spawn(process.execPath, args, { stdio: 'inherit' }); | ||
|  |     } | ||
|  | 
 | ||
|  |     const signals = ['SIGUSR1', 'SIGUSR2', 'SIGTERM', 'SIGINT', 'SIGHUP']; | ||
|  |     signals.forEach((signal) => { | ||
|  |       // @ts-ignore
 | ||
|  |       process.on(signal, () => { | ||
|  |         if (proc.killed === false && proc.exitCode === null) { | ||
|  |           proc.kill(signal); | ||
|  |         } | ||
|  |       }); | ||
|  |     }); | ||
|  | 
 | ||
|  |     // By default terminate process when spawned process terminates.
 | ||
|  |     // Suppressing the exit if exitCallback defined is a bit messy and of limited use, but does allow process to stay running!
 | ||
|  |     const exitCallback = this._exitCallback; | ||
|  |     if (!exitCallback) { | ||
|  |       proc.on('close', process.exit.bind(process)); | ||
|  |     } else { | ||
|  |       proc.on('close', () => { | ||
|  |         exitCallback(new CommanderError(process.exitCode || 0, 'commander.executeSubCommandAsync', '(close)')); | ||
|  |       }); | ||
|  |     } | ||
|  |     proc.on('error', (err) => { | ||
|  |       // @ts-ignore
 | ||
|  |       if (err.code === 'ENOENT') { | ||
|  |         const executableMissing = `'${bin}' does not exist
 | ||
|  |  - if '${subcommand._name}' 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`;
 | ||
|  |         throw new Error(executableMissing); | ||
|  |       // @ts-ignore
 | ||
|  |       } else if (err.code === 'EACCES') { | ||
|  |         throw new Error(`'${bin}' not executable`); | ||
|  |       } | ||
|  |       if (!exitCallback) { | ||
|  |         process.exit(1); | ||
|  |       } else { | ||
|  |         const wrappedError = new CommanderError(1, 'commander.executeSubCommandAsync', '(error)'); | ||
|  |         wrappedError.nestedError = err; | ||
|  |         exitCallback(wrappedError); | ||
|  |       } | ||
|  |     }); | ||
|  | 
 | ||
|  |     // Store the reference to the child process
 | ||
|  |     this.runningCommand = proc; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * @api private | ||
|  |    */ | ||
|  |   _dispatchSubcommand(commandName, operands, unknown) { | ||
|  |     const subCommand = this._findCommand(commandName); | ||
|  |     if (!subCommand) this._helpAndError(); | ||
|  | 
 | ||
|  |     if (subCommand._executableHandler) { | ||
|  |       this._executeSubCommand(subCommand, operands.concat(unknown)); | ||
|  |     } else { | ||
|  |       subCommand._parseCommand(operands, unknown); | ||
|  |     } | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Process arguments in context of this command. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _parseCommand(operands, unknown) { | ||
|  |     const parsed = this.parseOptions(unknown); | ||
|  |     operands = operands.concat(parsed.operands); | ||
|  |     unknown = parsed.unknown; | ||
|  |     this.args = operands.concat(unknown); | ||
|  | 
 | ||
|  |     if (operands && this._findCommand(operands[0])) { | ||
|  |       this._dispatchSubcommand(operands[0], operands.slice(1), unknown); | ||
|  |     } else if (this._lazyHasImplicitHelpCommand() && operands[0] === this._helpCommandName) { | ||
|  |       if (operands.length === 1) { | ||
|  |         this.help(); | ||
|  |       } else { | ||
|  |         this._dispatchSubcommand(operands[1], [], [this._helpLongFlag]); | ||
|  |       } | ||
|  |     } else if (this._defaultCommandName) { | ||
|  |       outputHelpIfRequested(this, unknown); // Run the help for default command from parent rather than passing to default command
 | ||
|  |       this._dispatchSubcommand(this._defaultCommandName, operands, unknown); | ||
|  |     } else { | ||
|  |       if (this.commands.length && this.args.length === 0 && !this._actionHandler && !this._defaultCommandName) { | ||
|  |         // probaby missing subcommand and no handler, user needs help
 | ||
|  |         this._helpAndError(); | ||
|  |       } | ||
|  | 
 | ||
|  |       outputHelpIfRequested(this, parsed.unknown); | ||
|  |       this._checkForMissingMandatoryOptions(); | ||
|  |       if (parsed.unknown.length > 0) { | ||
|  |         this.unknownOption(parsed.unknown[0]); | ||
|  |       } | ||
|  | 
 | ||
|  |       if (this._actionHandler) { | ||
|  |         const args = this.args.slice(); | ||
|  |         this._args.forEach((arg, i) => { | ||
|  |           if (arg.required && args[i] == null) { | ||
|  |             this.missingArgument(arg.name); | ||
|  |           } else if (arg.variadic) { | ||
|  |             args[i] = args.splice(i); | ||
|  |           } | ||
|  |         }); | ||
|  | 
 | ||
|  |         this._actionHandler(args); | ||
|  |         this.emit('command:' + this.name(), operands, unknown); | ||
|  |       } else if (operands.length) { | ||
|  |         if (this._findCommand('*')) { | ||
|  |           this._dispatchSubcommand('*', operands, unknown); | ||
|  |         } else if (this.listenerCount('command:*')) { | ||
|  |           this.emit('command:*', operands, unknown); | ||
|  |         } else if (this.commands.length) { | ||
|  |           this.unknownCommand(); | ||
|  |         } | ||
|  |       } else if (this.commands.length) { | ||
|  |         // This command has subcommands and nothing hooked up at this level, so display help.
 | ||
|  |         this._helpAndError(); | ||
|  |       } else { | ||
|  |         // fall through for caller to handle after calling .parse()
 | ||
|  |       } | ||
|  |     } | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Find matching command. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  |   _findCommand(name) { | ||
|  |     if (!name) return undefined; | ||
|  |     return this.commands.find(cmd => cmd._name === name || cmd._aliases.includes(name)); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return an option matching `arg` if any. | ||
|  |    * | ||
|  |    * @param {string} arg | ||
|  |    * @return {Option} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _findOption(arg) { | ||
|  |     return this.options.find(option => option.is(arg)); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Display an error message if a mandatory option does not have a value. | ||
|  |    * Lazy calling after checking for help flags from leaf subcommand. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _checkForMissingMandatoryOptions() { | ||
|  |     // Walk up hierarchy so can call in subcommand after checking for displaying help.
 | ||
|  |     for (let cmd = this; cmd; cmd = cmd.parent) { | ||
|  |       cmd.options.forEach((anOption) => { | ||
|  |         if (anOption.mandatory && (cmd._getOptionValue(anOption.attributeName()) === undefined)) { | ||
|  |           cmd.missingMandatoryOptionValue(anOption); | ||
|  |         } | ||
|  |       }); | ||
|  |     } | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Parse options from `argv` removing known options, | ||
|  |    * and return argv split into operands and unknown arguments. | ||
|  |    * | ||
|  |    * Examples: | ||
|  |    * | ||
|  |    *    argv => operands, unknown | ||
|  |    *    --known kkk op => [op], [] | ||
|  |    *    op --known kkk => [op], [] | ||
|  |    *    sub --unknown uuu op => [sub], [--unknown uuu op] | ||
|  |    *    sub -- --unknown uuu op => [sub --unknown uuu op], [] | ||
|  |    * | ||
|  |    * @param {String[]} argv | ||
|  |    * @return {{operands: String[], unknown: String[]}} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   parseOptions(argv) { | ||
|  |     const operands = []; // operands, not options or values
 | ||
|  |     const unknown = []; // first unknown option and remaining unknown args
 | ||
|  |     let dest = operands; | ||
|  |     const args = argv.slice(); | ||
|  | 
 | ||
|  |     function maybeOption(arg) { | ||
|  |       return arg.length > 1 && arg[0] === '-'; | ||
|  |     } | ||
|  | 
 | ||
|  |     // parse options
 | ||
|  |     while (args.length) { | ||
|  |       const arg = args.shift(); | ||
|  | 
 | ||
|  |       // literal
 | ||
|  |       if (arg === '--') { | ||
|  |         if (dest === unknown) dest.push(arg); | ||
|  |         dest.push(...args); | ||
|  |         break; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (maybeOption(arg)) { | ||
|  |         const option = this._findOption(arg); | ||
|  |         // recognised option, call listener to assign value with possible custom processing
 | ||
|  |         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; | ||
|  |             // historical behaviour is optional value is following arg unless an option
 | ||
|  |             if (args.length > 0 && !maybeOption(args[0])) { | ||
|  |               value = args.shift(); | ||
|  |             } | ||
|  |             this.emit(`option:${option.name()}`, value); | ||
|  |           } else { // boolean flag
 | ||
|  |             this.emit(`option:${option.name()}`); | ||
|  |           } | ||
|  |           continue; | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       // Look for combo options following single dash, eat first one if known.
 | ||
|  |       if (arg.length > 2 && arg[0] === '-' && arg[1] !== '-') { | ||
|  |         const option = this._findOption(`-${arg[1]}`); | ||
|  |         if (option) { | ||
|  |           if (option.required || option.optional) { | ||
|  |             // option with value following in same argument
 | ||
|  |             this.emit(`option:${option.name()}`, arg.slice(2)); | ||
|  |           } else { | ||
|  |             // boolean option, emit and put back remainder of arg for further processing
 | ||
|  |             this.emit(`option:${option.name()}`); | ||
|  |             args.unshift(`-${arg.slice(2)}`); | ||
|  |           } | ||
|  |           continue; | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       // Look for known long flag with value, like --foo=bar
 | ||
|  |       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; | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       // looks like an option but unknown, unknowns from here
 | ||
|  |       if (arg.length > 1 && arg[0] === '-') { | ||
|  |         dest = unknown; | ||
|  |       } | ||
|  | 
 | ||
|  |       // add arg
 | ||
|  |       dest.push(arg); | ||
|  |     } | ||
|  | 
 | ||
|  |     return { operands, unknown }; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return an object containing options as key-value pairs | ||
|  |    * | ||
|  |    * @return {Object} | ||
|  |    * @api public | ||
|  |    */ | ||
|  |   opts() { | ||
|  |     if (this._storeOptionsAsProperties) { | ||
|  |       // Preserve original behaviour so backwards compatible when still using properties
 | ||
|  |       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; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Argument `name` is missing. | ||
|  |    * | ||
|  |    * @param {string} name | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   missingArgument(name) { | ||
|  |     const message = `error: missing required argument '${name}'`; | ||
|  |     console.error(message); | ||
|  |     this._exit(1, 'commander.missingArgument', message); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * `Option` is missing an argument, but received `flag` or nothing. | ||
|  |    * | ||
|  |    * @param {Option} option | ||
|  |    * @param {string} [flag] | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   optionMissingArgument(option, flag) { | ||
|  |     let message; | ||
|  |     if (flag) { | ||
|  |       message = `error: option '${option.flags}' argument missing, got '${flag}'`; | ||
|  |     } else { | ||
|  |       message = `error: option '${option.flags}' argument missing`; | ||
|  |     } | ||
|  |     console.error(message); | ||
|  |     this._exit(1, 'commander.optionMissingArgument', message); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * `Option` does not have a value, and is a mandatory option. | ||
|  |    * | ||
|  |    * @param {Option} option | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   missingMandatoryOptionValue(option) { | ||
|  |     const message = `error: required option '${option.flags}' not specified`; | ||
|  |     console.error(message); | ||
|  |     this._exit(1, 'commander.missingMandatoryOptionValue', message); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Unknown option `flag`. | ||
|  |    * | ||
|  |    * @param {string} flag | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   unknownOption(flag) { | ||
|  |     if (this._allowUnknownOption) return; | ||
|  |     const message = `error: unknown option '${flag}'`; | ||
|  |     console.error(message); | ||
|  |     this._exit(1, 'commander.unknownOption', message); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Unknown command. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   unknownCommand() { | ||
|  |     const partCommands = [this.name()]; | ||
|  |     for (let parentCmd = this.parent; parentCmd; parentCmd = parentCmd.parent) { | ||
|  |       partCommands.unshift(parentCmd.name()); | ||
|  |     } | ||
|  |     const fullCommand = partCommands.join(' '); | ||
|  |     const message = `error: unknown command '${this.args[0]}'. See '${fullCommand} ${this._helpLongFlag}'.`; | ||
|  |     console.error(message); | ||
|  |     this._exit(1, 'commander.unknownCommand', message); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Set the program version to `str`. | ||
|  |    * | ||
|  |    * This method auto-registers the "-V, --version" flag | ||
|  |    * which will print the version number when passed. | ||
|  |    * | ||
|  |    * You can optionally supply the  flags and description to override the defaults. | ||
|  |    * | ||
|  |    * @param {string} str | ||
|  |    * @param {string} [flags] | ||
|  |    * @param {string} [description] | ||
|  |    * @return {this | string} `this` command for chaining, or version string if no arguments | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   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 = new Option(flags, description); | ||
|  |     this._versionOptionName = versionOption.long.substr(2) || 'version'; | ||
|  |     this.options.push(versionOption); | ||
|  |     this.on('option:' + this._versionOptionName, () => { | ||
|  |       process.stdout.write(str + '\n'); | ||
|  |       this._exit(0, 'commander.version', str); | ||
|  |     }); | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Set the description to `str`. | ||
|  |    * | ||
|  |    * @param {string} str | ||
|  |    * @param {Object} [argsDescription] | ||
|  |    * @return {string|Command} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   description(str, argsDescription) { | ||
|  |     if (str === undefined && argsDescription === undefined) return this._description; | ||
|  |     this._description = str; | ||
|  |     this._argsDescription = argsDescription; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Set an alias for the command. | ||
|  |    * | ||
|  |    * You may call more than once to add multiple aliases. Only the first alias is shown in the auto-generated help. | ||
|  |    * | ||
|  |    * @param {string} [alias] | ||
|  |    * @return {string|Command} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   alias(alias) { | ||
|  |     if (alias === undefined) return this._aliases[0]; // just return first, for backwards compatibility
 | ||
|  | 
 | ||
|  |     let command = this; | ||
|  |     if (this.commands.length !== 0 && this.commands[this.commands.length - 1]._executableHandler) { | ||
|  |       // assume adding alias for last added executable subcommand, rather than this
 | ||
|  |       command = this.commands[this.commands.length - 1]; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (alias === command._name) throw new Error('Command alias can\'t be the same as its name'); | ||
|  | 
 | ||
|  |     command._aliases.push(alias); | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Set aliases for the command. | ||
|  |    * | ||
|  |    * Only the first alias is shown in the auto-generated help. | ||
|  |    * | ||
|  |    * @param {string[]} [aliases] | ||
|  |    * @return {string[]|Command} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   aliases(aliases) { | ||
|  |     // Getter for the array of aliases is the main reason for having aliases() in addition to alias().
 | ||
|  |     if (aliases === undefined) return this._aliases; | ||
|  | 
 | ||
|  |     aliases.forEach((alias) => this.alias(alias)); | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Set / get the command usage `str`. | ||
|  |    * | ||
|  |    * @param {string} [str] | ||
|  |    * @return {String|Command} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   usage(str) { | ||
|  |     if (str === undefined) { | ||
|  |       if (this._usage) return this._usage; | ||
|  | 
 | ||
|  |       const args = this._args.map((arg) => { | ||
|  |         return humanReadableArgName(arg); | ||
|  |       }); | ||
|  |       return '[options]' + | ||
|  |         (this.commands.length ? ' [command]' : '') + | ||
|  |         (this._args.length ? ' ' + args.join(' ') : ''); | ||
|  |     } | ||
|  | 
 | ||
|  |     this._usage = str; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Get or set the name of the command | ||
|  |    * | ||
|  |    * @param {string} [str] | ||
|  |    * @return {String|Command} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   name(str) { | ||
|  |     if (str === undefined) return this._name; | ||
|  |     this._name = str; | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return prepared commands. | ||
|  |    * | ||
|  |    * @return {Array} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   prepareCommands() { | ||
|  |     const commandDetails = this.commands.filter((cmd) => { | ||
|  |       return !cmd._hidden; | ||
|  |     }).map((cmd) => { | ||
|  |       const args = cmd._args.map((arg) => { | ||
|  |         return humanReadableArgName(arg); | ||
|  |       }).join(' '); | ||
|  | 
 | ||
|  |       return [ | ||
|  |         cmd._name + | ||
|  |           (cmd._aliases[0] ? '|' + cmd._aliases[0] : '') + | ||
|  |           (cmd.options.length ? ' [options]' : '') + | ||
|  |           (args ? ' ' + args : ''), | ||
|  |         cmd._description | ||
|  |       ]; | ||
|  |     }); | ||
|  | 
 | ||
|  |     if (this._lazyHasImplicitHelpCommand()) { | ||
|  |       commandDetails.push([this._helpCommandnameAndArgs, this._helpCommandDescription]); | ||
|  |     } | ||
|  |     return commandDetails; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return the largest command length. | ||
|  |    * | ||
|  |    * @return {number} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   largestCommandLength() { | ||
|  |     const commands = this.prepareCommands(); | ||
|  |     return commands.reduce((max, command) => { | ||
|  |       return Math.max(max, command[0].length); | ||
|  |     }, 0); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return the largest option length. | ||
|  |    * | ||
|  |    * @return {number} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   largestOptionLength() { | ||
|  |     const options = [].slice.call(this.options); | ||
|  |     options.push({ | ||
|  |       flags: this._helpFlags | ||
|  |     }); | ||
|  | 
 | ||
|  |     return options.reduce((max, option) => { | ||
|  |       return Math.max(max, option.flags.length); | ||
|  |     }, 0); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return the largest arg length. | ||
|  |    * | ||
|  |    * @return {number} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   largestArgLength() { | ||
|  |     return this._args.reduce((max, arg) => { | ||
|  |       return Math.max(max, arg.name.length); | ||
|  |     }, 0); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return the pad width. | ||
|  |    * | ||
|  |    * @return {number} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   padWidth() { | ||
|  |     let width = this.largestOptionLength(); | ||
|  |     if (this._argsDescription && this._args.length) { | ||
|  |       if (this.largestArgLength() > width) { | ||
|  |         width = this.largestArgLength(); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     if (this.commands && this.commands.length) { | ||
|  |       if (this.largestCommandLength() > width) { | ||
|  |         width = this.largestCommandLength(); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     return width; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return help for options. | ||
|  |    * | ||
|  |    * @return {string} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   optionHelp() { | ||
|  |     const width = this.padWidth(); | ||
|  |     const columns = process.stdout.columns || 80; | ||
|  |     const descriptionWidth = columns - width - 4; | ||
|  |     function padOptionDetails(flags, description) { | ||
|  |       return pad(flags, width) + '  ' + optionalWrap(description, descriptionWidth, width + 2); | ||
|  |     }; | ||
|  | 
 | ||
|  |     // Explicit options (including version)
 | ||
|  |     const help = this.options.map((option) => { | ||
|  |       const fullDesc = option.description + | ||
|  |         ((!option.negate && option.defaultValue !== undefined) ? ' (default: ' + JSON.stringify(option.defaultValue) + ')' : ''); | ||
|  |       return padOptionDetails(option.flags, fullDesc); | ||
|  |     }); | ||
|  | 
 | ||
|  |     // Implicit help
 | ||
|  |     const showShortHelpFlag = this._helpShortFlag && !this._findOption(this._helpShortFlag); | ||
|  |     const showLongHelpFlag = !this._findOption(this._helpLongFlag); | ||
|  |     if (showShortHelpFlag || showLongHelpFlag) { | ||
|  |       let helpFlags = this._helpFlags; | ||
|  |       if (!showShortHelpFlag) { | ||
|  |         helpFlags = this._helpLongFlag; | ||
|  |       } else if (!showLongHelpFlag) { | ||
|  |         helpFlags = this._helpShortFlag; | ||
|  |       } | ||
|  |       help.push(padOptionDetails(helpFlags, this._helpDescription)); | ||
|  |     } | ||
|  | 
 | ||
|  |     return help.join('\n'); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return command help documentation. | ||
|  |    * | ||
|  |    * @return {string} | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   commandHelp() { | ||
|  |     if (!this.commands.length && !this._lazyHasImplicitHelpCommand()) return ''; | ||
|  | 
 | ||
|  |     const commands = this.prepareCommands(); | ||
|  |     const width = this.padWidth(); | ||
|  | 
 | ||
|  |     const columns = process.stdout.columns || 80; | ||
|  |     const descriptionWidth = columns - width - 4; | ||
|  | 
 | ||
|  |     return [ | ||
|  |       'Commands:', | ||
|  |       commands.map((cmd) => { | ||
|  |         const desc = cmd[1] ? '  ' + cmd[1] : ''; | ||
|  |         return (desc ? pad(cmd[0], width) : cmd[0]) + optionalWrap(desc, descriptionWidth, width + 2); | ||
|  |       }).join('\n').replace(/^/gm, '  '), | ||
|  |       '' | ||
|  |     ].join('\n'); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Return program help documentation. | ||
|  |    * | ||
|  |    * @return {string} | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   helpInformation() { | ||
|  |     let desc = []; | ||
|  |     if (this._description) { | ||
|  |       desc = [ | ||
|  |         this._description, | ||
|  |         '' | ||
|  |       ]; | ||
|  | 
 | ||
|  |       const argsDescription = this._argsDescription; | ||
|  |       if (argsDescription && this._args.length) { | ||
|  |         const width = this.padWidth(); | ||
|  |         const columns = process.stdout.columns || 80; | ||
|  |         const descriptionWidth = columns - width - 5; | ||
|  |         desc.push('Arguments:'); | ||
|  |         desc.push(''); | ||
|  |         this._args.forEach((arg) => { | ||
|  |           desc.push('  ' + pad(arg.name, width) + '  ' + wrap(argsDescription[arg.name], descriptionWidth, width + 4)); | ||
|  |         }); | ||
|  |         desc.push(''); | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     let cmdName = this._name; | ||
|  |     if (this._aliases[0]) { | ||
|  |       cmdName = cmdName + '|' + this._aliases[0]; | ||
|  |     } | ||
|  |     let parentCmdNames = ''; | ||
|  |     for (let parentCmd = this.parent; parentCmd; parentCmd = parentCmd.parent) { | ||
|  |       parentCmdNames = parentCmd.name() + ' ' + parentCmdNames; | ||
|  |     } | ||
|  |     const usage = [ | ||
|  |       'Usage: ' + parentCmdNames + cmdName + ' ' + this.usage(), | ||
|  |       '' | ||
|  |     ]; | ||
|  | 
 | ||
|  |     let cmds = []; | ||
|  |     const commandHelp = this.commandHelp(); | ||
|  |     if (commandHelp) cmds = [commandHelp]; | ||
|  | 
 | ||
|  |     const options = [ | ||
|  |       'Options:', | ||
|  |       '' + this.optionHelp().replace(/^/gm, '  '), | ||
|  |       '' | ||
|  |     ]; | ||
|  | 
 | ||
|  |     return usage | ||
|  |       .concat(desc) | ||
|  |       .concat(options) | ||
|  |       .concat(cmds) | ||
|  |       .join('\n'); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Output help information for this command. | ||
|  |    * | ||
|  |    * When listener(s) are available for the helpLongFlag | ||
|  |    * those callbacks are invoked. | ||
|  |    * | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   outputHelp(cb) { | ||
|  |     if (!cb) { | ||
|  |       cb = (passthru) => { | ||
|  |         return passthru; | ||
|  |       }; | ||
|  |     } | ||
|  |     const cbOutput = cb(this.helpInformation()); | ||
|  |     if (typeof cbOutput !== 'string' && !Buffer.isBuffer(cbOutput)) { | ||
|  |       throw new Error('outputHelp callback must return a string or a Buffer'); | ||
|  |     } | ||
|  |     process.stdout.write(cbOutput); | ||
|  |     this.emit(this._helpLongFlag); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * You can pass in flags and a description to override the help | ||
|  |    * flags and help description for your command. | ||
|  |    * | ||
|  |    * @param {string} [flags] | ||
|  |    * @param {string} [description] | ||
|  |    * @return {Command} `this` command for chaining | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   helpOption(flags, description) { | ||
|  |     this._helpFlags = flags || this._helpFlags; | ||
|  |     this._helpDescription = description || this._helpDescription; | ||
|  | 
 | ||
|  |     const splitFlags = this._helpFlags.split(/[ ,|]+/); | ||
|  | 
 | ||
|  |     this._helpShortFlag = undefined; | ||
|  |     if (splitFlags.length > 1) this._helpShortFlag = splitFlags.shift(); | ||
|  | 
 | ||
|  |     this._helpLongFlag = splitFlags.shift(); | ||
|  | 
 | ||
|  |     return this; | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Output help information and exit. | ||
|  |    * | ||
|  |    * @param {Function} [cb] | ||
|  |    * @api public | ||
|  |    */ | ||
|  | 
 | ||
|  |   help(cb) { | ||
|  |     this.outputHelp(cb); | ||
|  |     // exitCode: preserving original behaviour which was calling process.exit()
 | ||
|  |     // message: do not have all displayed text available so only passing placeholder.
 | ||
|  |     this._exit(process.exitCode || 0, 'commander.help', '(outputHelp)'); | ||
|  |   }; | ||
|  | 
 | ||
|  |   /** | ||
|  |    * Output help information and exit. Display for error situations. | ||
|  |    * | ||
|  |    * @api private | ||
|  |    */ | ||
|  | 
 | ||
|  |   _helpAndError() { | ||
|  |     this.outputHelp(); | ||
|  |     // message: do not have all displayed text available so only passing placeholder.
 | ||
|  |     this._exit(1, 'commander.help', '(outputHelp)'); | ||
|  |   }; | ||
|  | }; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Expose the root command. | ||
|  |  */ | ||
|  | 
 | ||
|  | exports = module.exports = new Command(); | ||
|  | exports.program = exports; // More explicit access to global command.
 | ||
|  | 
 | ||
|  | /** | ||
|  |  * Expose classes | ||
|  |  */ | ||
|  | 
 | ||
|  | exports.Command = Command; | ||
|  | exports.Option = Option; | ||
|  | exports.CommanderError = CommanderError; | ||
|  | 
 | ||
|  | /** | ||
|  |  * Camel-case the given `flag` | ||
|  |  * | ||
|  |  * @param {string} flag | ||
|  |  * @return {string} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function camelcase(flag) { | ||
|  |   return flag.split('-').reduce((str, word) => { | ||
|  |     return str + word[0].toUpperCase() + word.slice(1); | ||
|  |   }); | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Pad `str` to `width`. | ||
|  |  * | ||
|  |  * @param {string} str | ||
|  |  * @param {number} width | ||
|  |  * @return {string} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function pad(str, width) { | ||
|  |   const len = Math.max(0, width - str.length); | ||
|  |   return str + Array(len + 1).join(' '); | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Wraps the given string with line breaks at the specified width while breaking | ||
|  |  * words and indenting every but the first line on the left. | ||
|  |  * | ||
|  |  * @param {string} str | ||
|  |  * @param {number} width | ||
|  |  * @param {number} indent | ||
|  |  * @return {string} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | function wrap(str, width, indent) { | ||
|  |   const regex = new RegExp('.{1,' + (width - 1) + '}([\\s\u200B]|$)|[^\\s\u200B]+?([\\s\u200B]|$)', 'g'); | ||
|  |   const lines = str.match(regex) || []; | ||
|  |   return lines.map((line, i) => { | ||
|  |     if (line.slice(-1) === '\n') { | ||
|  |       line = line.slice(0, line.length - 1); | ||
|  |     } | ||
|  |     return ((i > 0 && indent) ? Array(indent + 1).join(' ') : '') + line.trimRight(); | ||
|  |   }).join('\n'); | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Optionally wrap the given str to a max width of width characters per line | ||
|  |  * while indenting with indent spaces. Do not wrap if insufficient width or | ||
|  |  * string is manually formatted. | ||
|  |  * | ||
|  |  * @param {string} str | ||
|  |  * @param {number} width | ||
|  |  * @param {number} indent | ||
|  |  * @return {string} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | function optionalWrap(str, width, indent) { | ||
|  |   // Detect manually wrapped and indented strings by searching for line breaks
 | ||
|  |   // followed by multiple spaces/tabs.
 | ||
|  |   if (str.match(/[\n]\s+/)) return str; | ||
|  |   // Do not wrap to narrow columns (or can end up with a word per line).
 | ||
|  |   const minWidth = 40; | ||
|  |   if (width < minWidth) return str; | ||
|  | 
 | ||
|  |   return wrap(str, width, indent); | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Output help information if help flags specified | ||
|  |  * | ||
|  |  * @param {Command} cmd - command to output help for | ||
|  |  * @param {Array} args - array of options to search for help flags | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function outputHelpIfRequested(cmd, args) { | ||
|  |   const helpOption = args.find(arg => arg === cmd._helpLongFlag || arg === cmd._helpShortFlag); | ||
|  |   if (helpOption) { | ||
|  |     cmd.outputHelp(); | ||
|  |     // (Do not have all displayed text available so only passing placeholder.)
 | ||
|  |     cmd._exit(0, 'commander.helpDisplayed', '(outputHelp)'); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Takes an argument and returns its human readable equivalent for help usage. | ||
|  |  * | ||
|  |  * @param {Object} arg | ||
|  |  * @return {string} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function humanReadableArgName(arg) { | ||
|  |   const nameOutput = arg.name + (arg.variadic === true ? '...' : ''); | ||
|  | 
 | ||
|  |   return arg.required | ||
|  |     ? '<' + nameOutput + '>' | ||
|  |     : '[' + nameOutput + ']'; | ||
|  | } | ||
|  | 
 | ||
|  | /** | ||
|  |  * Scan arguments and increment port number for inspect calls (to avoid conflicts when spawning new command). | ||
|  |  * | ||
|  |  * @param {string[]} args - array of arguments from node.execArgv | ||
|  |  * @returns {string[]} | ||
|  |  * @api private | ||
|  |  */ | ||
|  | 
 | ||
|  | function incrementNodeInspectorPort(args) { | ||
|  |   // Testing for these options:
 | ||
|  |   //  --inspect[=[host:]port]
 | ||
|  |   //  --inspect-brk[=[host:]port]
 | ||
|  |   //  --inspect-port=[host:]port
 | ||
|  |   return args.map((arg) => { | ||
|  |     let result = arg; | ||
|  |     if (arg.indexOf('--inspect') === 0) { | ||
|  |       let debugOption; | ||
|  |       let debugHost = '127.0.0.1'; | ||
|  |       let debugPort = '9229'; | ||
|  |       let match; | ||
|  |       if ((match = arg.match(/^(--inspect(-brk)?)$/)) !== null) { | ||
|  |         // e.g. --inspect
 | ||
|  |         debugOption = match[1]; | ||
|  |       } else if ((match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+)$/)) !== null) { | ||
|  |         debugOption = match[1]; | ||
|  |         if (/^\d+$/.test(match[3])) { | ||
|  |           // e.g. --inspect=1234
 | ||
|  |           debugPort = match[3]; | ||
|  |         } else { | ||
|  |           // e.g. --inspect=localhost
 | ||
|  |           debugHost = match[3]; | ||
|  |         } | ||
|  |       } else if ((match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+):(\d+)$/)) !== null) { | ||
|  |         // e.g. --inspect=localhost:1234
 | ||
|  |         debugOption = match[1]; | ||
|  |         debugHost = match[3]; | ||
|  |         debugPort = match[4]; | ||
|  |       } | ||
|  | 
 | ||
|  |       if (debugOption && debugPort !== '0') { | ||
|  |         result = `${debugOption}=${debugHost}:${parseInt(debugPort) + 1}`; | ||
|  |       } | ||
|  |     } | ||
|  |     return result; | ||
|  |   }); | ||
|  | } |