coupeeeez 0.0.1 → 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +2 -1
- package/stream-api/node_modules/.package-lock.json +0 -9
- package/stream-api/package-lock.json +0 -10
- package/stream-api/package.json +0 -1
- package/stream-api/node_modules/commander/LICENSE +0 -22
- package/stream-api/node_modules/commander/Readme.md +0 -1159
- package/stream-api/node_modules/commander/esm.mjs +0 -16
- package/stream-api/node_modules/commander/index.js +0 -24
- package/stream-api/node_modules/commander/lib/argument.js +0 -150
- package/stream-api/node_modules/commander/lib/command.js +0 -2777
- package/stream-api/node_modules/commander/lib/error.js +0 -39
- package/stream-api/node_modules/commander/lib/help.js +0 -747
- package/stream-api/node_modules/commander/lib/option.js +0 -380
- package/stream-api/node_modules/commander/lib/suggestSimilar.js +0 -101
- package/stream-api/node_modules/commander/package-support.json +0 -16
- package/stream-api/node_modules/commander/package.json +0 -82
- package/stream-api/node_modules/commander/typings/esm.d.mts +0 -3
- package/stream-api/node_modules/commander/typings/index.d.ts +0 -1113
|
@@ -1,747 +0,0 @@
|
|
|
1
|
-
const { humanReadableArgName } = require('./argument.js');
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* TypeScript import types for JSDoc, used by Visual Studio Code IntelliSense and `npm run typescript-checkJS`
|
|
5
|
-
* https://www.typescriptlang.org/docs/handbook/jsdoc-supported-types.html#import-types
|
|
6
|
-
* @typedef { import("./argument.js").Argument } Argument
|
|
7
|
-
* @typedef { import("./command.js").Command } Command
|
|
8
|
-
* @typedef { import("./option.js").Option } Option
|
|
9
|
-
*/
|
|
10
|
-
|
|
11
|
-
// Although this is a class, methods are static in style to allow override using subclass or just functions.
|
|
12
|
-
class Help {
|
|
13
|
-
constructor() {
|
|
14
|
-
this.helpWidth = undefined;
|
|
15
|
-
this.minWidthToWrap = 40;
|
|
16
|
-
this.sortSubcommands = false;
|
|
17
|
-
this.sortOptions = false;
|
|
18
|
-
this.showGlobalOptions = false;
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
/**
|
|
22
|
-
* prepareContext is called by Commander after applying overrides from `Command.configureHelp()`
|
|
23
|
-
* and just before calling `formatHelp()`.
|
|
24
|
-
*
|
|
25
|
-
* Commander just uses the helpWidth and the rest is provided for optional use by more complex subclasses.
|
|
26
|
-
*
|
|
27
|
-
* @param {{ error?: boolean, helpWidth?: number, outputHasColors?: boolean }} contextOptions
|
|
28
|
-
*/
|
|
29
|
-
prepareContext(contextOptions) {
|
|
30
|
-
this.helpWidth = this.helpWidth ?? contextOptions.helpWidth ?? 80;
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/**
|
|
34
|
-
* Get an array of the visible subcommands. Includes a placeholder for the implicit help command, if there is one.
|
|
35
|
-
*
|
|
36
|
-
* @param {Command} cmd
|
|
37
|
-
* @returns {Command[]}
|
|
38
|
-
*/
|
|
39
|
-
|
|
40
|
-
visibleCommands(cmd) {
|
|
41
|
-
const visibleCommands = cmd.commands.filter((cmd) => !cmd._hidden);
|
|
42
|
-
const helpCommand = cmd._getHelpCommand();
|
|
43
|
-
if (helpCommand && !helpCommand._hidden) {
|
|
44
|
-
visibleCommands.push(helpCommand);
|
|
45
|
-
}
|
|
46
|
-
if (this.sortSubcommands) {
|
|
47
|
-
visibleCommands.sort((a, b) => {
|
|
48
|
-
// @ts-ignore: because overloaded return type
|
|
49
|
-
return a.name().localeCompare(b.name());
|
|
50
|
-
});
|
|
51
|
-
}
|
|
52
|
-
return visibleCommands;
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
/**
|
|
56
|
-
* Compare options for sort.
|
|
57
|
-
*
|
|
58
|
-
* @param {Option} a
|
|
59
|
-
* @param {Option} b
|
|
60
|
-
* @returns {number}
|
|
61
|
-
*/
|
|
62
|
-
compareOptions(a, b) {
|
|
63
|
-
const getSortKey = (option) => {
|
|
64
|
-
// WYSIWYG for order displayed in help. Short used for comparison if present. No special handling for negated.
|
|
65
|
-
return option.short
|
|
66
|
-
? option.short.replace(/^-/, '')
|
|
67
|
-
: option.long.replace(/^--/, '');
|
|
68
|
-
};
|
|
69
|
-
return getSortKey(a).localeCompare(getSortKey(b));
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Get an array of the visible options. Includes a placeholder for the implicit help option, if there is one.
|
|
74
|
-
*
|
|
75
|
-
* @param {Command} cmd
|
|
76
|
-
* @returns {Option[]}
|
|
77
|
-
*/
|
|
78
|
-
|
|
79
|
-
visibleOptions(cmd) {
|
|
80
|
-
const visibleOptions = cmd.options.filter((option) => !option.hidden);
|
|
81
|
-
// Built-in help option.
|
|
82
|
-
const helpOption = cmd._getHelpOption();
|
|
83
|
-
if (helpOption && !helpOption.hidden) {
|
|
84
|
-
// Automatically hide conflicting flags. Bit dubious but a historical behaviour that is convenient for single-command programs.
|
|
85
|
-
const removeShort = helpOption.short && cmd._findOption(helpOption.short);
|
|
86
|
-
const removeLong = helpOption.long && cmd._findOption(helpOption.long);
|
|
87
|
-
if (!removeShort && !removeLong) {
|
|
88
|
-
visibleOptions.push(helpOption); // no changes needed
|
|
89
|
-
} else if (helpOption.long && !removeLong) {
|
|
90
|
-
visibleOptions.push(
|
|
91
|
-
cmd.createOption(helpOption.long, helpOption.description),
|
|
92
|
-
);
|
|
93
|
-
} else if (helpOption.short && !removeShort) {
|
|
94
|
-
visibleOptions.push(
|
|
95
|
-
cmd.createOption(helpOption.short, helpOption.description),
|
|
96
|
-
);
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
if (this.sortOptions) {
|
|
100
|
-
visibleOptions.sort(this.compareOptions);
|
|
101
|
-
}
|
|
102
|
-
return visibleOptions;
|
|
103
|
-
}
|
|
104
|
-
|
|
105
|
-
/**
|
|
106
|
-
* Get an array of the visible global options. (Not including help.)
|
|
107
|
-
*
|
|
108
|
-
* @param {Command} cmd
|
|
109
|
-
* @returns {Option[]}
|
|
110
|
-
*/
|
|
111
|
-
|
|
112
|
-
visibleGlobalOptions(cmd) {
|
|
113
|
-
if (!this.showGlobalOptions) return [];
|
|
114
|
-
|
|
115
|
-
const globalOptions = [];
|
|
116
|
-
for (
|
|
117
|
-
let ancestorCmd = cmd.parent;
|
|
118
|
-
ancestorCmd;
|
|
119
|
-
ancestorCmd = ancestorCmd.parent
|
|
120
|
-
) {
|
|
121
|
-
const visibleOptions = ancestorCmd.options.filter(
|
|
122
|
-
(option) => !option.hidden,
|
|
123
|
-
);
|
|
124
|
-
globalOptions.push(...visibleOptions);
|
|
125
|
-
}
|
|
126
|
-
if (this.sortOptions) {
|
|
127
|
-
globalOptions.sort(this.compareOptions);
|
|
128
|
-
}
|
|
129
|
-
return globalOptions;
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
/**
|
|
133
|
-
* Get an array of the arguments if any have a description.
|
|
134
|
-
*
|
|
135
|
-
* @param {Command} cmd
|
|
136
|
-
* @returns {Argument[]}
|
|
137
|
-
*/
|
|
138
|
-
|
|
139
|
-
visibleArguments(cmd) {
|
|
140
|
-
// Side effect! Apply the legacy descriptions before the arguments are displayed.
|
|
141
|
-
if (cmd._argsDescription) {
|
|
142
|
-
cmd.registeredArguments.forEach((argument) => {
|
|
143
|
-
argument.description =
|
|
144
|
-
argument.description || cmd._argsDescription[argument.name()] || '';
|
|
145
|
-
});
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
// If there are any arguments with a description then return all the arguments.
|
|
149
|
-
if (cmd.registeredArguments.find((argument) => argument.description)) {
|
|
150
|
-
return cmd.registeredArguments;
|
|
151
|
-
}
|
|
152
|
-
return [];
|
|
153
|
-
}
|
|
154
|
-
|
|
155
|
-
/**
|
|
156
|
-
* Get the command term to show in the list of subcommands.
|
|
157
|
-
*
|
|
158
|
-
* @param {Command} cmd
|
|
159
|
-
* @returns {string}
|
|
160
|
-
*/
|
|
161
|
-
|
|
162
|
-
subcommandTerm(cmd) {
|
|
163
|
-
// Legacy. Ignores custom usage string, and nested commands.
|
|
164
|
-
const args = cmd.registeredArguments
|
|
165
|
-
.map((arg) => humanReadableArgName(arg))
|
|
166
|
-
.join(' ');
|
|
167
|
-
return (
|
|
168
|
-
cmd._name +
|
|
169
|
-
(cmd._aliases[0] ? '|' + cmd._aliases[0] : '') +
|
|
170
|
-
(cmd.options.length ? ' [options]' : '') + // simplistic check for non-help option
|
|
171
|
-
(args ? ' ' + args : '')
|
|
172
|
-
);
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
/**
|
|
176
|
-
* Get the option term to show in the list of options.
|
|
177
|
-
*
|
|
178
|
-
* @param {Option} option
|
|
179
|
-
* @returns {string}
|
|
180
|
-
*/
|
|
181
|
-
|
|
182
|
-
optionTerm(option) {
|
|
183
|
-
return option.flags;
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
/**
|
|
187
|
-
* Get the argument term to show in the list of arguments.
|
|
188
|
-
*
|
|
189
|
-
* @param {Argument} argument
|
|
190
|
-
* @returns {string}
|
|
191
|
-
*/
|
|
192
|
-
|
|
193
|
-
argumentTerm(argument) {
|
|
194
|
-
return argument.name();
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
/**
|
|
198
|
-
* Get the longest command term length.
|
|
199
|
-
*
|
|
200
|
-
* @param {Command} cmd
|
|
201
|
-
* @param {Help} helper
|
|
202
|
-
* @returns {number}
|
|
203
|
-
*/
|
|
204
|
-
|
|
205
|
-
longestSubcommandTermLength(cmd, helper) {
|
|
206
|
-
return helper.visibleCommands(cmd).reduce((max, command) => {
|
|
207
|
-
return Math.max(
|
|
208
|
-
max,
|
|
209
|
-
this.displayWidth(
|
|
210
|
-
helper.styleSubcommandTerm(helper.subcommandTerm(command)),
|
|
211
|
-
),
|
|
212
|
-
);
|
|
213
|
-
}, 0);
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
/**
|
|
217
|
-
* Get the longest option term length.
|
|
218
|
-
*
|
|
219
|
-
* @param {Command} cmd
|
|
220
|
-
* @param {Help} helper
|
|
221
|
-
* @returns {number}
|
|
222
|
-
*/
|
|
223
|
-
|
|
224
|
-
longestOptionTermLength(cmd, helper) {
|
|
225
|
-
return helper.visibleOptions(cmd).reduce((max, option) => {
|
|
226
|
-
return Math.max(
|
|
227
|
-
max,
|
|
228
|
-
this.displayWidth(helper.styleOptionTerm(helper.optionTerm(option))),
|
|
229
|
-
);
|
|
230
|
-
}, 0);
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
/**
|
|
234
|
-
* Get the longest global option term length.
|
|
235
|
-
*
|
|
236
|
-
* @param {Command} cmd
|
|
237
|
-
* @param {Help} helper
|
|
238
|
-
* @returns {number}
|
|
239
|
-
*/
|
|
240
|
-
|
|
241
|
-
longestGlobalOptionTermLength(cmd, helper) {
|
|
242
|
-
return helper.visibleGlobalOptions(cmd).reduce((max, option) => {
|
|
243
|
-
return Math.max(
|
|
244
|
-
max,
|
|
245
|
-
this.displayWidth(helper.styleOptionTerm(helper.optionTerm(option))),
|
|
246
|
-
);
|
|
247
|
-
}, 0);
|
|
248
|
-
}
|
|
249
|
-
|
|
250
|
-
/**
|
|
251
|
-
* Get the longest argument term length.
|
|
252
|
-
*
|
|
253
|
-
* @param {Command} cmd
|
|
254
|
-
* @param {Help} helper
|
|
255
|
-
* @returns {number}
|
|
256
|
-
*/
|
|
257
|
-
|
|
258
|
-
longestArgumentTermLength(cmd, helper) {
|
|
259
|
-
return helper.visibleArguments(cmd).reduce((max, argument) => {
|
|
260
|
-
return Math.max(
|
|
261
|
-
max,
|
|
262
|
-
this.displayWidth(
|
|
263
|
-
helper.styleArgumentTerm(helper.argumentTerm(argument)),
|
|
264
|
-
),
|
|
265
|
-
);
|
|
266
|
-
}, 0);
|
|
267
|
-
}
|
|
268
|
-
|
|
269
|
-
/**
|
|
270
|
-
* Get the command usage to be displayed at the top of the built-in help.
|
|
271
|
-
*
|
|
272
|
-
* @param {Command} cmd
|
|
273
|
-
* @returns {string}
|
|
274
|
-
*/
|
|
275
|
-
|
|
276
|
-
commandUsage(cmd) {
|
|
277
|
-
// Usage
|
|
278
|
-
let cmdName = cmd._name;
|
|
279
|
-
if (cmd._aliases[0]) {
|
|
280
|
-
cmdName = cmdName + '|' + cmd._aliases[0];
|
|
281
|
-
}
|
|
282
|
-
let ancestorCmdNames = '';
|
|
283
|
-
for (
|
|
284
|
-
let ancestorCmd = cmd.parent;
|
|
285
|
-
ancestorCmd;
|
|
286
|
-
ancestorCmd = ancestorCmd.parent
|
|
287
|
-
) {
|
|
288
|
-
ancestorCmdNames = ancestorCmd.name() + ' ' + ancestorCmdNames;
|
|
289
|
-
}
|
|
290
|
-
return ancestorCmdNames + cmdName + ' ' + cmd.usage();
|
|
291
|
-
}
|
|
292
|
-
|
|
293
|
-
/**
|
|
294
|
-
* Get the description for the command.
|
|
295
|
-
*
|
|
296
|
-
* @param {Command} cmd
|
|
297
|
-
* @returns {string}
|
|
298
|
-
*/
|
|
299
|
-
|
|
300
|
-
commandDescription(cmd) {
|
|
301
|
-
// @ts-ignore: because overloaded return type
|
|
302
|
-
return cmd.description();
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
/**
|
|
306
|
-
* Get the subcommand summary to show in the list of subcommands.
|
|
307
|
-
* (Fallback to description for backwards compatibility.)
|
|
308
|
-
*
|
|
309
|
-
* @param {Command} cmd
|
|
310
|
-
* @returns {string}
|
|
311
|
-
*/
|
|
312
|
-
|
|
313
|
-
subcommandDescription(cmd) {
|
|
314
|
-
// @ts-ignore: because overloaded return type
|
|
315
|
-
return cmd.summary() || cmd.description();
|
|
316
|
-
}
|
|
317
|
-
|
|
318
|
-
/**
|
|
319
|
-
* Get the option description to show in the list of options.
|
|
320
|
-
*
|
|
321
|
-
* @param {Option} option
|
|
322
|
-
* @return {string}
|
|
323
|
-
*/
|
|
324
|
-
|
|
325
|
-
optionDescription(option) {
|
|
326
|
-
const extraInfo = [];
|
|
327
|
-
|
|
328
|
-
if (option.argChoices) {
|
|
329
|
-
extraInfo.push(
|
|
330
|
-
// use stringify to match the display of the default value
|
|
331
|
-
`choices: ${option.argChoices.map((choice) => JSON.stringify(choice)).join(', ')}`,
|
|
332
|
-
);
|
|
333
|
-
}
|
|
334
|
-
if (option.defaultValue !== undefined) {
|
|
335
|
-
// default for boolean and negated more for programmer than end user,
|
|
336
|
-
// but show true/false for boolean option as may be for hand-rolled env or config processing.
|
|
337
|
-
const showDefault =
|
|
338
|
-
option.required ||
|
|
339
|
-
option.optional ||
|
|
340
|
-
(option.isBoolean() && typeof option.defaultValue === 'boolean');
|
|
341
|
-
if (showDefault) {
|
|
342
|
-
extraInfo.push(
|
|
343
|
-
`default: ${option.defaultValueDescription || JSON.stringify(option.defaultValue)}`,
|
|
344
|
-
);
|
|
345
|
-
}
|
|
346
|
-
}
|
|
347
|
-
// preset for boolean and negated are more for programmer than end user
|
|
348
|
-
if (option.presetArg !== undefined && option.optional) {
|
|
349
|
-
extraInfo.push(`preset: ${JSON.stringify(option.presetArg)}`);
|
|
350
|
-
}
|
|
351
|
-
if (option.envVar !== undefined) {
|
|
352
|
-
extraInfo.push(`env: ${option.envVar}`);
|
|
353
|
-
}
|
|
354
|
-
if (extraInfo.length > 0) {
|
|
355
|
-
const extraDescription = `(${extraInfo.join(', ')})`;
|
|
356
|
-
if (option.description) {
|
|
357
|
-
return `${option.description} ${extraDescription}`;
|
|
358
|
-
}
|
|
359
|
-
return extraDescription;
|
|
360
|
-
}
|
|
361
|
-
|
|
362
|
-
return option.description;
|
|
363
|
-
}
|
|
364
|
-
|
|
365
|
-
/**
|
|
366
|
-
* Get the argument description to show in the list of arguments.
|
|
367
|
-
*
|
|
368
|
-
* @param {Argument} argument
|
|
369
|
-
* @return {string}
|
|
370
|
-
*/
|
|
371
|
-
|
|
372
|
-
argumentDescription(argument) {
|
|
373
|
-
const extraInfo = [];
|
|
374
|
-
if (argument.argChoices) {
|
|
375
|
-
extraInfo.push(
|
|
376
|
-
// use stringify to match the display of the default value
|
|
377
|
-
`choices: ${argument.argChoices.map((choice) => JSON.stringify(choice)).join(', ')}`,
|
|
378
|
-
);
|
|
379
|
-
}
|
|
380
|
-
if (argument.defaultValue !== undefined) {
|
|
381
|
-
extraInfo.push(
|
|
382
|
-
`default: ${argument.defaultValueDescription || JSON.stringify(argument.defaultValue)}`,
|
|
383
|
-
);
|
|
384
|
-
}
|
|
385
|
-
if (extraInfo.length > 0) {
|
|
386
|
-
const extraDescription = `(${extraInfo.join(', ')})`;
|
|
387
|
-
if (argument.description) {
|
|
388
|
-
return `${argument.description} ${extraDescription}`;
|
|
389
|
-
}
|
|
390
|
-
return extraDescription;
|
|
391
|
-
}
|
|
392
|
-
return argument.description;
|
|
393
|
-
}
|
|
394
|
-
|
|
395
|
-
/**
|
|
396
|
-
* Format a list of items, given a heading and an array of formatted items.
|
|
397
|
-
*
|
|
398
|
-
* @param {string} heading
|
|
399
|
-
* @param {string[]} items
|
|
400
|
-
* @param {Help} helper
|
|
401
|
-
* @returns string[]
|
|
402
|
-
*/
|
|
403
|
-
formatItemList(heading, items, helper) {
|
|
404
|
-
if (items.length === 0) return [];
|
|
405
|
-
|
|
406
|
-
return [helper.styleTitle(heading), ...items, ''];
|
|
407
|
-
}
|
|
408
|
-
|
|
409
|
-
/**
|
|
410
|
-
* Group items by their help group heading.
|
|
411
|
-
*
|
|
412
|
-
* @param {Command[] | Option[]} unsortedItems
|
|
413
|
-
* @param {Command[] | Option[]} visibleItems
|
|
414
|
-
* @param {Function} getGroup
|
|
415
|
-
* @returns {Map<string, Command[] | Option[]>}
|
|
416
|
-
*/
|
|
417
|
-
groupItems(unsortedItems, visibleItems, getGroup) {
|
|
418
|
-
const result = new Map();
|
|
419
|
-
// Add groups in order of appearance in unsortedItems.
|
|
420
|
-
unsortedItems.forEach((item) => {
|
|
421
|
-
const group = getGroup(item);
|
|
422
|
-
if (!result.has(group)) result.set(group, []);
|
|
423
|
-
});
|
|
424
|
-
// Add items in order of appearance in visibleItems.
|
|
425
|
-
visibleItems.forEach((item) => {
|
|
426
|
-
const group = getGroup(item);
|
|
427
|
-
if (!result.has(group)) {
|
|
428
|
-
result.set(group, []);
|
|
429
|
-
}
|
|
430
|
-
result.get(group).push(item);
|
|
431
|
-
});
|
|
432
|
-
return result;
|
|
433
|
-
}
|
|
434
|
-
|
|
435
|
-
/**
|
|
436
|
-
* Generate the built-in help text.
|
|
437
|
-
*
|
|
438
|
-
* @param {Command} cmd
|
|
439
|
-
* @param {Help} helper
|
|
440
|
-
* @returns {string}
|
|
441
|
-
*/
|
|
442
|
-
|
|
443
|
-
formatHelp(cmd, helper) {
|
|
444
|
-
const termWidth = helper.padWidth(cmd, helper);
|
|
445
|
-
const helpWidth = helper.helpWidth ?? 80; // in case prepareContext() was not called
|
|
446
|
-
|
|
447
|
-
function callFormatItem(term, description) {
|
|
448
|
-
return helper.formatItem(term, termWidth, description, helper);
|
|
449
|
-
}
|
|
450
|
-
|
|
451
|
-
// Usage
|
|
452
|
-
let output = [
|
|
453
|
-
`${helper.styleTitle('Usage:')} ${helper.styleUsage(helper.commandUsage(cmd))}`,
|
|
454
|
-
'',
|
|
455
|
-
];
|
|
456
|
-
|
|
457
|
-
// Description
|
|
458
|
-
const commandDescription = helper.commandDescription(cmd);
|
|
459
|
-
if (commandDescription.length > 0) {
|
|
460
|
-
output = output.concat([
|
|
461
|
-
helper.boxWrap(
|
|
462
|
-
helper.styleCommandDescription(commandDescription),
|
|
463
|
-
helpWidth,
|
|
464
|
-
),
|
|
465
|
-
'',
|
|
466
|
-
]);
|
|
467
|
-
}
|
|
468
|
-
|
|
469
|
-
// Arguments
|
|
470
|
-
const argumentList = helper.visibleArguments(cmd).map((argument) => {
|
|
471
|
-
return callFormatItem(
|
|
472
|
-
helper.styleArgumentTerm(helper.argumentTerm(argument)),
|
|
473
|
-
helper.styleArgumentDescription(helper.argumentDescription(argument)),
|
|
474
|
-
);
|
|
475
|
-
});
|
|
476
|
-
output = output.concat(
|
|
477
|
-
this.formatItemList('Arguments:', argumentList, helper),
|
|
478
|
-
);
|
|
479
|
-
|
|
480
|
-
// Options
|
|
481
|
-
const optionGroups = this.groupItems(
|
|
482
|
-
cmd.options,
|
|
483
|
-
helper.visibleOptions(cmd),
|
|
484
|
-
(option) => option.helpGroupHeading ?? 'Options:',
|
|
485
|
-
);
|
|
486
|
-
optionGroups.forEach((options, group) => {
|
|
487
|
-
const optionList = options.map((option) => {
|
|
488
|
-
return callFormatItem(
|
|
489
|
-
helper.styleOptionTerm(helper.optionTerm(option)),
|
|
490
|
-
helper.styleOptionDescription(helper.optionDescription(option)),
|
|
491
|
-
);
|
|
492
|
-
});
|
|
493
|
-
output = output.concat(this.formatItemList(group, optionList, helper));
|
|
494
|
-
});
|
|
495
|
-
|
|
496
|
-
if (helper.showGlobalOptions) {
|
|
497
|
-
const globalOptionList = helper
|
|
498
|
-
.visibleGlobalOptions(cmd)
|
|
499
|
-
.map((option) => {
|
|
500
|
-
return callFormatItem(
|
|
501
|
-
helper.styleOptionTerm(helper.optionTerm(option)),
|
|
502
|
-
helper.styleOptionDescription(helper.optionDescription(option)),
|
|
503
|
-
);
|
|
504
|
-
});
|
|
505
|
-
output = output.concat(
|
|
506
|
-
this.formatItemList('Global Options:', globalOptionList, helper),
|
|
507
|
-
);
|
|
508
|
-
}
|
|
509
|
-
|
|
510
|
-
// Commands
|
|
511
|
-
const commandGroups = this.groupItems(
|
|
512
|
-
cmd.commands,
|
|
513
|
-
helper.visibleCommands(cmd),
|
|
514
|
-
(sub) => sub.helpGroup() || 'Commands:',
|
|
515
|
-
);
|
|
516
|
-
commandGroups.forEach((commands, group) => {
|
|
517
|
-
const commandList = commands.map((sub) => {
|
|
518
|
-
return callFormatItem(
|
|
519
|
-
helper.styleSubcommandTerm(helper.subcommandTerm(sub)),
|
|
520
|
-
helper.styleSubcommandDescription(helper.subcommandDescription(sub)),
|
|
521
|
-
);
|
|
522
|
-
});
|
|
523
|
-
output = output.concat(this.formatItemList(group, commandList, helper));
|
|
524
|
-
});
|
|
525
|
-
|
|
526
|
-
return output.join('\n');
|
|
527
|
-
}
|
|
528
|
-
|
|
529
|
-
/**
|
|
530
|
-
* Return display width of string, ignoring ANSI escape sequences. Used in padding and wrapping calculations.
|
|
531
|
-
*
|
|
532
|
-
* @param {string} str
|
|
533
|
-
* @returns {number}
|
|
534
|
-
*/
|
|
535
|
-
displayWidth(str) {
|
|
536
|
-
return stripColor(str).length;
|
|
537
|
-
}
|
|
538
|
-
|
|
539
|
-
/**
|
|
540
|
-
* Style the title for displaying in the help. Called with 'Usage:', 'Options:', etc.
|
|
541
|
-
*
|
|
542
|
-
* @param {string} str
|
|
543
|
-
* @returns {string}
|
|
544
|
-
*/
|
|
545
|
-
styleTitle(str) {
|
|
546
|
-
return str;
|
|
547
|
-
}
|
|
548
|
-
|
|
549
|
-
styleUsage(str) {
|
|
550
|
-
// Usage has lots of parts the user might like to color separately! Assume default usage string which is formed like:
|
|
551
|
-
// command subcommand [options] [command] <foo> [bar]
|
|
552
|
-
return str
|
|
553
|
-
.split(' ')
|
|
554
|
-
.map((word) => {
|
|
555
|
-
if (word === '[options]') return this.styleOptionText(word);
|
|
556
|
-
if (word === '[command]') return this.styleSubcommandText(word);
|
|
557
|
-
if (word[0] === '[' || word[0] === '<')
|
|
558
|
-
return this.styleArgumentText(word);
|
|
559
|
-
return this.styleCommandText(word); // Restrict to initial words?
|
|
560
|
-
})
|
|
561
|
-
.join(' ');
|
|
562
|
-
}
|
|
563
|
-
styleCommandDescription(str) {
|
|
564
|
-
return this.styleDescriptionText(str);
|
|
565
|
-
}
|
|
566
|
-
styleOptionDescription(str) {
|
|
567
|
-
return this.styleDescriptionText(str);
|
|
568
|
-
}
|
|
569
|
-
styleSubcommandDescription(str) {
|
|
570
|
-
return this.styleDescriptionText(str);
|
|
571
|
-
}
|
|
572
|
-
styleArgumentDescription(str) {
|
|
573
|
-
return this.styleDescriptionText(str);
|
|
574
|
-
}
|
|
575
|
-
styleDescriptionText(str) {
|
|
576
|
-
return str;
|
|
577
|
-
}
|
|
578
|
-
styleOptionTerm(str) {
|
|
579
|
-
return this.styleOptionText(str);
|
|
580
|
-
}
|
|
581
|
-
styleSubcommandTerm(str) {
|
|
582
|
-
// This is very like usage with lots of parts! Assume default string which is formed like:
|
|
583
|
-
// subcommand [options] <foo> [bar]
|
|
584
|
-
return str
|
|
585
|
-
.split(' ')
|
|
586
|
-
.map((word) => {
|
|
587
|
-
if (word === '[options]') return this.styleOptionText(word);
|
|
588
|
-
if (word[0] === '[' || word[0] === '<')
|
|
589
|
-
return this.styleArgumentText(word);
|
|
590
|
-
return this.styleSubcommandText(word); // Restrict to initial words?
|
|
591
|
-
})
|
|
592
|
-
.join(' ');
|
|
593
|
-
}
|
|
594
|
-
styleArgumentTerm(str) {
|
|
595
|
-
return this.styleArgumentText(str);
|
|
596
|
-
}
|
|
597
|
-
styleOptionText(str) {
|
|
598
|
-
return str;
|
|
599
|
-
}
|
|
600
|
-
styleArgumentText(str) {
|
|
601
|
-
return str;
|
|
602
|
-
}
|
|
603
|
-
styleSubcommandText(str) {
|
|
604
|
-
return str;
|
|
605
|
-
}
|
|
606
|
-
styleCommandText(str) {
|
|
607
|
-
return str;
|
|
608
|
-
}
|
|
609
|
-
|
|
610
|
-
/**
|
|
611
|
-
* Calculate the pad width from the maximum term length.
|
|
612
|
-
*
|
|
613
|
-
* @param {Command} cmd
|
|
614
|
-
* @param {Help} helper
|
|
615
|
-
* @returns {number}
|
|
616
|
-
*/
|
|
617
|
-
|
|
618
|
-
padWidth(cmd, helper) {
|
|
619
|
-
return Math.max(
|
|
620
|
-
helper.longestOptionTermLength(cmd, helper),
|
|
621
|
-
helper.longestGlobalOptionTermLength(cmd, helper),
|
|
622
|
-
helper.longestSubcommandTermLength(cmd, helper),
|
|
623
|
-
helper.longestArgumentTermLength(cmd, helper),
|
|
624
|
-
);
|
|
625
|
-
}
|
|
626
|
-
|
|
627
|
-
/**
|
|
628
|
-
* Detect manually wrapped and indented strings by checking for line break followed by whitespace.
|
|
629
|
-
*
|
|
630
|
-
* @param {string} str
|
|
631
|
-
* @returns {boolean}
|
|
632
|
-
*/
|
|
633
|
-
preformatted(str) {
|
|
634
|
-
return /\n[^\S\r\n]/.test(str);
|
|
635
|
-
}
|
|
636
|
-
|
|
637
|
-
/**
|
|
638
|
-
* Format the "item", which consists of a term and description. Pad the term and wrap the description, indenting the following lines.
|
|
639
|
-
*
|
|
640
|
-
* So "TTT", 5, "DDD DDDD DD DDD" might be formatted for this.helpWidth=17 like so:
|
|
641
|
-
* TTT DDD DDDD
|
|
642
|
-
* DD DDD
|
|
643
|
-
*
|
|
644
|
-
* @param {string} term
|
|
645
|
-
* @param {number} termWidth
|
|
646
|
-
* @param {string} description
|
|
647
|
-
* @param {Help} helper
|
|
648
|
-
* @returns {string}
|
|
649
|
-
*/
|
|
650
|
-
formatItem(term, termWidth, description, helper) {
|
|
651
|
-
const itemIndent = 2;
|
|
652
|
-
const itemIndentStr = ' '.repeat(itemIndent);
|
|
653
|
-
if (!description) return itemIndentStr + term;
|
|
654
|
-
|
|
655
|
-
// Pad the term out to a consistent width, so descriptions are aligned.
|
|
656
|
-
const paddedTerm = term.padEnd(
|
|
657
|
-
termWidth + term.length - helper.displayWidth(term),
|
|
658
|
-
);
|
|
659
|
-
|
|
660
|
-
// Format the description.
|
|
661
|
-
const spacerWidth = 2; // between term and description
|
|
662
|
-
const helpWidth = this.helpWidth ?? 80; // in case prepareContext() was not called
|
|
663
|
-
const remainingWidth = helpWidth - termWidth - spacerWidth - itemIndent;
|
|
664
|
-
let formattedDescription;
|
|
665
|
-
if (
|
|
666
|
-
remainingWidth < this.minWidthToWrap ||
|
|
667
|
-
helper.preformatted(description)
|
|
668
|
-
) {
|
|
669
|
-
formattedDescription = description;
|
|
670
|
-
} else {
|
|
671
|
-
const wrappedDescription = helper.boxWrap(description, remainingWidth);
|
|
672
|
-
formattedDescription = wrappedDescription.replace(
|
|
673
|
-
/\n/g,
|
|
674
|
-
'\n' + ' '.repeat(termWidth + spacerWidth),
|
|
675
|
-
);
|
|
676
|
-
}
|
|
677
|
-
|
|
678
|
-
// Construct and overall indent.
|
|
679
|
-
return (
|
|
680
|
-
itemIndentStr +
|
|
681
|
-
paddedTerm +
|
|
682
|
-
' '.repeat(spacerWidth) +
|
|
683
|
-
formattedDescription.replace(/\n/g, `\n${itemIndentStr}`)
|
|
684
|
-
);
|
|
685
|
-
}
|
|
686
|
-
|
|
687
|
-
/**
|
|
688
|
-
* Wrap a string at whitespace, preserving existing line breaks.
|
|
689
|
-
* Wrapping is skipped if the width is less than `minWidthToWrap`.
|
|
690
|
-
*
|
|
691
|
-
* @param {string} str
|
|
692
|
-
* @param {number} width
|
|
693
|
-
* @returns {string}
|
|
694
|
-
*/
|
|
695
|
-
boxWrap(str, width) {
|
|
696
|
-
if (width < this.minWidthToWrap) return str;
|
|
697
|
-
|
|
698
|
-
const rawLines = str.split(/\r\n|\n/);
|
|
699
|
-
// split up text by whitespace
|
|
700
|
-
const chunkPattern = /[\s]*[^\s]+/g;
|
|
701
|
-
const wrappedLines = [];
|
|
702
|
-
rawLines.forEach((line) => {
|
|
703
|
-
const chunks = line.match(chunkPattern);
|
|
704
|
-
if (chunks === null) {
|
|
705
|
-
wrappedLines.push('');
|
|
706
|
-
return;
|
|
707
|
-
}
|
|
708
|
-
|
|
709
|
-
let sumChunks = [chunks.shift()];
|
|
710
|
-
let sumWidth = this.displayWidth(sumChunks[0]);
|
|
711
|
-
chunks.forEach((chunk) => {
|
|
712
|
-
const visibleWidth = this.displayWidth(chunk);
|
|
713
|
-
// Accumulate chunks while they fit into width.
|
|
714
|
-
if (sumWidth + visibleWidth <= width) {
|
|
715
|
-
sumChunks.push(chunk);
|
|
716
|
-
sumWidth += visibleWidth;
|
|
717
|
-
return;
|
|
718
|
-
}
|
|
719
|
-
wrappedLines.push(sumChunks.join(''));
|
|
720
|
-
|
|
721
|
-
const nextChunk = chunk.trimStart(); // trim space at line break
|
|
722
|
-
sumChunks = [nextChunk];
|
|
723
|
-
sumWidth = this.displayWidth(nextChunk);
|
|
724
|
-
});
|
|
725
|
-
wrappedLines.push(sumChunks.join(''));
|
|
726
|
-
});
|
|
727
|
-
|
|
728
|
-
return wrappedLines.join('\n');
|
|
729
|
-
}
|
|
730
|
-
}
|
|
731
|
-
|
|
732
|
-
/**
|
|
733
|
-
* Strip style ANSI escape sequences from the string. In particular, SGR (Select Graphic Rendition) codes.
|
|
734
|
-
*
|
|
735
|
-
* @param {string} str
|
|
736
|
-
* @returns {string}
|
|
737
|
-
* @package
|
|
738
|
-
*/
|
|
739
|
-
|
|
740
|
-
function stripColor(str) {
|
|
741
|
-
// eslint-disable-next-line no-control-regex
|
|
742
|
-
const sgrPattern = /\x1b\[\d*(;\d*)*m/g;
|
|
743
|
-
return str.replace(sgrPattern, '');
|
|
744
|
-
}
|
|
745
|
-
|
|
746
|
-
exports.Help = Help;
|
|
747
|
-
exports.stripColor = stripColor;
|