1 | /**
|
---|
2 | * Module dependencies.
|
---|
3 | */
|
---|
4 |
|
---|
5 | const EventEmitter = require('events').EventEmitter;
|
---|
6 | const childProcess = require('child_process');
|
---|
7 | const path = require('path');
|
---|
8 | const fs = require('fs');
|
---|
9 |
|
---|
10 | // @ts-check
|
---|
11 |
|
---|
12 | // Although this is a class, methods are static in style to allow override using subclass or just functions.
|
---|
13 | class Help {
|
---|
14 | constructor() {
|
---|
15 | this.helpWidth = undefined;
|
---|
16 | this.sortSubcommands = false;
|
---|
17 | this.sortOptions = false;
|
---|
18 | }
|
---|
19 |
|
---|
20 | /**
|
---|
21 | * Get an array of the visible subcommands. Includes a placeholder for the implicit help command, if there is one.
|
---|
22 | *
|
---|
23 | * @param {Command} cmd
|
---|
24 | * @returns {Command[]}
|
---|
25 | */
|
---|
26 |
|
---|
27 | visibleCommands(cmd) {
|
---|
28 | const visibleCommands = cmd.commands.filter(cmd => !cmd._hidden);
|
---|
29 | if (cmd._hasImplicitHelpCommand()) {
|
---|
30 | // Create a command matching the implicit help command.
|
---|
31 | const args = cmd._helpCommandnameAndArgs.split(/ +/);
|
---|
32 | const helpCommand = cmd.createCommand(args.shift())
|
---|
33 | .helpOption(false);
|
---|
34 | helpCommand.description(cmd._helpCommandDescription);
|
---|
35 | helpCommand._parseExpectedArgs(args);
|
---|
36 | visibleCommands.push(helpCommand);
|
---|
37 | }
|
---|
38 | if (this.sortSubcommands) {
|
---|
39 | visibleCommands.sort((a, b) => {
|
---|
40 | return a.name().localeCompare(b.name());
|
---|
41 | });
|
---|
42 | }
|
---|
43 | return visibleCommands;
|
---|
44 | }
|
---|
45 |
|
---|
46 | /**
|
---|
47 | * Get an array of the visible options. Includes a placeholder for the implicit help option, if there is one.
|
---|
48 | *
|
---|
49 | * @param {Command} cmd
|
---|
50 | * @returns {Option[]}
|
---|
51 | */
|
---|
52 |
|
---|
53 | visibleOptions(cmd) {
|
---|
54 | const visibleOptions = cmd.options.filter((option) => !option.hidden);
|
---|
55 | // Implicit help
|
---|
56 | const showShortHelpFlag = cmd._hasHelpOption && cmd._helpShortFlag && !cmd._findOption(cmd._helpShortFlag);
|
---|
57 | const showLongHelpFlag = cmd._hasHelpOption && !cmd._findOption(cmd._helpLongFlag);
|
---|
58 | if (showShortHelpFlag || showLongHelpFlag) {
|
---|
59 | let helpOption;
|
---|
60 | if (!showShortHelpFlag) {
|
---|
61 | helpOption = cmd.createOption(cmd._helpLongFlag, cmd._helpDescription);
|
---|
62 | } else if (!showLongHelpFlag) {
|
---|
63 | helpOption = cmd.createOption(cmd._helpShortFlag, cmd._helpDescription);
|
---|
64 | } else {
|
---|
65 | helpOption = cmd.createOption(cmd._helpFlags, cmd._helpDescription);
|
---|
66 | }
|
---|
67 | visibleOptions.push(helpOption);
|
---|
68 | }
|
---|
69 | if (this.sortOptions) {
|
---|
70 | const getSortKey = (option) => {
|
---|
71 | // WYSIWYG for order displayed in help with short before long, no special handling for negated.
|
---|
72 | return option.short ? option.short.replace(/^-/, '') : option.long.replace(/^--/, '');
|
---|
73 | };
|
---|
74 | visibleOptions.sort((a, b) => {
|
---|
75 | return getSortKey(a).localeCompare(getSortKey(b));
|
---|
76 | });
|
---|
77 | }
|
---|
78 | return visibleOptions;
|
---|
79 | }
|
---|
80 |
|
---|
81 | /**
|
---|
82 | * Get an array of the arguments which have descriptions.
|
---|
83 | *
|
---|
84 | * @param {Command} cmd
|
---|
85 | * @returns {{ term: string, description:string }[]}
|
---|
86 | */
|
---|
87 |
|
---|
88 | visibleArguments(cmd) {
|
---|
89 | if (cmd._argsDescription && cmd._args.length) {
|
---|
90 | return cmd._args.map((argument) => {
|
---|
91 | return { term: argument.name, description: cmd._argsDescription[argument.name] || '' };
|
---|
92 | }, 0);
|
---|
93 | }
|
---|
94 | return [];
|
---|
95 | }
|
---|
96 |
|
---|
97 | /**
|
---|
98 | * Get the command term to show in the list of subcommands.
|
---|
99 | *
|
---|
100 | * @param {Command} cmd
|
---|
101 | * @returns {string}
|
---|
102 | */
|
---|
103 |
|
---|
104 | subcommandTerm(cmd) {
|
---|
105 | // Legacy. Ignores custom usage string, and nested commands.
|
---|
106 | const args = cmd._args.map(arg => humanReadableArgName(arg)).join(' ');
|
---|
107 | return cmd._name +
|
---|
108 | (cmd._aliases[0] ? '|' + cmd._aliases[0] : '') +
|
---|
109 | (cmd.options.length ? ' [options]' : '') + // simplistic check for non-help option
|
---|
110 | (args ? ' ' + args : '');
|
---|
111 | }
|
---|
112 |
|
---|
113 | /**
|
---|
114 | * Get the option term to show in the list of options.
|
---|
115 | *
|
---|
116 | * @param {Option} option
|
---|
117 | * @returns {string}
|
---|
118 | */
|
---|
119 |
|
---|
120 | optionTerm(option) {
|
---|
121 | return option.flags;
|
---|
122 | }
|
---|
123 |
|
---|
124 | /**
|
---|
125 | * Get the longest command term length.
|
---|
126 | *
|
---|
127 | * @param {Command} cmd
|
---|
128 | * @param {Help} helper
|
---|
129 | * @returns {number}
|
---|
130 | */
|
---|
131 |
|
---|
132 | longestSubcommandTermLength(cmd, helper) {
|
---|
133 | return helper.visibleCommands(cmd).reduce((max, command) => {
|
---|
134 | return Math.max(max, helper.subcommandTerm(command).length);
|
---|
135 | }, 0);
|
---|
136 | };
|
---|
137 |
|
---|
138 | /**
|
---|
139 | * Get the longest option term length.
|
---|
140 | *
|
---|
141 | * @param {Command} cmd
|
---|
142 | * @param {Help} helper
|
---|
143 | * @returns {number}
|
---|
144 | */
|
---|
145 |
|
---|
146 | longestOptionTermLength(cmd, helper) {
|
---|
147 | return helper.visibleOptions(cmd).reduce((max, option) => {
|
---|
148 | return Math.max(max, helper.optionTerm(option).length);
|
---|
149 | }, 0);
|
---|
150 | };
|
---|
151 |
|
---|
152 | /**
|
---|
153 | * Get the longest argument term length.
|
---|
154 | *
|
---|
155 | * @param {Command} cmd
|
---|
156 | * @param {Help} helper
|
---|
157 | * @returns {number}
|
---|
158 | */
|
---|
159 |
|
---|
160 | longestArgumentTermLength(cmd, helper) {
|
---|
161 | return helper.visibleArguments(cmd).reduce((max, argument) => {
|
---|
162 | return Math.max(max, argument.term.length);
|
---|
163 | }, 0);
|
---|
164 | };
|
---|
165 |
|
---|
166 | /**
|
---|
167 | * Get the command usage to be displayed at the top of the built-in help.
|
---|
168 | *
|
---|
169 | * @param {Command} cmd
|
---|
170 | * @returns {string}
|
---|
171 | */
|
---|
172 |
|
---|
173 | commandUsage(cmd) {
|
---|
174 | // Usage
|
---|
175 | let cmdName = cmd._name;
|
---|
176 | if (cmd._aliases[0]) {
|
---|
177 | cmdName = cmdName + '|' + cmd._aliases[0];
|
---|
178 | }
|
---|
179 | let parentCmdNames = '';
|
---|
180 | for (let parentCmd = cmd.parent; parentCmd; parentCmd = parentCmd.parent) {
|
---|
181 | parentCmdNames = parentCmd.name() + ' ' + parentCmdNames;
|
---|
182 | }
|
---|
183 | return parentCmdNames + cmdName + ' ' + cmd.usage();
|
---|
184 | }
|
---|
185 |
|
---|
186 | /**
|
---|
187 | * Get the description for the command.
|
---|
188 | *
|
---|
189 | * @param {Command} cmd
|
---|
190 | * @returns {string}
|
---|
191 | */
|
---|
192 |
|
---|
193 | commandDescription(cmd) {
|
---|
194 | // @ts-ignore: overloaded return type
|
---|
195 | return cmd.description();
|
---|
196 | }
|
---|
197 |
|
---|
198 | /**
|
---|
199 | * Get the command description to show in the list of subcommands.
|
---|
200 | *
|
---|
201 | * @param {Command} cmd
|
---|
202 | * @returns {string}
|
---|
203 | */
|
---|
204 |
|
---|
205 | subcommandDescription(cmd) {
|
---|
206 | // @ts-ignore: overloaded return type
|
---|
207 | return cmd.description();
|
---|
208 | }
|
---|
209 |
|
---|
210 | /**
|
---|
211 | * Get the option description to show in the list of options.
|
---|
212 | *
|
---|
213 | * @param {Option} option
|
---|
214 | * @return {string}
|
---|
215 | */
|
---|
216 |
|
---|
217 | optionDescription(option) {
|
---|
218 | if (option.negate) {
|
---|
219 | return option.description;
|
---|
220 | }
|
---|
221 | const extraInfo = [];
|
---|
222 | if (option.argChoices) {
|
---|
223 | extraInfo.push(
|
---|
224 | // use stringify to match the display of the default value
|
---|
225 | `choices: ${option.argChoices.map((choice) => JSON.stringify(choice)).join(', ')}`);
|
---|
226 | }
|
---|
227 | if (option.defaultValue !== undefined) {
|
---|
228 | extraInfo.push(`default: ${option.defaultValueDescription || JSON.stringify(option.defaultValue)}`);
|
---|
229 | }
|
---|
230 | if (extraInfo.length > 0) {
|
---|
231 | return `${option.description} (${extraInfo.join(', ')})`;
|
---|
232 | }
|
---|
233 | return option.description;
|
---|
234 | };
|
---|
235 |
|
---|
236 | /**
|
---|
237 | * Generate the built-in help text.
|
---|
238 | *
|
---|
239 | * @param {Command} cmd
|
---|
240 | * @param {Help} helper
|
---|
241 | * @returns {string}
|
---|
242 | */
|
---|
243 |
|
---|
244 | formatHelp(cmd, helper) {
|
---|
245 | const termWidth = helper.padWidth(cmd, helper);
|
---|
246 | const helpWidth = helper.helpWidth || 80;
|
---|
247 | const itemIndentWidth = 2;
|
---|
248 | const itemSeparatorWidth = 2; // between term and description
|
---|
249 | function formatItem(term, description) {
|
---|
250 | if (description) {
|
---|
251 | const fullText = `${term.padEnd(termWidth + itemSeparatorWidth)}${description}`;
|
---|
252 | return helper.wrap(fullText, helpWidth - itemIndentWidth, termWidth + itemSeparatorWidth);
|
---|
253 | }
|
---|
254 | return term;
|
---|
255 | };
|
---|
256 | function formatList(textArray) {
|
---|
257 | return textArray.join('\n').replace(/^/gm, ' '.repeat(itemIndentWidth));
|
---|
258 | }
|
---|
259 |
|
---|
260 | // Usage
|
---|
261 | let output = [`Usage: ${helper.commandUsage(cmd)}`, ''];
|
---|
262 |
|
---|
263 | // Description
|
---|
264 | const commandDescription = helper.commandDescription(cmd);
|
---|
265 | if (commandDescription.length > 0) {
|
---|
266 | output = output.concat([commandDescription, '']);
|
---|
267 | }
|
---|
268 |
|
---|
269 | // Arguments
|
---|
270 | const argumentList = helper.visibleArguments(cmd).map((argument) => {
|
---|
271 | return formatItem(argument.term, argument.description);
|
---|
272 | });
|
---|
273 | if (argumentList.length > 0) {
|
---|
274 | output = output.concat(['Arguments:', formatList(argumentList), '']);
|
---|
275 | }
|
---|
276 |
|
---|
277 | // Options
|
---|
278 | const optionList = helper.visibleOptions(cmd).map((option) => {
|
---|
279 | return formatItem(helper.optionTerm(option), helper.optionDescription(option));
|
---|
280 | });
|
---|
281 | if (optionList.length > 0) {
|
---|
282 | output = output.concat(['Options:', formatList(optionList), '']);
|
---|
283 | }
|
---|
284 |
|
---|
285 | // Commands
|
---|
286 | const commandList = helper.visibleCommands(cmd).map((cmd) => {
|
---|
287 | return formatItem(helper.subcommandTerm(cmd), helper.subcommandDescription(cmd));
|
---|
288 | });
|
---|
289 | if (commandList.length > 0) {
|
---|
290 | output = output.concat(['Commands:', formatList(commandList), '']);
|
---|
291 | }
|
---|
292 |
|
---|
293 | return output.join('\n');
|
---|
294 | }
|
---|
295 |
|
---|
296 | /**
|
---|
297 | * Calculate the pad width from the maximum term length.
|
---|
298 | *
|
---|
299 | * @param {Command} cmd
|
---|
300 | * @param {Help} helper
|
---|
301 | * @returns {number}
|
---|
302 | */
|
---|
303 |
|
---|
304 | padWidth(cmd, helper) {
|
---|
305 | return Math.max(
|
---|
306 | helper.longestOptionTermLength(cmd, helper),
|
---|
307 | helper.longestSubcommandTermLength(cmd, helper),
|
---|
308 | helper.longestArgumentTermLength(cmd, helper)
|
---|
309 | );
|
---|
310 | };
|
---|
311 |
|
---|
312 | /**
|
---|
313 | * Wrap the given string to width characters per line, with lines after the first indented.
|
---|
314 | * Do not wrap if insufficient room for wrapping (minColumnWidth), or string is manually formatted.
|
---|
315 | *
|
---|
316 | * @param {string} str
|
---|
317 | * @param {number} width
|
---|
318 | * @param {number} indent
|
---|
319 | * @param {number} [minColumnWidth=40]
|
---|
320 | * @return {string}
|
---|
321 | *
|
---|
322 | */
|
---|
323 |
|
---|
324 | wrap(str, width, indent, minColumnWidth = 40) {
|
---|
325 | // Detect manually wrapped and indented strings by searching for line breaks
|
---|
326 | // followed by multiple spaces/tabs.
|
---|
327 | if (str.match(/[\n]\s+/)) return str;
|
---|
328 | // Do not wrap if not enough room for a wrapped column of text (as could end up with a word per line).
|
---|
329 | const columnWidth = width - indent;
|
---|
330 | if (columnWidth < minColumnWidth) return str;
|
---|
331 |
|
---|
332 | const leadingStr = str.substr(0, indent);
|
---|
333 | const columnText = str.substr(indent);
|
---|
334 |
|
---|
335 | const indentString = ' '.repeat(indent);
|
---|
336 | const regex = new RegExp('.{1,' + (columnWidth - 1) + '}([\\s\u200B]|$)|[^\\s\u200B]+?([\\s\u200B]|$)', 'g');
|
---|
337 | const lines = columnText.match(regex) || [];
|
---|
338 | return leadingStr + lines.map((line, i) => {
|
---|
339 | if (line.slice(-1) === '\n') {
|
---|
340 | line = line.slice(0, line.length - 1);
|
---|
341 | }
|
---|
342 | return ((i > 0) ? indentString : '') + line.trimRight();
|
---|
343 | }).join('\n');
|
---|
344 | }
|
---|
345 | }
|
---|
346 |
|
---|
347 | class Option {
|
---|
348 | /**
|
---|
349 | * Initialize a new `Option` with the given `flags` and `description`.
|
---|
350 | *
|
---|
351 | * @param {string} flags
|
---|
352 | * @param {string} [description]
|
---|
353 | */
|
---|
354 |
|
---|
355 | constructor(flags, description) {
|
---|
356 | this.flags = flags;
|
---|
357 | this.description = description || '';
|
---|
358 |
|
---|
359 | this.required = flags.includes('<'); // A value must be supplied when the option is specified.
|
---|
360 | this.optional = flags.includes('['); // A value is optional when the option is specified.
|
---|
361 | // variadic test ignores <value,...> et al which might be used to describe custom splitting of single argument
|
---|
362 | this.variadic = /\w\.\.\.[>\]]$/.test(flags); // The option can take multiple values.
|
---|
363 | this.mandatory = false; // The option must have a value after parsing, which usually means it must be specified on command line.
|
---|
364 | const optionFlags = _parseOptionFlags(flags);
|
---|
365 | this.short = optionFlags.shortFlag;
|
---|
366 | this.long = optionFlags.longFlag;
|
---|
367 | this.negate = false;
|
---|
368 | if (this.long) {
|
---|
369 | this.negate = this.long.startsWith('--no-');
|
---|
370 | }
|
---|
371 | this.defaultValue = undefined;
|
---|
372 | this.defaultValueDescription = undefined;
|
---|
373 | this.parseArg = undefined;
|
---|
374 | this.hidden = false;
|
---|
375 | this.argChoices = undefined;
|
---|
376 | }
|
---|
377 |
|
---|
378 | /**
|
---|
379 | * Set the default value, and optionally supply the description to be displayed in the help.
|
---|
380 | *
|
---|
381 | * @param {any} value
|
---|
382 | * @param {string} [description]
|
---|
383 | * @return {Option}
|
---|
384 | */
|
---|
385 |
|
---|
386 | default(value, description) {
|
---|
387 | this.defaultValue = value;
|
---|
388 | this.defaultValueDescription = description;
|
---|
389 | return this;
|
---|
390 | };
|
---|
391 |
|
---|
392 | /**
|
---|
393 | * Set the custom handler for processing CLI option arguments into option values.
|
---|
394 | *
|
---|
395 | * @param {Function} [fn]
|
---|
396 | * @return {Option}
|
---|
397 | */
|
---|
398 |
|
---|
399 | argParser(fn) {
|
---|
400 | this.parseArg = fn;
|
---|
401 | return this;
|
---|
402 | };
|
---|
403 |
|
---|
404 | /**
|
---|
405 | * Whether the option is mandatory and must have a value after parsing.
|
---|
406 | *
|
---|
407 | * @param {boolean} [mandatory=true]
|
---|
408 | * @return {Option}
|
---|
409 | */
|
---|
410 |
|
---|
411 | makeOptionMandatory(mandatory = true) {
|
---|
412 | this.mandatory = !!mandatory;
|
---|
413 | return this;
|
---|
414 | };
|
---|
415 |
|
---|
416 | /**
|
---|
417 | * Hide option in help.
|
---|
418 | *
|
---|
419 | * @param {boolean} [hide=true]
|
---|
420 | * @return {Option}
|
---|
421 | */
|
---|
422 |
|
---|
423 | hideHelp(hide = true) {
|
---|
424 | this.hidden = !!hide;
|
---|
425 | return this;
|
---|
426 | };
|
---|
427 |
|
---|
428 | /**
|
---|
429 | * @api private
|
---|
430 | */
|
---|
431 |
|
---|
432 | _concatValue(value, previous) {
|
---|
433 | if (previous === this.defaultValue || !Array.isArray(previous)) {
|
---|
434 | return [value];
|
---|
435 | }
|
---|
436 |
|
---|
437 | return previous.concat(value);
|
---|
438 | }
|
---|
439 |
|
---|
440 | /**
|
---|
441 | * Only allow option value to be one of choices.
|
---|
442 | *
|
---|
443 | * @param {string[]} values
|
---|
444 | * @return {Option}
|
---|
445 | */
|
---|
446 |
|
---|
447 | choices(values) {
|
---|
448 | this.argChoices = values;
|
---|
449 | this.parseArg = (arg, previous) => {
|
---|
450 | if (!values.includes(arg)) {
|
---|
451 | throw new InvalidOptionArgumentError(`Allowed choices are ${values.join(', ')}.`);
|
---|
452 | }
|
---|
453 | if (this.variadic) {
|
---|
454 | return this._concatValue(arg, previous);
|
---|
455 | }
|
---|
456 | return arg;
|
---|
457 | };
|
---|
458 | return this;
|
---|
459 | };
|
---|
460 |
|
---|
461 | /**
|
---|
462 | * Return option name.
|
---|
463 | *
|
---|
464 | * @return {string}
|
---|
465 | */
|
---|
466 |
|
---|
467 | name() {
|
---|
468 | if (this.long) {
|
---|
469 | return this.long.replace(/^--/, '');
|
---|
470 | }
|
---|
471 | return this.short.replace(/^-/, '');
|
---|
472 | };
|
---|
473 |
|
---|
474 | /**
|
---|
475 | * Return option name, in a camelcase format that can be used
|
---|
476 | * as a object attribute key.
|
---|
477 | *
|
---|
478 | * @return {string}
|
---|
479 | * @api private
|
---|
480 | */
|
---|
481 |
|
---|
482 | attributeName() {
|
---|
483 | return camelcase(this.name().replace(/^no-/, ''));
|
---|
484 | };
|
---|
485 |
|
---|
486 | /**
|
---|
487 | * Check if `arg` matches the short or long flag.
|
---|
488 | *
|
---|
489 | * @param {string} arg
|
---|
490 | * @return {boolean}
|
---|
491 | * @api private
|
---|
492 | */
|
---|
493 |
|
---|
494 | is(arg) {
|
---|
495 | return this.short === arg || this.long === arg;
|
---|
496 | };
|
---|
497 | }
|
---|
498 |
|
---|
499 | /**
|
---|
500 | * CommanderError class
|
---|
501 | * @class
|
---|
502 | */
|
---|
503 | class CommanderError extends Error {
|
---|
504 | /**
|
---|
505 | * Constructs the CommanderError class
|
---|
506 | * @param {number} exitCode suggested exit code which could be used with process.exit
|
---|
507 | * @param {string} code an id string representing the error
|
---|
508 | * @param {string} message human-readable description of the error
|
---|
509 | * @constructor
|
---|
510 | */
|
---|
511 | constructor(exitCode, code, message) {
|
---|
512 | super(message);
|
---|
513 | // properly capture stack trace in Node.js
|
---|
514 | Error.captureStackTrace(this, this.constructor);
|
---|
515 | this.name = this.constructor.name;
|
---|
516 | this.code = code;
|
---|
517 | this.exitCode = exitCode;
|
---|
518 | this.nestedError = undefined;
|
---|
519 | }
|
---|
520 | }
|
---|
521 |
|
---|
522 | /**
|
---|
523 | * InvalidOptionArgumentError class
|
---|
524 | * @class
|
---|
525 | */
|
---|
526 | class InvalidOptionArgumentError extends CommanderError {
|
---|
527 | /**
|
---|
528 | * Constructs the InvalidOptionArgumentError class
|
---|
529 | * @param {string} [message] explanation of why argument is invalid
|
---|
530 | * @constructor
|
---|
531 | */
|
---|
532 | constructor(message) {
|
---|
533 | super(1, 'commander.invalidOptionArgument', message);
|
---|
534 | // properly capture stack trace in Node.js
|
---|
535 | Error.captureStackTrace(this, this.constructor);
|
---|
536 | this.name = this.constructor.name;
|
---|
537 | }
|
---|
538 | }
|
---|
539 |
|
---|
540 | class Command extends EventEmitter {
|
---|
541 | /**
|
---|
542 | * Initialize a new `Command`.
|
---|
543 | *
|
---|
544 | * @param {string} [name]
|
---|
545 | */
|
---|
546 |
|
---|
547 | constructor(name) {
|
---|
548 | super();
|
---|
549 | this.commands = [];
|
---|
550 | this.options = [];
|
---|
551 | this.parent = null;
|
---|
552 | this._allowUnknownOption = false;
|
---|
553 | this._allowExcessArguments = true;
|
---|
554 | this._args = [];
|
---|
555 | this.rawArgs = null;
|
---|
556 | this._scriptPath = null;
|
---|
557 | this._name = name || '';
|
---|
558 | this._optionValues = {};
|
---|
559 | this._storeOptionsAsProperties = false;
|
---|
560 | this._actionResults = [];
|
---|
561 | this._actionHandler = null;
|
---|
562 | this._executableHandler = false;
|
---|
563 | this._executableFile = null; // custom name for executable
|
---|
564 | this._defaultCommandName = null;
|
---|
565 | this._exitCallback = null;
|
---|
566 | this._aliases = [];
|
---|
567 | this._combineFlagAndOptionalValue = true;
|
---|
568 | this._description = '';
|
---|
569 | this._argsDescription = undefined;
|
---|
570 | this._enablePositionalOptions = false;
|
---|
571 | this._passThroughOptions = false;
|
---|
572 |
|
---|
573 | // see .configureOutput() for docs
|
---|
574 | this._outputConfiguration = {
|
---|
575 | writeOut: (str) => process.stdout.write(str),
|
---|
576 | writeErr: (str) => process.stderr.write(str),
|
---|
577 | getOutHelpWidth: () => process.stdout.isTTY ? process.stdout.columns : undefined,
|
---|
578 | getErrHelpWidth: () => process.stderr.isTTY ? process.stderr.columns : undefined,
|
---|
579 | outputError: (str, write) => write(str)
|
---|
580 | };
|
---|
581 |
|
---|
582 | this._hidden = false;
|
---|
583 | this._hasHelpOption = true;
|
---|
584 | this._helpFlags = '-h, --help';
|
---|
585 | this._helpDescription = 'display help for command';
|
---|
586 | this._helpShortFlag = '-h';
|
---|
587 | this._helpLongFlag = '--help';
|
---|
588 | this._addImplicitHelpCommand = undefined; // Deliberately undefined, not decided whether true or false
|
---|
589 | this._helpCommandName = 'help';
|
---|
590 | this._helpCommandnameAndArgs = 'help [command]';
|
---|
591 | this._helpCommandDescription = 'display help for command';
|
---|
592 | this._helpConfiguration = {};
|
---|
593 | }
|
---|
594 |
|
---|
595 | /**
|
---|
596 | * Define a command.
|
---|
597 | *
|
---|
598 | * There are two styles of command: pay attention to where to put the description.
|
---|
599 | *
|
---|
600 | * Examples:
|
---|
601 | *
|
---|
602 | * // Command implemented using action handler (description is supplied separately to `.command`)
|
---|
603 | * program
|
---|
604 | * .command('clone <source> [destination]')
|
---|
605 | * .description('clone a repository into a newly created directory')
|
---|
606 | * .action((source, destination) => {
|
---|
607 | * console.log('clone command called');
|
---|
608 | * });
|
---|
609 | *
|
---|
610 | * // Command implemented using separate executable file (description is second parameter to `.command`)
|
---|
611 | * program
|
---|
612 | * .command('start <service>', 'start named service')
|
---|
613 | * .command('stop [service]', 'stop named service, or all if no name supplied');
|
---|
614 | *
|
---|
615 | * @param {string} nameAndArgs - command name and arguments, args are `<required>` or `[optional]` and last may also be `variadic...`
|
---|
616 | * @param {Object|string} [actionOptsOrExecDesc] - configuration options (for action), or description (for executable)
|
---|
617 | * @param {Object} [execOpts] - configuration options (for executable)
|
---|
618 | * @return {Command} returns new command for action handler, or `this` for executable command
|
---|
619 | */
|
---|
620 |
|
---|
621 | command(nameAndArgs, actionOptsOrExecDesc, execOpts) {
|
---|
622 | let desc = actionOptsOrExecDesc;
|
---|
623 | let opts = execOpts;
|
---|
624 | if (typeof desc === 'object' && desc !== null) {
|
---|
625 | opts = desc;
|
---|
626 | desc = null;
|
---|
627 | }
|
---|
628 | opts = opts || {};
|
---|
629 | const args = nameAndArgs.split(/ +/);
|
---|
630 | const cmd = this.createCommand(args.shift());
|
---|
631 |
|
---|
632 | if (desc) {
|
---|
633 | cmd.description(desc);
|
---|
634 | cmd._executableHandler = true;
|
---|
635 | }
|
---|
636 | if (opts.isDefault) this._defaultCommandName = cmd._name;
|
---|
637 |
|
---|
638 | cmd._outputConfiguration = this._outputConfiguration;
|
---|
639 |
|
---|
640 | cmd._hidden = !!(opts.noHelp || opts.hidden); // noHelp is deprecated old name for hidden
|
---|
641 | cmd._hasHelpOption = this._hasHelpOption;
|
---|
642 | cmd._helpFlags = this._helpFlags;
|
---|
643 | cmd._helpDescription = this._helpDescription;
|
---|
644 | cmd._helpShortFlag = this._helpShortFlag;
|
---|
645 | cmd._helpLongFlag = this._helpLongFlag;
|
---|
646 | cmd._helpCommandName = this._helpCommandName;
|
---|
647 | cmd._helpCommandnameAndArgs = this._helpCommandnameAndArgs;
|
---|
648 | cmd._helpCommandDescription = this._helpCommandDescription;
|
---|
649 | cmd._helpConfiguration = this._helpConfiguration;
|
---|
650 | cmd._exitCallback = this._exitCallback;
|
---|
651 | cmd._storeOptionsAsProperties = this._storeOptionsAsProperties;
|
---|
652 | cmd._combineFlagAndOptionalValue = this._combineFlagAndOptionalValue;
|
---|
653 | cmd._allowExcessArguments = this._allowExcessArguments;
|
---|
654 | cmd._enablePositionalOptions = this._enablePositionalOptions;
|
---|
655 |
|
---|
656 | cmd._executableFile = opts.executableFile || null; // Custom name for executable file, set missing to null to match constructor
|
---|
657 | this.commands.push(cmd);
|
---|
658 | cmd._parseExpectedArgs(args);
|
---|
659 | cmd.parent = this;
|
---|
660 |
|
---|
661 | if (desc) return this;
|
---|
662 | return cmd;
|
---|
663 | };
|
---|
664 |
|
---|
665 | /**
|
---|
666 | * Factory routine to create a new unattached command.
|
---|
667 | *
|
---|
668 | * See .command() for creating an attached subcommand, which uses this routine to
|
---|
669 | * create the command. You can override createCommand to customise subcommands.
|
---|
670 | *
|
---|
671 | * @param {string} [name]
|
---|
672 | * @return {Command} new command
|
---|
673 | */
|
---|
674 |
|
---|
675 | createCommand(name) {
|
---|
676 | return new Command(name);
|
---|
677 | };
|
---|
678 |
|
---|
679 | /**
|
---|
680 | * You can customise the help with a subclass of Help by overriding createHelp,
|
---|
681 | * or by overriding Help properties using configureHelp().
|
---|
682 | *
|
---|
683 | * @return {Help}
|
---|
684 | */
|
---|
685 |
|
---|
686 | createHelp() {
|
---|
687 | return Object.assign(new Help(), this.configureHelp());
|
---|
688 | };
|
---|
689 |
|
---|
690 | /**
|
---|
691 | * You can customise the help by overriding Help properties using configureHelp(),
|
---|
692 | * or with a subclass of Help by overriding createHelp().
|
---|
693 | *
|
---|
694 | * @param {Object} [configuration] - configuration options
|
---|
695 | * @return {Command|Object} `this` command for chaining, or stored configuration
|
---|
696 | */
|
---|
697 |
|
---|
698 | configureHelp(configuration) {
|
---|
699 | if (configuration === undefined) return this._helpConfiguration;
|
---|
700 |
|
---|
701 | this._helpConfiguration = configuration;
|
---|
702 | return this;
|
---|
703 | }
|
---|
704 |
|
---|
705 | /**
|
---|
706 | * The default output goes to stdout and stderr. You can customise this for special
|
---|
707 | * applications. You can also customise the display of errors by overriding outputError.
|
---|
708 | *
|
---|
709 | * The configuration properties are all functions:
|
---|
710 | *
|
---|
711 | * // functions to change where being written, stdout and stderr
|
---|
712 | * writeOut(str)
|
---|
713 | * writeErr(str)
|
---|
714 | * // matching functions to specify width for wrapping help
|
---|
715 | * getOutHelpWidth()
|
---|
716 | * getErrHelpWidth()
|
---|
717 | * // functions based on what is being written out
|
---|
718 | * outputError(str, write) // used for displaying errors, and not used for displaying help
|
---|
719 | *
|
---|
720 | * @param {Object} [configuration] - configuration options
|
---|
721 | * @return {Command|Object} `this` command for chaining, or stored configuration
|
---|
722 | */
|
---|
723 |
|
---|
724 | configureOutput(configuration) {
|
---|
725 | if (configuration === undefined) return this._outputConfiguration;
|
---|
726 |
|
---|
727 | Object.assign(this._outputConfiguration, configuration);
|
---|
728 | return this;
|
---|
729 | }
|
---|
730 |
|
---|
731 | /**
|
---|
732 | * Add a prepared subcommand.
|
---|
733 | *
|
---|
734 | * See .command() for creating an attached subcommand which inherits settings from its parent.
|
---|
735 | *
|
---|
736 | * @param {Command} cmd - new subcommand
|
---|
737 | * @param {Object} [opts] - configuration options
|
---|
738 | * @return {Command} `this` command for chaining
|
---|
739 | */
|
---|
740 |
|
---|
741 | addCommand(cmd, opts) {
|
---|
742 | if (!cmd._name) throw new Error('Command passed to .addCommand() must have a name');
|
---|
743 |
|
---|
744 | // To keep things simple, block automatic name generation for deeply nested executables.
|
---|
745 | // Fail fast and detect when adding rather than later when parsing.
|
---|
746 | function checkExplicitNames(commandArray) {
|
---|
747 | commandArray.forEach((cmd) => {
|
---|
748 | if (cmd._executableHandler && !cmd._executableFile) {
|
---|
749 | throw new Error(`Must specify executableFile for deeply nested executable: ${cmd.name()}`);
|
---|
750 | }
|
---|
751 | checkExplicitNames(cmd.commands);
|
---|
752 | });
|
---|
753 | }
|
---|
754 | checkExplicitNames(cmd.commands);
|
---|
755 |
|
---|
756 | opts = opts || {};
|
---|
757 | if (opts.isDefault) this._defaultCommandName = cmd._name;
|
---|
758 | if (opts.noHelp || opts.hidden) cmd._hidden = true; // modifying passed command due to existing implementation
|
---|
759 |
|
---|
760 | this.commands.push(cmd);
|
---|
761 | cmd.parent = this;
|
---|
762 | return this;
|
---|
763 | };
|
---|
764 |
|
---|
765 | /**
|
---|
766 | * Define argument syntax for the command.
|
---|
767 | */
|
---|
768 |
|
---|
769 | arguments(desc) {
|
---|
770 | return this._parseExpectedArgs(desc.split(/ +/));
|
---|
771 | };
|
---|
772 |
|
---|
773 | /**
|
---|
774 | * Override default decision whether to add implicit help command.
|
---|
775 | *
|
---|
776 | * addHelpCommand() // force on
|
---|
777 | * addHelpCommand(false); // force off
|
---|
778 | * addHelpCommand('help [cmd]', 'display help for [cmd]'); // force on with custom details
|
---|
779 | *
|
---|
780 | * @return {Command} `this` command for chaining
|
---|
781 | */
|
---|
782 |
|
---|
783 | addHelpCommand(enableOrNameAndArgs, description) {
|
---|
784 | if (enableOrNameAndArgs === false) {
|
---|
785 | this._addImplicitHelpCommand = false;
|
---|
786 | } else {
|
---|
787 | this._addImplicitHelpCommand = true;
|
---|
788 | if (typeof enableOrNameAndArgs === 'string') {
|
---|
789 | this._helpCommandName = enableOrNameAndArgs.split(' ')[0];
|
---|
790 | this._helpCommandnameAndArgs = enableOrNameAndArgs;
|
---|
791 | }
|
---|
792 | this._helpCommandDescription = description || this._helpCommandDescription;
|
---|
793 | }
|
---|
794 | return this;
|
---|
795 | };
|
---|
796 |
|
---|
797 | /**
|
---|
798 | * @return {boolean}
|
---|
799 | * @api private
|
---|
800 | */
|
---|
801 |
|
---|
802 | _hasImplicitHelpCommand() {
|
---|
803 | if (this._addImplicitHelpCommand === undefined) {
|
---|
804 | return this.commands.length && !this._actionHandler && !this._findCommand('help');
|
---|
805 | }
|
---|
806 | return this._addImplicitHelpCommand;
|
---|
807 | };
|
---|
808 |
|
---|
809 | /**
|
---|
810 | * Parse expected `args`.
|
---|
811 | *
|
---|
812 | * For example `["[type]"]` becomes `[{ required: false, name: 'type' }]`.
|
---|
813 | *
|
---|
814 | * @param {Array} args
|
---|
815 | * @return {Command} `this` command for chaining
|
---|
816 | * @api private
|
---|
817 | */
|
---|
818 |
|
---|
819 | _parseExpectedArgs(args) {
|
---|
820 | if (!args.length) return;
|
---|
821 | args.forEach((arg) => {
|
---|
822 | const argDetails = {
|
---|
823 | required: false,
|
---|
824 | name: '',
|
---|
825 | variadic: false
|
---|
826 | };
|
---|
827 |
|
---|
828 | switch (arg[0]) {
|
---|
829 | case '<':
|
---|
830 | argDetails.required = true;
|
---|
831 | argDetails.name = arg.slice(1, -1);
|
---|
832 | break;
|
---|
833 | case '[':
|
---|
834 | argDetails.name = arg.slice(1, -1);
|
---|
835 | break;
|
---|
836 | }
|
---|
837 |
|
---|
838 | if (argDetails.name.length > 3 && argDetails.name.slice(-3) === '...') {
|
---|
839 | argDetails.variadic = true;
|
---|
840 | argDetails.name = argDetails.name.slice(0, -3);
|
---|
841 | }
|
---|
842 | if (argDetails.name) {
|
---|
843 | this._args.push(argDetails);
|
---|
844 | }
|
---|
845 | });
|
---|
846 | this._args.forEach((arg, i) => {
|
---|
847 | if (arg.variadic && i < this._args.length - 1) {
|
---|
848 | throw new Error(`only the last argument can be variadic '${arg.name}'`);
|
---|
849 | }
|
---|
850 | });
|
---|
851 | return this;
|
---|
852 | };
|
---|
853 |
|
---|
854 | /**
|
---|
855 | * Register callback to use as replacement for calling process.exit.
|
---|
856 | *
|
---|
857 | * @param {Function} [fn] optional callback which will be passed a CommanderError, defaults to throwing
|
---|
858 | * @return {Command} `this` command for chaining
|
---|
859 | */
|
---|
860 |
|
---|
861 | exitOverride(fn) {
|
---|
862 | if (fn) {
|
---|
863 | this._exitCallback = fn;
|
---|
864 | } else {
|
---|
865 | this._exitCallback = (err) => {
|
---|
866 | if (err.code !== 'commander.executeSubCommandAsync') {
|
---|
867 | throw err;
|
---|
868 | } else {
|
---|
869 | // Async callback from spawn events, not useful to throw.
|
---|
870 | }
|
---|
871 | };
|
---|
872 | }
|
---|
873 | return this;
|
---|
874 | };
|
---|
875 |
|
---|
876 | /**
|
---|
877 | * Call process.exit, and _exitCallback if defined.
|
---|
878 | *
|
---|
879 | * @param {number} exitCode exit code for using with process.exit
|
---|
880 | * @param {string} code an id string representing the error
|
---|
881 | * @param {string} message human-readable description of the error
|
---|
882 | * @return never
|
---|
883 | * @api private
|
---|
884 | */
|
---|
885 |
|
---|
886 | _exit(exitCode, code, message) {
|
---|
887 | if (this._exitCallback) {
|
---|
888 | this._exitCallback(new CommanderError(exitCode, code, message));
|
---|
889 | // Expecting this line is not reached.
|
---|
890 | }
|
---|
891 | process.exit(exitCode);
|
---|
892 | };
|
---|
893 |
|
---|
894 | /**
|
---|
895 | * Register callback `fn` for the command.
|
---|
896 | *
|
---|
897 | * Examples:
|
---|
898 | *
|
---|
899 | * program
|
---|
900 | * .command('help')
|
---|
901 | * .description('display verbose help')
|
---|
902 | * .action(function() {
|
---|
903 | * // output help here
|
---|
904 | * });
|
---|
905 | *
|
---|
906 | * @param {Function} fn
|
---|
907 | * @return {Command} `this` command for chaining
|
---|
908 | */
|
---|
909 |
|
---|
910 | action(fn) {
|
---|
911 | const listener = (args) => {
|
---|
912 | // The .action callback takes an extra parameter which is the command or options.
|
---|
913 | const expectedArgsCount = this._args.length;
|
---|
914 | const actionArgs = args.slice(0, expectedArgsCount);
|
---|
915 | if (this._storeOptionsAsProperties) {
|
---|
916 | actionArgs[expectedArgsCount] = this; // backwards compatible "options"
|
---|
917 | } else {
|
---|
918 | actionArgs[expectedArgsCount] = this.opts();
|
---|
919 | }
|
---|
920 | actionArgs.push(this);
|
---|
921 |
|
---|
922 | const actionResult = fn.apply(this, actionArgs);
|
---|
923 | // Remember result in case it is async. Assume parseAsync getting called on root.
|
---|
924 | let rootCommand = this;
|
---|
925 | while (rootCommand.parent) {
|
---|
926 | rootCommand = rootCommand.parent;
|
---|
927 | }
|
---|
928 | rootCommand._actionResults.push(actionResult);
|
---|
929 | };
|
---|
930 | this._actionHandler = listener;
|
---|
931 | return this;
|
---|
932 | };
|
---|
933 |
|
---|
934 | /**
|
---|
935 | * Factory routine to create a new unattached option.
|
---|
936 | *
|
---|
937 | * See .option() for creating an attached option, which uses this routine to
|
---|
938 | * create the option. You can override createOption to return a custom option.
|
---|
939 | *
|
---|
940 | * @param {string} flags
|
---|
941 | * @param {string} [description]
|
---|
942 | * @return {Option} new option
|
---|
943 | */
|
---|
944 |
|
---|
945 | createOption(flags, description) {
|
---|
946 | return new Option(flags, description);
|
---|
947 | };
|
---|
948 |
|
---|
949 | /**
|
---|
950 | * Add an option.
|
---|
951 | *
|
---|
952 | * @param {Option} option
|
---|
953 | * @return {Command} `this` command for chaining
|
---|
954 | */
|
---|
955 | addOption(option) {
|
---|
956 | const oname = option.name();
|
---|
957 | const name = option.attributeName();
|
---|
958 |
|
---|
959 | let defaultValue = option.defaultValue;
|
---|
960 |
|
---|
961 | // preassign default value for --no-*, [optional], <required>, or plain flag if boolean value
|
---|
962 | if (option.negate || option.optional || option.required || typeof defaultValue === 'boolean') {
|
---|
963 | // when --no-foo we make sure default is true, unless a --foo option is already defined
|
---|
964 | if (option.negate) {
|
---|
965 | const positiveLongFlag = option.long.replace(/^--no-/, '--');
|
---|
966 | defaultValue = this._findOption(positiveLongFlag) ? this._getOptionValue(name) : true;
|
---|
967 | }
|
---|
968 | // preassign only if we have a default
|
---|
969 | if (defaultValue !== undefined) {
|
---|
970 | this._setOptionValue(name, defaultValue);
|
---|
971 | }
|
---|
972 | }
|
---|
973 |
|
---|
974 | // register the option
|
---|
975 | this.options.push(option);
|
---|
976 |
|
---|
977 | // when it's passed assign the value
|
---|
978 | // and conditionally invoke the callback
|
---|
979 | this.on('option:' + oname, (val) => {
|
---|
980 | const oldValue = this._getOptionValue(name);
|
---|
981 |
|
---|
982 | // custom processing
|
---|
983 | if (val !== null && option.parseArg) {
|
---|
984 | try {
|
---|
985 | val = option.parseArg(val, oldValue === undefined ? defaultValue : oldValue);
|
---|
986 | } catch (err) {
|
---|
987 | if (err.code === 'commander.invalidOptionArgument') {
|
---|
988 | const message = `error: option '${option.flags}' argument '${val}' is invalid. ${err.message}`;
|
---|
989 | this._displayError(err.exitCode, err.code, message);
|
---|
990 | }
|
---|
991 | throw err;
|
---|
992 | }
|
---|
993 | } else if (val !== null && option.variadic) {
|
---|
994 | val = option._concatValue(val, oldValue);
|
---|
995 | }
|
---|
996 |
|
---|
997 | // unassigned or boolean value
|
---|
998 | if (typeof oldValue === 'boolean' || typeof oldValue === 'undefined') {
|
---|
999 | // if no value, negate false, and we have a default, then use it!
|
---|
1000 | if (val == null) {
|
---|
1001 | this._setOptionValue(name, option.negate
|
---|
1002 | ? false
|
---|
1003 | : defaultValue || true);
|
---|
1004 | } else {
|
---|
1005 | this._setOptionValue(name, val);
|
---|
1006 | }
|
---|
1007 | } else if (val !== null) {
|
---|
1008 | // reassign
|
---|
1009 | this._setOptionValue(name, option.negate ? false : val);
|
---|
1010 | }
|
---|
1011 | });
|
---|
1012 |
|
---|
1013 | return this;
|
---|
1014 | }
|
---|
1015 |
|
---|
1016 | /**
|
---|
1017 | * Internal implementation shared by .option() and .requiredOption()
|
---|
1018 | *
|
---|
1019 | * @api private
|
---|
1020 | */
|
---|
1021 | _optionEx(config, flags, description, fn, defaultValue) {
|
---|
1022 | const option = this.createOption(flags, description);
|
---|
1023 | option.makeOptionMandatory(!!config.mandatory);
|
---|
1024 | if (typeof fn === 'function') {
|
---|
1025 | option.default(defaultValue).argParser(fn);
|
---|
1026 | } else if (fn instanceof RegExp) {
|
---|
1027 | // deprecated
|
---|
1028 | const regex = fn;
|
---|
1029 | fn = (val, def) => {
|
---|
1030 | const m = regex.exec(val);
|
---|
1031 | return m ? m[0] : def;
|
---|
1032 | };
|
---|
1033 | option.default(defaultValue).argParser(fn);
|
---|
1034 | } else {
|
---|
1035 | option.default(fn);
|
---|
1036 | }
|
---|
1037 |
|
---|
1038 | return this.addOption(option);
|
---|
1039 | }
|
---|
1040 |
|
---|
1041 | /**
|
---|
1042 | * Define option with `flags`, `description` and optional
|
---|
1043 | * coercion `fn`.
|
---|
1044 | *
|
---|
1045 | * The `flags` string contains the short and/or long flags,
|
---|
1046 | * separated by comma, a pipe or space. The following are all valid
|
---|
1047 | * all will output this way when `--help` is used.
|
---|
1048 | *
|
---|
1049 | * "-p, --pepper"
|
---|
1050 | * "-p|--pepper"
|
---|
1051 | * "-p --pepper"
|
---|
1052 | *
|
---|
1053 | * Examples:
|
---|
1054 | *
|
---|
1055 | * // simple boolean defaulting to undefined
|
---|
1056 | * program.option('-p, --pepper', 'add pepper');
|
---|
1057 | *
|
---|
1058 | * program.pepper
|
---|
1059 | * // => undefined
|
---|
1060 | *
|
---|
1061 | * --pepper
|
---|
1062 | * program.pepper
|
---|
1063 | * // => true
|
---|
1064 | *
|
---|
1065 | * // simple boolean defaulting to true (unless non-negated option is also defined)
|
---|
1066 | * program.option('-C, --no-cheese', 'remove cheese');
|
---|
1067 | *
|
---|
1068 | * program.cheese
|
---|
1069 | * // => true
|
---|
1070 | *
|
---|
1071 | * --no-cheese
|
---|
1072 | * program.cheese
|
---|
1073 | * // => false
|
---|
1074 | *
|
---|
1075 | * // required argument
|
---|
1076 | * program.option('-C, --chdir <path>', 'change the working directory');
|
---|
1077 | *
|
---|
1078 | * --chdir /tmp
|
---|
1079 | * program.chdir
|
---|
1080 | * // => "/tmp"
|
---|
1081 | *
|
---|
1082 | * // optional argument
|
---|
1083 | * program.option('-c, --cheese [type]', 'add cheese [marble]');
|
---|
1084 | *
|
---|
1085 | * @param {string} flags
|
---|
1086 | * @param {string} [description]
|
---|
1087 | * @param {Function|*} [fn] - custom option processing function or default value
|
---|
1088 | * @param {*} [defaultValue]
|
---|
1089 | * @return {Command} `this` command for chaining
|
---|
1090 | */
|
---|
1091 |
|
---|
1092 | option(flags, description, fn, defaultValue) {
|
---|
1093 | return this._optionEx({}, flags, description, fn, defaultValue);
|
---|
1094 | };
|
---|
1095 |
|
---|
1096 | /**
|
---|
1097 | * Add a required option which must have a value after parsing. This usually means
|
---|
1098 | * the option must be specified on the command line. (Otherwise the same as .option().)
|
---|
1099 | *
|
---|
1100 | * The `flags` string contains the short and/or long flags, separated by comma, a pipe or space.
|
---|
1101 | *
|
---|
1102 | * @param {string} flags
|
---|
1103 | * @param {string} [description]
|
---|
1104 | * @param {Function|*} [fn] - custom option processing function or default value
|
---|
1105 | * @param {*} [defaultValue]
|
---|
1106 | * @return {Command} `this` command for chaining
|
---|
1107 | */
|
---|
1108 |
|
---|
1109 | requiredOption(flags, description, fn, defaultValue) {
|
---|
1110 | return this._optionEx({ mandatory: true }, flags, description, fn, defaultValue);
|
---|
1111 | };
|
---|
1112 |
|
---|
1113 | /**
|
---|
1114 | * Alter parsing of short flags with optional values.
|
---|
1115 | *
|
---|
1116 | * Examples:
|
---|
1117 | *
|
---|
1118 | * // for `.option('-f,--flag [value]'):
|
---|
1119 | * .combineFlagAndOptionalValue(true) // `-f80` is treated like `--flag=80`, this is the default behaviour
|
---|
1120 | * .combineFlagAndOptionalValue(false) // `-fb` is treated like `-f -b`
|
---|
1121 | *
|
---|
1122 | * @param {Boolean} [combine=true] - if `true` or omitted, an optional value can be specified directly after the flag.
|
---|
1123 | */
|
---|
1124 | combineFlagAndOptionalValue(combine = true) {
|
---|
1125 | this._combineFlagAndOptionalValue = !!combine;
|
---|
1126 | return this;
|
---|
1127 | };
|
---|
1128 |
|
---|
1129 | /**
|
---|
1130 | * Allow unknown options on the command line.
|
---|
1131 | *
|
---|
1132 | * @param {Boolean} [allowUnknown=true] - if `true` or omitted, no error will be thrown
|
---|
1133 | * for unknown options.
|
---|
1134 | */
|
---|
1135 | allowUnknownOption(allowUnknown = true) {
|
---|
1136 | this._allowUnknownOption = !!allowUnknown;
|
---|
1137 | return this;
|
---|
1138 | };
|
---|
1139 |
|
---|
1140 | /**
|
---|
1141 | * Allow excess command-arguments on the command line. Pass false to make excess arguments an error.
|
---|
1142 | *
|
---|
1143 | * @param {Boolean} [allowExcess=true] - if `true` or omitted, no error will be thrown
|
---|
1144 | * for excess arguments.
|
---|
1145 | */
|
---|
1146 | allowExcessArguments(allowExcess = true) {
|
---|
1147 | this._allowExcessArguments = !!allowExcess;
|
---|
1148 | return this;
|
---|
1149 | };
|
---|
1150 |
|
---|
1151 | /**
|
---|
1152 | * Enable positional options. Positional means global options are specified before subcommands which lets
|
---|
1153 | * subcommands reuse the same option names, and also enables subcommands to turn on passThroughOptions.
|
---|
1154 | * The default behaviour is non-positional and global options may appear anywhere on the command line.
|
---|
1155 | *
|
---|
1156 | * @param {Boolean} [positional=true]
|
---|
1157 | */
|
---|
1158 | enablePositionalOptions(positional = true) {
|
---|
1159 | this._enablePositionalOptions = !!positional;
|
---|
1160 | return this;
|
---|
1161 | };
|
---|
1162 |
|
---|
1163 | /**
|
---|
1164 | * Pass through options that come after command-arguments rather than treat them as command-options,
|
---|
1165 | * so actual command-options come before command-arguments. Turning this on for a subcommand requires
|
---|
1166 | * positional options to have been enabled on the program (parent commands).
|
---|
1167 | * The default behaviour is non-positional and options may appear before or after command-arguments.
|
---|
1168 | *
|
---|
1169 | * @param {Boolean} [passThrough=true]
|
---|
1170 | * for unknown options.
|
---|
1171 | */
|
---|
1172 | passThroughOptions(passThrough = true) {
|
---|
1173 | this._passThroughOptions = !!passThrough;
|
---|
1174 | if (!!this.parent && passThrough && !this.parent._enablePositionalOptions) {
|
---|
1175 | throw new Error('passThroughOptions can not be used without turning on enablePositionalOptions for parent command(s)');
|
---|
1176 | }
|
---|
1177 | return this;
|
---|
1178 | };
|
---|
1179 |
|
---|
1180 | /**
|
---|
1181 | * Whether to store option values as properties on command object,
|
---|
1182 | * or store separately (specify false). In both cases the option values can be accessed using .opts().
|
---|
1183 | *
|
---|
1184 | * @param {boolean} [storeAsProperties=true]
|
---|
1185 | * @return {Command} `this` command for chaining
|
---|
1186 | */
|
---|
1187 |
|
---|
1188 | storeOptionsAsProperties(storeAsProperties = true) {
|
---|
1189 | this._storeOptionsAsProperties = !!storeAsProperties;
|
---|
1190 | if (this.options.length) {
|
---|
1191 | throw new Error('call .storeOptionsAsProperties() before adding options');
|
---|
1192 | }
|
---|
1193 | return this;
|
---|
1194 | };
|
---|
1195 |
|
---|
1196 | /**
|
---|
1197 | * Store option value
|
---|
1198 | *
|
---|
1199 | * @param {string} key
|
---|
1200 | * @param {Object} value
|
---|
1201 | * @api private
|
---|
1202 | */
|
---|
1203 |
|
---|
1204 | _setOptionValue(key, value) {
|
---|
1205 | if (this._storeOptionsAsProperties) {
|
---|
1206 | this[key] = value;
|
---|
1207 | } else {
|
---|
1208 | this._optionValues[key] = value;
|
---|
1209 | }
|
---|
1210 | };
|
---|
1211 |
|
---|
1212 | /**
|
---|
1213 | * Retrieve option value
|
---|
1214 | *
|
---|
1215 | * @param {string} key
|
---|
1216 | * @return {Object} value
|
---|
1217 | * @api private
|
---|
1218 | */
|
---|
1219 |
|
---|
1220 | _getOptionValue(key) {
|
---|
1221 | if (this._storeOptionsAsProperties) {
|
---|
1222 | return this[key];
|
---|
1223 | }
|
---|
1224 | return this._optionValues[key];
|
---|
1225 | };
|
---|
1226 |
|
---|
1227 | /**
|
---|
1228 | * Parse `argv`, setting options and invoking commands when defined.
|
---|
1229 | *
|
---|
1230 | * The default expectation is that the arguments are from node and have the application as argv[0]
|
---|
1231 | * and the script being run in argv[1], with user parameters after that.
|
---|
1232 | *
|
---|
1233 | * Examples:
|
---|
1234 | *
|
---|
1235 | * program.parse(process.argv);
|
---|
1236 | * program.parse(); // implicitly use process.argv and auto-detect node vs electron conventions
|
---|
1237 | * program.parse(my-args, { from: 'user' }); // just user supplied arguments, nothing special about argv[0]
|
---|
1238 | *
|
---|
1239 | * @param {string[]} [argv] - optional, defaults to process.argv
|
---|
1240 | * @param {Object} [parseOptions] - optionally specify style of options with from: node/user/electron
|
---|
1241 | * @param {string} [parseOptions.from] - where the args are from: 'node', 'user', 'electron'
|
---|
1242 | * @return {Command} `this` command for chaining
|
---|
1243 | */
|
---|
1244 |
|
---|
1245 | parse(argv, parseOptions) {
|
---|
1246 | if (argv !== undefined && !Array.isArray(argv)) {
|
---|
1247 | throw new Error('first parameter to parse must be array or undefined');
|
---|
1248 | }
|
---|
1249 | parseOptions = parseOptions || {};
|
---|
1250 |
|
---|
1251 | // Default to using process.argv
|
---|
1252 | if (argv === undefined) {
|
---|
1253 | argv = process.argv;
|
---|
1254 | // @ts-ignore: unknown property
|
---|
1255 | if (process.versions && process.versions.electron) {
|
---|
1256 | parseOptions.from = 'electron';
|
---|
1257 | }
|
---|
1258 | }
|
---|
1259 | this.rawArgs = argv.slice();
|
---|
1260 |
|
---|
1261 | // make it a little easier for callers by supporting various argv conventions
|
---|
1262 | let userArgs;
|
---|
1263 | switch (parseOptions.from) {
|
---|
1264 | case undefined:
|
---|
1265 | case 'node':
|
---|
1266 | this._scriptPath = argv[1];
|
---|
1267 | userArgs = argv.slice(2);
|
---|
1268 | break;
|
---|
1269 | case 'electron':
|
---|
1270 | // @ts-ignore: unknown property
|
---|
1271 | if (process.defaultApp) {
|
---|
1272 | this._scriptPath = argv[1];
|
---|
1273 | userArgs = argv.slice(2);
|
---|
1274 | } else {
|
---|
1275 | userArgs = argv.slice(1);
|
---|
1276 | }
|
---|
1277 | break;
|
---|
1278 | case 'user':
|
---|
1279 | userArgs = argv.slice(0);
|
---|
1280 | break;
|
---|
1281 | default:
|
---|
1282 | throw new Error(`unexpected parse option { from: '${parseOptions.from}' }`);
|
---|
1283 | }
|
---|
1284 | if (!this._scriptPath && require.main) {
|
---|
1285 | this._scriptPath = require.main.filename;
|
---|
1286 | }
|
---|
1287 |
|
---|
1288 | // Guess name, used in usage in help.
|
---|
1289 | this._name = this._name || (this._scriptPath && path.basename(this._scriptPath, path.extname(this._scriptPath)));
|
---|
1290 |
|
---|
1291 | // Let's go!
|
---|
1292 | this._parseCommand([], userArgs);
|
---|
1293 |
|
---|
1294 | return this;
|
---|
1295 | };
|
---|
1296 |
|
---|
1297 | /**
|
---|
1298 | * Parse `argv`, setting options and invoking commands when defined.
|
---|
1299 | *
|
---|
1300 | * Use parseAsync instead of parse if any of your action handlers are async. Returns a Promise.
|
---|
1301 | *
|
---|
1302 | * The default expectation is that the arguments are from node and have the application as argv[0]
|
---|
1303 | * and the script being run in argv[1], with user parameters after that.
|
---|
1304 | *
|
---|
1305 | * Examples:
|
---|
1306 | *
|
---|
1307 | * program.parseAsync(process.argv);
|
---|
1308 | * program.parseAsync(); // implicitly use process.argv and auto-detect node vs electron conventions
|
---|
1309 | * program.parseAsync(my-args, { from: 'user' }); // just user supplied arguments, nothing special about argv[0]
|
---|
1310 | *
|
---|
1311 | * @param {string[]} [argv]
|
---|
1312 | * @param {Object} [parseOptions]
|
---|
1313 | * @param {string} parseOptions.from - where the args are from: 'node', 'user', 'electron'
|
---|
1314 | * @return {Promise}
|
---|
1315 | */
|
---|
1316 |
|
---|
1317 | parseAsync(argv, parseOptions) {
|
---|
1318 | this.parse(argv, parseOptions);
|
---|
1319 | return Promise.all(this._actionResults).then(() => this);
|
---|
1320 | };
|
---|
1321 |
|
---|
1322 | /**
|
---|
1323 | * Execute a sub-command executable.
|
---|
1324 | *
|
---|
1325 | * @api private
|
---|
1326 | */
|
---|
1327 |
|
---|
1328 | _executeSubCommand(subcommand, args) {
|
---|
1329 | args = args.slice();
|
---|
1330 | let launchWithNode = false; // Use node for source targets so do not need to get permissions correct, and on Windows.
|
---|
1331 | const sourceExt = ['.js', '.ts', '.tsx', '.mjs', '.cjs'];
|
---|
1332 |
|
---|
1333 | // Not checking for help first. Unlikely to have mandatory and executable, and can't robustly test for help flags in external command.
|
---|
1334 | this._checkForMissingMandatoryOptions();
|
---|
1335 |
|
---|
1336 | // Want the entry script as the reference for command name and directory for searching for other files.
|
---|
1337 | let scriptPath = this._scriptPath;
|
---|
1338 | // Fallback in case not set, due to how Command created or called.
|
---|
1339 | if (!scriptPath && require.main) {
|
---|
1340 | scriptPath = require.main.filename;
|
---|
1341 | }
|
---|
1342 |
|
---|
1343 | let baseDir;
|
---|
1344 | try {
|
---|
1345 | const resolvedLink = fs.realpathSync(scriptPath);
|
---|
1346 | baseDir = path.dirname(resolvedLink);
|
---|
1347 | } catch (e) {
|
---|
1348 | baseDir = '.'; // dummy, probably not going to find executable!
|
---|
1349 | }
|
---|
1350 |
|
---|
1351 | // name of the subcommand, like `pm-install`
|
---|
1352 | let bin = path.basename(scriptPath, path.extname(scriptPath)) + '-' + subcommand._name;
|
---|
1353 | if (subcommand._executableFile) {
|
---|
1354 | bin = subcommand._executableFile;
|
---|
1355 | }
|
---|
1356 |
|
---|
1357 | const localBin = path.join(baseDir, bin);
|
---|
1358 | if (fs.existsSync(localBin)) {
|
---|
1359 | // prefer local `./<bin>` to bin in the $PATH
|
---|
1360 | bin = localBin;
|
---|
1361 | } else {
|
---|
1362 | // Look for source files.
|
---|
1363 | sourceExt.forEach((ext) => {
|
---|
1364 | if (fs.existsSync(`${localBin}${ext}`)) {
|
---|
1365 | bin = `${localBin}${ext}`;
|
---|
1366 | }
|
---|
1367 | });
|
---|
1368 | }
|
---|
1369 | launchWithNode = sourceExt.includes(path.extname(bin));
|
---|
1370 |
|
---|
1371 | let proc;
|
---|
1372 | if (process.platform !== 'win32') {
|
---|
1373 | if (launchWithNode) {
|
---|
1374 | args.unshift(bin);
|
---|
1375 | // add executable arguments to spawn
|
---|
1376 | args = incrementNodeInspectorPort(process.execArgv).concat(args);
|
---|
1377 |
|
---|
1378 | proc = childProcess.spawn(process.argv[0], args, { stdio: 'inherit' });
|
---|
1379 | } else {
|
---|
1380 | proc = childProcess.spawn(bin, args, { stdio: 'inherit' });
|
---|
1381 | }
|
---|
1382 | } else {
|
---|
1383 | args.unshift(bin);
|
---|
1384 | // add executable arguments to spawn
|
---|
1385 | args = incrementNodeInspectorPort(process.execArgv).concat(args);
|
---|
1386 | proc = childProcess.spawn(process.execPath, args, { stdio: 'inherit' });
|
---|
1387 | }
|
---|
1388 |
|
---|
1389 | const signals = ['SIGUSR1', 'SIGUSR2', 'SIGTERM', 'SIGINT', 'SIGHUP'];
|
---|
1390 | signals.forEach((signal) => {
|
---|
1391 | // @ts-ignore
|
---|
1392 | process.on(signal, () => {
|
---|
1393 | if (proc.killed === false && proc.exitCode === null) {
|
---|
1394 | proc.kill(signal);
|
---|
1395 | }
|
---|
1396 | });
|
---|
1397 | });
|
---|
1398 |
|
---|
1399 | // By default terminate process when spawned process terminates.
|
---|
1400 | // Suppressing the exit if exitCallback defined is a bit messy and of limited use, but does allow process to stay running!
|
---|
1401 | const exitCallback = this._exitCallback;
|
---|
1402 | if (!exitCallback) {
|
---|
1403 | proc.on('close', process.exit.bind(process));
|
---|
1404 | } else {
|
---|
1405 | proc.on('close', () => {
|
---|
1406 | exitCallback(new CommanderError(process.exitCode || 0, 'commander.executeSubCommandAsync', '(close)'));
|
---|
1407 | });
|
---|
1408 | }
|
---|
1409 | proc.on('error', (err) => {
|
---|
1410 | // @ts-ignore
|
---|
1411 | if (err.code === 'ENOENT') {
|
---|
1412 | const executableMissing = `'${bin}' does not exist
|
---|
1413 | - if '${subcommand._name}' is not meant to be an executable command, remove description parameter from '.command()' and use '.description()' instead
|
---|
1414 | - if the default executable name is not suitable, use the executableFile option to supply a custom name`;
|
---|
1415 | throw new Error(executableMissing);
|
---|
1416 | // @ts-ignore
|
---|
1417 | } else if (err.code === 'EACCES') {
|
---|
1418 | throw new Error(`'${bin}' not executable`);
|
---|
1419 | }
|
---|
1420 | if (!exitCallback) {
|
---|
1421 | process.exit(1);
|
---|
1422 | } else {
|
---|
1423 | const wrappedError = new CommanderError(1, 'commander.executeSubCommandAsync', '(error)');
|
---|
1424 | wrappedError.nestedError = err;
|
---|
1425 | exitCallback(wrappedError);
|
---|
1426 | }
|
---|
1427 | });
|
---|
1428 |
|
---|
1429 | // Store the reference to the child process
|
---|
1430 | this.runningCommand = proc;
|
---|
1431 | };
|
---|
1432 |
|
---|
1433 | /**
|
---|
1434 | * @api private
|
---|
1435 | */
|
---|
1436 | _dispatchSubcommand(commandName, operands, unknown) {
|
---|
1437 | const subCommand = this._findCommand(commandName);
|
---|
1438 | if (!subCommand) this.help({ error: true });
|
---|
1439 |
|
---|
1440 | if (subCommand._executableHandler) {
|
---|
1441 | this._executeSubCommand(subCommand, operands.concat(unknown));
|
---|
1442 | } else {
|
---|
1443 | subCommand._parseCommand(operands, unknown);
|
---|
1444 | }
|
---|
1445 | };
|
---|
1446 |
|
---|
1447 | /**
|
---|
1448 | * Process arguments in context of this command.
|
---|
1449 | *
|
---|
1450 | * @api private
|
---|
1451 | */
|
---|
1452 |
|
---|
1453 | _parseCommand(operands, unknown) {
|
---|
1454 | const parsed = this.parseOptions(unknown);
|
---|
1455 | operands = operands.concat(parsed.operands);
|
---|
1456 | unknown = parsed.unknown;
|
---|
1457 | this.args = operands.concat(unknown);
|
---|
1458 |
|
---|
1459 | if (operands && this._findCommand(operands[0])) {
|
---|
1460 | this._dispatchSubcommand(operands[0], operands.slice(1), unknown);
|
---|
1461 | } else if (this._hasImplicitHelpCommand() && operands[0] === this._helpCommandName) {
|
---|
1462 | if (operands.length === 1) {
|
---|
1463 | this.help();
|
---|
1464 | } else {
|
---|
1465 | this._dispatchSubcommand(operands[1], [], [this._helpLongFlag]);
|
---|
1466 | }
|
---|
1467 | } else if (this._defaultCommandName) {
|
---|
1468 | outputHelpIfRequested(this, unknown); // Run the help for default command from parent rather than passing to default command
|
---|
1469 | this._dispatchSubcommand(this._defaultCommandName, operands, unknown);
|
---|
1470 | } else {
|
---|
1471 | if (this.commands.length && this.args.length === 0 && !this._actionHandler && !this._defaultCommandName) {
|
---|
1472 | // probably missing subcommand and no handler, user needs help
|
---|
1473 | this.help({ error: true });
|
---|
1474 | }
|
---|
1475 |
|
---|
1476 | outputHelpIfRequested(this, parsed.unknown);
|
---|
1477 | this._checkForMissingMandatoryOptions();
|
---|
1478 |
|
---|
1479 | // We do not always call this check to avoid masking a "better" error, like unknown command.
|
---|
1480 | const checkForUnknownOptions = () => {
|
---|
1481 | if (parsed.unknown.length > 0) {
|
---|
1482 | this.unknownOption(parsed.unknown[0]);
|
---|
1483 | }
|
---|
1484 | };
|
---|
1485 |
|
---|
1486 | const commandEvent = `command:${this.name()}`;
|
---|
1487 | if (this._actionHandler) {
|
---|
1488 | checkForUnknownOptions();
|
---|
1489 | // Check expected arguments and collect variadic together.
|
---|
1490 | const args = this.args.slice();
|
---|
1491 | this._args.forEach((arg, i) => {
|
---|
1492 | if (arg.required && args[i] == null) {
|
---|
1493 | this.missingArgument(arg.name);
|
---|
1494 | } else if (arg.variadic) {
|
---|
1495 | args[i] = args.splice(i);
|
---|
1496 | args.length = Math.min(i + 1, args.length);
|
---|
1497 | }
|
---|
1498 | });
|
---|
1499 | if (args.length > this._args.length) {
|
---|
1500 | this._excessArguments(args);
|
---|
1501 | }
|
---|
1502 |
|
---|
1503 | this._actionHandler(args);
|
---|
1504 | if (this.parent) this.parent.emit(commandEvent, operands, unknown); // legacy
|
---|
1505 | } else if (this.parent && this.parent.listenerCount(commandEvent)) {
|
---|
1506 | checkForUnknownOptions();
|
---|
1507 | this.parent.emit(commandEvent, operands, unknown); // legacy
|
---|
1508 | } else if (operands.length) {
|
---|
1509 | if (this._findCommand('*')) { // legacy default command
|
---|
1510 | this._dispatchSubcommand('*', operands, unknown);
|
---|
1511 | } else if (this.listenerCount('command:*')) {
|
---|
1512 | // skip option check, emit event for possible misspelling suggestion
|
---|
1513 | this.emit('command:*', operands, unknown);
|
---|
1514 | } else if (this.commands.length) {
|
---|
1515 | this.unknownCommand();
|
---|
1516 | } else {
|
---|
1517 | checkForUnknownOptions();
|
---|
1518 | }
|
---|
1519 | } else if (this.commands.length) {
|
---|
1520 | // This command has subcommands and nothing hooked up at this level, so display help.
|
---|
1521 | this.help({ error: true });
|
---|
1522 | } else {
|
---|
1523 | checkForUnknownOptions();
|
---|
1524 | // fall through for caller to handle after calling .parse()
|
---|
1525 | }
|
---|
1526 | }
|
---|
1527 | };
|
---|
1528 |
|
---|
1529 | /**
|
---|
1530 | * Find matching command.
|
---|
1531 | *
|
---|
1532 | * @api private
|
---|
1533 | */
|
---|
1534 | _findCommand(name) {
|
---|
1535 | if (!name) return undefined;
|
---|
1536 | return this.commands.find(cmd => cmd._name === name || cmd._aliases.includes(name));
|
---|
1537 | };
|
---|
1538 |
|
---|
1539 | /**
|
---|
1540 | * Return an option matching `arg` if any.
|
---|
1541 | *
|
---|
1542 | * @param {string} arg
|
---|
1543 | * @return {Option}
|
---|
1544 | * @api private
|
---|
1545 | */
|
---|
1546 |
|
---|
1547 | _findOption(arg) {
|
---|
1548 | return this.options.find(option => option.is(arg));
|
---|
1549 | };
|
---|
1550 |
|
---|
1551 | /**
|
---|
1552 | * Display an error message if a mandatory option does not have a value.
|
---|
1553 | * Lazy calling after checking for help flags from leaf subcommand.
|
---|
1554 | *
|
---|
1555 | * @api private
|
---|
1556 | */
|
---|
1557 |
|
---|
1558 | _checkForMissingMandatoryOptions() {
|
---|
1559 | // Walk up hierarchy so can call in subcommand after checking for displaying help.
|
---|
1560 | for (let cmd = this; cmd; cmd = cmd.parent) {
|
---|
1561 | cmd.options.forEach((anOption) => {
|
---|
1562 | if (anOption.mandatory && (cmd._getOptionValue(anOption.attributeName()) === undefined)) {
|
---|
1563 | cmd.missingMandatoryOptionValue(anOption);
|
---|
1564 | }
|
---|
1565 | });
|
---|
1566 | }
|
---|
1567 | };
|
---|
1568 |
|
---|
1569 | /**
|
---|
1570 | * Parse options from `argv` removing known options,
|
---|
1571 | * and return argv split into operands and unknown arguments.
|
---|
1572 | *
|
---|
1573 | * Examples:
|
---|
1574 | *
|
---|
1575 | * argv => operands, unknown
|
---|
1576 | * --known kkk op => [op], []
|
---|
1577 | * op --known kkk => [op], []
|
---|
1578 | * sub --unknown uuu op => [sub], [--unknown uuu op]
|
---|
1579 | * sub -- --unknown uuu op => [sub --unknown uuu op], []
|
---|
1580 | *
|
---|
1581 | * @param {String[]} argv
|
---|
1582 | * @return {{operands: String[], unknown: String[]}}
|
---|
1583 | */
|
---|
1584 |
|
---|
1585 | parseOptions(argv) {
|
---|
1586 | const operands = []; // operands, not options or values
|
---|
1587 | const unknown = []; // first unknown option and remaining unknown args
|
---|
1588 | let dest = operands;
|
---|
1589 | const args = argv.slice();
|
---|
1590 |
|
---|
1591 | function maybeOption(arg) {
|
---|
1592 | return arg.length > 1 && arg[0] === '-';
|
---|
1593 | }
|
---|
1594 |
|
---|
1595 | // parse options
|
---|
1596 | let activeVariadicOption = null;
|
---|
1597 | while (args.length) {
|
---|
1598 | const arg = args.shift();
|
---|
1599 |
|
---|
1600 | // literal
|
---|
1601 | if (arg === '--') {
|
---|
1602 | if (dest === unknown) dest.push(arg);
|
---|
1603 | dest.push(...args);
|
---|
1604 | break;
|
---|
1605 | }
|
---|
1606 |
|
---|
1607 | if (activeVariadicOption && !maybeOption(arg)) {
|
---|
1608 | this.emit(`option:${activeVariadicOption.name()}`, arg);
|
---|
1609 | continue;
|
---|
1610 | }
|
---|
1611 | activeVariadicOption = null;
|
---|
1612 |
|
---|
1613 | if (maybeOption(arg)) {
|
---|
1614 | const option = this._findOption(arg);
|
---|
1615 | // recognised option, call listener to assign value with possible custom processing
|
---|
1616 | if (option) {
|
---|
1617 | if (option.required) {
|
---|
1618 | const value = args.shift();
|
---|
1619 | if (value === undefined) this.optionMissingArgument(option);
|
---|
1620 | this.emit(`option:${option.name()}`, value);
|
---|
1621 | } else if (option.optional) {
|
---|
1622 | let value = null;
|
---|
1623 | // historical behaviour is optional value is following arg unless an option
|
---|
1624 | if (args.length > 0 && !maybeOption(args[0])) {
|
---|
1625 | value = args.shift();
|
---|
1626 | }
|
---|
1627 | this.emit(`option:${option.name()}`, value);
|
---|
1628 | } else { // boolean flag
|
---|
1629 | this.emit(`option:${option.name()}`);
|
---|
1630 | }
|
---|
1631 | activeVariadicOption = option.variadic ? option : null;
|
---|
1632 | continue;
|
---|
1633 | }
|
---|
1634 | }
|
---|
1635 |
|
---|
1636 | // Look for combo options following single dash, eat first one if known.
|
---|
1637 | if (arg.length > 2 && arg[0] === '-' && arg[1] !== '-') {
|
---|
1638 | const option = this._findOption(`-${arg[1]}`);
|
---|
1639 | if (option) {
|
---|
1640 | if (option.required || (option.optional && this._combineFlagAndOptionalValue)) {
|
---|
1641 | // option with value following in same argument
|
---|
1642 | this.emit(`option:${option.name()}`, arg.slice(2));
|
---|
1643 | } else {
|
---|
1644 | // boolean option, emit and put back remainder of arg for further processing
|
---|
1645 | this.emit(`option:${option.name()}`);
|
---|
1646 | args.unshift(`-${arg.slice(2)}`);
|
---|
1647 | }
|
---|
1648 | continue;
|
---|
1649 | }
|
---|
1650 | }
|
---|
1651 |
|
---|
1652 | // Look for known long flag with value, like --foo=bar
|
---|
1653 | if (/^--[^=]+=/.test(arg)) {
|
---|
1654 | const index = arg.indexOf('=');
|
---|
1655 | const option = this._findOption(arg.slice(0, index));
|
---|
1656 | if (option && (option.required || option.optional)) {
|
---|
1657 | this.emit(`option:${option.name()}`, arg.slice(index + 1));
|
---|
1658 | continue;
|
---|
1659 | }
|
---|
1660 | }
|
---|
1661 |
|
---|
1662 | // Not a recognised option by this command.
|
---|
1663 | // Might be a command-argument, or subcommand option, or unknown option, or help command or option.
|
---|
1664 |
|
---|
1665 | // An unknown option means further arguments also classified as unknown so can be reprocessed by subcommands.
|
---|
1666 | if (maybeOption(arg)) {
|
---|
1667 | dest = unknown;
|
---|
1668 | }
|
---|
1669 |
|
---|
1670 | // If using positionalOptions, stop processing our options at subcommand.
|
---|
1671 | if ((this._enablePositionalOptions || this._passThroughOptions) && operands.length === 0 && unknown.length === 0) {
|
---|
1672 | if (this._findCommand(arg)) {
|
---|
1673 | operands.push(arg);
|
---|
1674 | if (args.length > 0) unknown.push(...args);
|
---|
1675 | break;
|
---|
1676 | } else if (arg === this._helpCommandName && this._hasImplicitHelpCommand()) {
|
---|
1677 | operands.push(arg);
|
---|
1678 | if (args.length > 0) operands.push(...args);
|
---|
1679 | break;
|
---|
1680 | } else if (this._defaultCommandName) {
|
---|
1681 | unknown.push(arg);
|
---|
1682 | if (args.length > 0) unknown.push(...args);
|
---|
1683 | break;
|
---|
1684 | }
|
---|
1685 | }
|
---|
1686 |
|
---|
1687 | // If using passThroughOptions, stop processing options at first command-argument.
|
---|
1688 | if (this._passThroughOptions) {
|
---|
1689 | dest.push(arg);
|
---|
1690 | if (args.length > 0) dest.push(...args);
|
---|
1691 | break;
|
---|
1692 | }
|
---|
1693 |
|
---|
1694 | // add arg
|
---|
1695 | dest.push(arg);
|
---|
1696 | }
|
---|
1697 |
|
---|
1698 | return { operands, unknown };
|
---|
1699 | };
|
---|
1700 |
|
---|
1701 | /**
|
---|
1702 | * Return an object containing options as key-value pairs
|
---|
1703 | *
|
---|
1704 | * @return {Object}
|
---|
1705 | */
|
---|
1706 | opts() {
|
---|
1707 | if (this._storeOptionsAsProperties) {
|
---|
1708 | // Preserve original behaviour so backwards compatible when still using properties
|
---|
1709 | const result = {};
|
---|
1710 | const len = this.options.length;
|
---|
1711 |
|
---|
1712 | for (let i = 0; i < len; i++) {
|
---|
1713 | const key = this.options[i].attributeName();
|
---|
1714 | result[key] = key === this._versionOptionName ? this._version : this[key];
|
---|
1715 | }
|
---|
1716 | return result;
|
---|
1717 | }
|
---|
1718 |
|
---|
1719 | return this._optionValues;
|
---|
1720 | };
|
---|
1721 |
|
---|
1722 | /**
|
---|
1723 | * Internal bottleneck for handling of parsing errors.
|
---|
1724 | *
|
---|
1725 | * @api private
|
---|
1726 | */
|
---|
1727 | _displayError(exitCode, code, message) {
|
---|
1728 | this._outputConfiguration.outputError(`${message}\n`, this._outputConfiguration.writeErr);
|
---|
1729 | this._exit(exitCode, code, message);
|
---|
1730 | }
|
---|
1731 |
|
---|
1732 | /**
|
---|
1733 | * Argument `name` is missing.
|
---|
1734 | *
|
---|
1735 | * @param {string} name
|
---|
1736 | * @api private
|
---|
1737 | */
|
---|
1738 |
|
---|
1739 | missingArgument(name) {
|
---|
1740 | const message = `error: missing required argument '${name}'`;
|
---|
1741 | this._displayError(1, 'commander.missingArgument', message);
|
---|
1742 | };
|
---|
1743 |
|
---|
1744 | /**
|
---|
1745 | * `Option` is missing an argument.
|
---|
1746 | *
|
---|
1747 | * @param {Option} option
|
---|
1748 | * @api private
|
---|
1749 | */
|
---|
1750 |
|
---|
1751 | optionMissingArgument(option) {
|
---|
1752 | const message = `error: option '${option.flags}' argument missing`;
|
---|
1753 | this._displayError(1, 'commander.optionMissingArgument', message);
|
---|
1754 | };
|
---|
1755 |
|
---|
1756 | /**
|
---|
1757 | * `Option` does not have a value, and is a mandatory option.
|
---|
1758 | *
|
---|
1759 | * @param {Option} option
|
---|
1760 | * @api private
|
---|
1761 | */
|
---|
1762 |
|
---|
1763 | missingMandatoryOptionValue(option) {
|
---|
1764 | const message = `error: required option '${option.flags}' not specified`;
|
---|
1765 | this._displayError(1, 'commander.missingMandatoryOptionValue', message);
|
---|
1766 | };
|
---|
1767 |
|
---|
1768 | /**
|
---|
1769 | * Unknown option `flag`.
|
---|
1770 | *
|
---|
1771 | * @param {string} flag
|
---|
1772 | * @api private
|
---|
1773 | */
|
---|
1774 |
|
---|
1775 | unknownOption(flag) {
|
---|
1776 | if (this._allowUnknownOption) return;
|
---|
1777 | const message = `error: unknown option '${flag}'`;
|
---|
1778 | this._displayError(1, 'commander.unknownOption', message);
|
---|
1779 | };
|
---|
1780 |
|
---|
1781 | /**
|
---|
1782 | * Excess arguments, more than expected.
|
---|
1783 | *
|
---|
1784 | * @param {string[]} receivedArgs
|
---|
1785 | * @api private
|
---|
1786 | */
|
---|
1787 |
|
---|
1788 | _excessArguments(receivedArgs) {
|
---|
1789 | if (this._allowExcessArguments) return;
|
---|
1790 |
|
---|
1791 | const expected = this._args.length;
|
---|
1792 | const s = (expected === 1) ? '' : 's';
|
---|
1793 | const forSubcommand = this.parent ? ` for '${this.name()}'` : '';
|
---|
1794 | const message = `error: too many arguments${forSubcommand}. Expected ${expected} argument${s} but got ${receivedArgs.length}.`;
|
---|
1795 | this._displayError(1, 'commander.excessArguments', message);
|
---|
1796 | };
|
---|
1797 |
|
---|
1798 | /**
|
---|
1799 | * Unknown command.
|
---|
1800 | *
|
---|
1801 | * @api private
|
---|
1802 | */
|
---|
1803 |
|
---|
1804 | unknownCommand() {
|
---|
1805 | const partCommands = [this.name()];
|
---|
1806 | for (let parentCmd = this.parent; parentCmd; parentCmd = parentCmd.parent) {
|
---|
1807 | partCommands.unshift(parentCmd.name());
|
---|
1808 | }
|
---|
1809 | const fullCommand = partCommands.join(' ');
|
---|
1810 | const message = `error: unknown command '${this.args[0]}'.` +
|
---|
1811 | (this._hasHelpOption ? ` See '${fullCommand} ${this._helpLongFlag}'.` : '');
|
---|
1812 | this._displayError(1, 'commander.unknownCommand', message);
|
---|
1813 | };
|
---|
1814 |
|
---|
1815 | /**
|
---|
1816 | * Set the program version to `str`.
|
---|
1817 | *
|
---|
1818 | * This method auto-registers the "-V, --version" flag
|
---|
1819 | * which will print the version number when passed.
|
---|
1820 | *
|
---|
1821 | * You can optionally supply the flags and description to override the defaults.
|
---|
1822 | *
|
---|
1823 | * @param {string} str
|
---|
1824 | * @param {string} [flags]
|
---|
1825 | * @param {string} [description]
|
---|
1826 | * @return {this | string} `this` command for chaining, or version string if no arguments
|
---|
1827 | */
|
---|
1828 |
|
---|
1829 | version(str, flags, description) {
|
---|
1830 | if (str === undefined) return this._version;
|
---|
1831 | this._version = str;
|
---|
1832 | flags = flags || '-V, --version';
|
---|
1833 | description = description || 'output the version number';
|
---|
1834 | const versionOption = this.createOption(flags, description);
|
---|
1835 | this._versionOptionName = versionOption.attributeName();
|
---|
1836 | this.options.push(versionOption);
|
---|
1837 | this.on('option:' + versionOption.name(), () => {
|
---|
1838 | this._outputConfiguration.writeOut(`${str}\n`);
|
---|
1839 | this._exit(0, 'commander.version', str);
|
---|
1840 | });
|
---|
1841 | return this;
|
---|
1842 | };
|
---|
1843 |
|
---|
1844 | /**
|
---|
1845 | * Set the description to `str`.
|
---|
1846 | *
|
---|
1847 | * @param {string} [str]
|
---|
1848 | * @param {Object} [argsDescription]
|
---|
1849 | * @return {string|Command}
|
---|
1850 | */
|
---|
1851 | description(str, argsDescription) {
|
---|
1852 | if (str === undefined && argsDescription === undefined) return this._description;
|
---|
1853 | this._description = str;
|
---|
1854 | this._argsDescription = argsDescription;
|
---|
1855 | return this;
|
---|
1856 | };
|
---|
1857 |
|
---|
1858 | /**
|
---|
1859 | * Set an alias for the command.
|
---|
1860 | *
|
---|
1861 | * You may call more than once to add multiple aliases. Only the first alias is shown in the auto-generated help.
|
---|
1862 | *
|
---|
1863 | * @param {string} [alias]
|
---|
1864 | * @return {string|Command}
|
---|
1865 | */
|
---|
1866 |
|
---|
1867 | alias(alias) {
|
---|
1868 | if (alias === undefined) return this._aliases[0]; // just return first, for backwards compatibility
|
---|
1869 |
|
---|
1870 | let command = this;
|
---|
1871 | if (this.commands.length !== 0 && this.commands[this.commands.length - 1]._executableHandler) {
|
---|
1872 | // assume adding alias for last added executable subcommand, rather than this
|
---|
1873 | command = this.commands[this.commands.length - 1];
|
---|
1874 | }
|
---|
1875 |
|
---|
1876 | if (alias === command._name) throw new Error('Command alias can\'t be the same as its name');
|
---|
1877 |
|
---|
1878 | command._aliases.push(alias);
|
---|
1879 | return this;
|
---|
1880 | };
|
---|
1881 |
|
---|
1882 | /**
|
---|
1883 | * Set aliases for the command.
|
---|
1884 | *
|
---|
1885 | * Only the first alias is shown in the auto-generated help.
|
---|
1886 | *
|
---|
1887 | * @param {string[]} [aliases]
|
---|
1888 | * @return {string[]|Command}
|
---|
1889 | */
|
---|
1890 |
|
---|
1891 | aliases(aliases) {
|
---|
1892 | // Getter for the array of aliases is the main reason for having aliases() in addition to alias().
|
---|
1893 | if (aliases === undefined) return this._aliases;
|
---|
1894 |
|
---|
1895 | aliases.forEach((alias) => this.alias(alias));
|
---|
1896 | return this;
|
---|
1897 | };
|
---|
1898 |
|
---|
1899 | /**
|
---|
1900 | * Set / get the command usage `str`.
|
---|
1901 | *
|
---|
1902 | * @param {string} [str]
|
---|
1903 | * @return {String|Command}
|
---|
1904 | */
|
---|
1905 |
|
---|
1906 | usage(str) {
|
---|
1907 | if (str === undefined) {
|
---|
1908 | if (this._usage) return this._usage;
|
---|
1909 |
|
---|
1910 | const args = this._args.map((arg) => {
|
---|
1911 | return humanReadableArgName(arg);
|
---|
1912 | });
|
---|
1913 | return [].concat(
|
---|
1914 | (this.options.length || this._hasHelpOption ? '[options]' : []),
|
---|
1915 | (this.commands.length ? '[command]' : []),
|
---|
1916 | (this._args.length ? args : [])
|
---|
1917 | ).join(' ');
|
---|
1918 | }
|
---|
1919 |
|
---|
1920 | this._usage = str;
|
---|
1921 | return this;
|
---|
1922 | };
|
---|
1923 |
|
---|
1924 | /**
|
---|
1925 | * Get or set the name of the command
|
---|
1926 | *
|
---|
1927 | * @param {string} [str]
|
---|
1928 | * @return {string|Command}
|
---|
1929 | */
|
---|
1930 |
|
---|
1931 | name(str) {
|
---|
1932 | if (str === undefined) return this._name;
|
---|
1933 | this._name = str;
|
---|
1934 | return this;
|
---|
1935 | };
|
---|
1936 |
|
---|
1937 | /**
|
---|
1938 | * Return program help documentation.
|
---|
1939 | *
|
---|
1940 | * @param {{ error: boolean }} [contextOptions] - pass {error:true} to wrap for stderr instead of stdout
|
---|
1941 | * @return {string}
|
---|
1942 | */
|
---|
1943 |
|
---|
1944 | helpInformation(contextOptions) {
|
---|
1945 | const helper = this.createHelp();
|
---|
1946 | if (helper.helpWidth === undefined) {
|
---|
1947 | helper.helpWidth = (contextOptions && contextOptions.error) ? this._outputConfiguration.getErrHelpWidth() : this._outputConfiguration.getOutHelpWidth();
|
---|
1948 | }
|
---|
1949 | return helper.formatHelp(this, helper);
|
---|
1950 | };
|
---|
1951 |
|
---|
1952 | /**
|
---|
1953 | * @api private
|
---|
1954 | */
|
---|
1955 |
|
---|
1956 | _getHelpContext(contextOptions) {
|
---|
1957 | contextOptions = contextOptions || {};
|
---|
1958 | const context = { error: !!contextOptions.error };
|
---|
1959 | let write;
|
---|
1960 | if (context.error) {
|
---|
1961 | write = (arg) => this._outputConfiguration.writeErr(arg);
|
---|
1962 | } else {
|
---|
1963 | write = (arg) => this._outputConfiguration.writeOut(arg);
|
---|
1964 | }
|
---|
1965 | context.write = contextOptions.write || write;
|
---|
1966 | context.command = this;
|
---|
1967 | return context;
|
---|
1968 | }
|
---|
1969 |
|
---|
1970 | /**
|
---|
1971 | * Output help information for this command.
|
---|
1972 | *
|
---|
1973 | * Outputs built-in help, and custom text added using `.addHelpText()`.
|
---|
1974 | *
|
---|
1975 | * @param {{ error: boolean } | Function} [contextOptions] - pass {error:true} to write to stderr instead of stdout
|
---|
1976 | */
|
---|
1977 |
|
---|
1978 | outputHelp(contextOptions) {
|
---|
1979 | let deprecatedCallback;
|
---|
1980 | if (typeof contextOptions === 'function') {
|
---|
1981 | deprecatedCallback = contextOptions;
|
---|
1982 | contextOptions = undefined;
|
---|
1983 | }
|
---|
1984 | const context = this._getHelpContext(contextOptions);
|
---|
1985 |
|
---|
1986 | const groupListeners = [];
|
---|
1987 | let command = this;
|
---|
1988 | while (command) {
|
---|
1989 | groupListeners.push(command); // ordered from current command to root
|
---|
1990 | command = command.parent;
|
---|
1991 | }
|
---|
1992 |
|
---|
1993 | groupListeners.slice().reverse().forEach(command => command.emit('beforeAllHelp', context));
|
---|
1994 | this.emit('beforeHelp', context);
|
---|
1995 |
|
---|
1996 | let helpInformation = this.helpInformation(context);
|
---|
1997 | if (deprecatedCallback) {
|
---|
1998 | helpInformation = deprecatedCallback(helpInformation);
|
---|
1999 | if (typeof helpInformation !== 'string' && !Buffer.isBuffer(helpInformation)) {
|
---|
2000 | throw new Error('outputHelp callback must return a string or a Buffer');
|
---|
2001 | }
|
---|
2002 | }
|
---|
2003 | context.write(helpInformation);
|
---|
2004 |
|
---|
2005 | this.emit(this._helpLongFlag); // deprecated
|
---|
2006 | this.emit('afterHelp', context);
|
---|
2007 | groupListeners.forEach(command => command.emit('afterAllHelp', context));
|
---|
2008 | };
|
---|
2009 |
|
---|
2010 | /**
|
---|
2011 | * You can pass in flags and a description to override the help
|
---|
2012 | * flags and help description for your command. Pass in false to
|
---|
2013 | * disable the built-in help option.
|
---|
2014 | *
|
---|
2015 | * @param {string | boolean} [flags]
|
---|
2016 | * @param {string} [description]
|
---|
2017 | * @return {Command} `this` command for chaining
|
---|
2018 | */
|
---|
2019 |
|
---|
2020 | helpOption(flags, description) {
|
---|
2021 | if (typeof flags === 'boolean') {
|
---|
2022 | this._hasHelpOption = flags;
|
---|
2023 | return this;
|
---|
2024 | }
|
---|
2025 | this._helpFlags = flags || this._helpFlags;
|
---|
2026 | this._helpDescription = description || this._helpDescription;
|
---|
2027 |
|
---|
2028 | const helpFlags = _parseOptionFlags(this._helpFlags);
|
---|
2029 | this._helpShortFlag = helpFlags.shortFlag;
|
---|
2030 | this._helpLongFlag = helpFlags.longFlag;
|
---|
2031 |
|
---|
2032 | return this;
|
---|
2033 | };
|
---|
2034 |
|
---|
2035 | /**
|
---|
2036 | * Output help information and exit.
|
---|
2037 | *
|
---|
2038 | * Outputs built-in help, and custom text added using `.addHelpText()`.
|
---|
2039 | *
|
---|
2040 | * @param {{ error: boolean }} [contextOptions] - pass {error:true} to write to stderr instead of stdout
|
---|
2041 | */
|
---|
2042 |
|
---|
2043 | help(contextOptions) {
|
---|
2044 | this.outputHelp(contextOptions);
|
---|
2045 | let exitCode = process.exitCode || 0;
|
---|
2046 | if (exitCode === 0 && contextOptions && typeof contextOptions !== 'function' && contextOptions.error) {
|
---|
2047 | exitCode = 1;
|
---|
2048 | }
|
---|
2049 | // message: do not have all displayed text available so only passing placeholder.
|
---|
2050 | this._exit(exitCode, 'commander.help', '(outputHelp)');
|
---|
2051 | };
|
---|
2052 |
|
---|
2053 | /**
|
---|
2054 | * Add additional text to be displayed with the built-in help.
|
---|
2055 | *
|
---|
2056 | * Position is 'before' or 'after' to affect just this command,
|
---|
2057 | * and 'beforeAll' or 'afterAll' to affect this command and all its subcommands.
|
---|
2058 | *
|
---|
2059 | * @param {string} position - before or after built-in help
|
---|
2060 | * @param {string | Function} text - string to add, or a function returning a string
|
---|
2061 | * @return {Command} `this` command for chaining
|
---|
2062 | */
|
---|
2063 | addHelpText(position, text) {
|
---|
2064 | const allowedValues = ['beforeAll', 'before', 'after', 'afterAll'];
|
---|
2065 | if (!allowedValues.includes(position)) {
|
---|
2066 | throw new Error(`Unexpected value for position to addHelpText.
|
---|
2067 | Expecting one of '${allowedValues.join("', '")}'`);
|
---|
2068 | }
|
---|
2069 | const helpEvent = `${position}Help`;
|
---|
2070 | this.on(helpEvent, (context) => {
|
---|
2071 | let helpStr;
|
---|
2072 | if (typeof text === 'function') {
|
---|
2073 | helpStr = text({ error: context.error, command: context.command });
|
---|
2074 | } else {
|
---|
2075 | helpStr = text;
|
---|
2076 | }
|
---|
2077 | // Ignore falsy value when nothing to output.
|
---|
2078 | if (helpStr) {
|
---|
2079 | context.write(`${helpStr}\n`);
|
---|
2080 | }
|
---|
2081 | });
|
---|
2082 | return this;
|
---|
2083 | }
|
---|
2084 | };
|
---|
2085 |
|
---|
2086 | /**
|
---|
2087 | * Expose the root command.
|
---|
2088 | */
|
---|
2089 |
|
---|
2090 | exports = module.exports = new Command();
|
---|
2091 | exports.program = exports; // More explicit access to global command.
|
---|
2092 |
|
---|
2093 | /**
|
---|
2094 | * Expose classes
|
---|
2095 | */
|
---|
2096 |
|
---|
2097 | exports.Command = Command;
|
---|
2098 | exports.Option = Option;
|
---|
2099 | exports.CommanderError = CommanderError;
|
---|
2100 | exports.InvalidOptionArgumentError = InvalidOptionArgumentError;
|
---|
2101 | exports.Help = Help;
|
---|
2102 |
|
---|
2103 | /**
|
---|
2104 | * Camel-case the given `flag`
|
---|
2105 | *
|
---|
2106 | * @param {string} flag
|
---|
2107 | * @return {string}
|
---|
2108 | * @api private
|
---|
2109 | */
|
---|
2110 |
|
---|
2111 | function camelcase(flag) {
|
---|
2112 | return flag.split('-').reduce((str, word) => {
|
---|
2113 | return str + word[0].toUpperCase() + word.slice(1);
|
---|
2114 | });
|
---|
2115 | }
|
---|
2116 |
|
---|
2117 | /**
|
---|
2118 | * Output help information if help flags specified
|
---|
2119 | *
|
---|
2120 | * @param {Command} cmd - command to output help for
|
---|
2121 | * @param {Array} args - array of options to search for help flags
|
---|
2122 | * @api private
|
---|
2123 | */
|
---|
2124 |
|
---|
2125 | function outputHelpIfRequested(cmd, args) {
|
---|
2126 | const helpOption = cmd._hasHelpOption && args.find(arg => arg === cmd._helpLongFlag || arg === cmd._helpShortFlag);
|
---|
2127 | if (helpOption) {
|
---|
2128 | cmd.outputHelp();
|
---|
2129 | // (Do not have all displayed text available so only passing placeholder.)
|
---|
2130 | cmd._exit(0, 'commander.helpDisplayed', '(outputHelp)');
|
---|
2131 | }
|
---|
2132 | }
|
---|
2133 |
|
---|
2134 | /**
|
---|
2135 | * Takes an argument and returns its human readable equivalent for help usage.
|
---|
2136 | *
|
---|
2137 | * @param {Object} arg
|
---|
2138 | * @return {string}
|
---|
2139 | * @api private
|
---|
2140 | */
|
---|
2141 |
|
---|
2142 | function humanReadableArgName(arg) {
|
---|
2143 | const nameOutput = arg.name + (arg.variadic === true ? '...' : '');
|
---|
2144 |
|
---|
2145 | return arg.required
|
---|
2146 | ? '<' + nameOutput + '>'
|
---|
2147 | : '[' + nameOutput + ']';
|
---|
2148 | }
|
---|
2149 |
|
---|
2150 | /**
|
---|
2151 | * Parse the short and long flag out of something like '-m,--mixed <value>'
|
---|
2152 | *
|
---|
2153 | * @api private
|
---|
2154 | */
|
---|
2155 |
|
---|
2156 | function _parseOptionFlags(flags) {
|
---|
2157 | let shortFlag;
|
---|
2158 | let longFlag;
|
---|
2159 | // Use original very loose parsing to maintain backwards compatibility for now,
|
---|
2160 | // which allowed for example unintended `-sw, --short-word` [sic].
|
---|
2161 | const flagParts = flags.split(/[ |,]+/);
|
---|
2162 | if (flagParts.length > 1 && !/^[[<]/.test(flagParts[1])) shortFlag = flagParts.shift();
|
---|
2163 | longFlag = flagParts.shift();
|
---|
2164 | // Add support for lone short flag without significantly changing parsing!
|
---|
2165 | if (!shortFlag && /^-[^-]$/.test(longFlag)) {
|
---|
2166 | shortFlag = longFlag;
|
---|
2167 | longFlag = undefined;
|
---|
2168 | }
|
---|
2169 | return { shortFlag, longFlag };
|
---|
2170 | }
|
---|
2171 |
|
---|
2172 | /**
|
---|
2173 | * Scan arguments and increment port number for inspect calls (to avoid conflicts when spawning new command).
|
---|
2174 | *
|
---|
2175 | * @param {string[]} args - array of arguments from node.execArgv
|
---|
2176 | * @returns {string[]}
|
---|
2177 | * @api private
|
---|
2178 | */
|
---|
2179 |
|
---|
2180 | function incrementNodeInspectorPort(args) {
|
---|
2181 | // Testing for these options:
|
---|
2182 | // --inspect[=[host:]port]
|
---|
2183 | // --inspect-brk[=[host:]port]
|
---|
2184 | // --inspect-port=[host:]port
|
---|
2185 | return args.map((arg) => {
|
---|
2186 | if (!arg.startsWith('--inspect')) {
|
---|
2187 | return arg;
|
---|
2188 | }
|
---|
2189 | let debugOption;
|
---|
2190 | let debugHost = '127.0.0.1';
|
---|
2191 | let debugPort = '9229';
|
---|
2192 | let match;
|
---|
2193 | if ((match = arg.match(/^(--inspect(-brk)?)$/)) !== null) {
|
---|
2194 | // e.g. --inspect
|
---|
2195 | debugOption = match[1];
|
---|
2196 | } else if ((match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+)$/)) !== null) {
|
---|
2197 | debugOption = match[1];
|
---|
2198 | if (/^\d+$/.test(match[3])) {
|
---|
2199 | // e.g. --inspect=1234
|
---|
2200 | debugPort = match[3];
|
---|
2201 | } else {
|
---|
2202 | // e.g. --inspect=localhost
|
---|
2203 | debugHost = match[3];
|
---|
2204 | }
|
---|
2205 | } else if ((match = arg.match(/^(--inspect(-brk|-port)?)=([^:]+):(\d+)$/)) !== null) {
|
---|
2206 | // e.g. --inspect=localhost:1234
|
---|
2207 | debugOption = match[1];
|
---|
2208 | debugHost = match[3];
|
---|
2209 | debugPort = match[4];
|
---|
2210 | }
|
---|
2211 |
|
---|
2212 | if (debugOption && debugPort !== '0') {
|
---|
2213 | return `${debugOption}=${debugHost}:${parseInt(debugPort) + 1}`;
|
---|
2214 | }
|
---|
2215 | return arg;
|
---|
2216 | });
|
---|
2217 | }
|
---|