-
Notifications
You must be signed in to change notification settings - Fork 10
/
index.js
391 lines (296 loc) · 12.5 KB
/
index.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
'use strict';
/* eslint max-len: 0 */
const test = require('tape');
const { parseArgs } = require('../index.js');
// Test results are as we expect
test('when short option used as flag then stored as flag', function(t) {
const passedArgs = ['-f'];
const expected = { flags: { f: true }, values: { f: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected);
t.end();
});
test('when short option used as flag before positional then stored as flag and positional (and not value)', function(t) {
const passedArgs = ['-f', 'bar'];
const expected = { flags: { f: true }, values: { f: undefined }, positionals: [ 'bar' ] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected);
t.end();
});
test('when short option `type: "string"` used with value then stored as value', function(t) {
const passedArgs = ['-f', 'bar'];
const passedOptions = { f: { type: 'string' } };
const expected = { flags: { f: true }, values: { f: 'bar' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('when short option listed in short used as flag then long option stored as flag', function(t) {
const passedArgs = ['-f'];
const passedOptions = { foo: { short: 'f' } };
const expected = { flags: { foo: true }, values: { foo: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('when short option listed in short and long listed in `type: "string"` and used with value then long option stored as value', function(t) {
const passedArgs = ['-f', 'bar'];
const passedOptions = { foo: { short: 'f', type: 'string' } };
const expected = { flags: { foo: true }, values: { foo: 'bar' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('when short option `type: "string"` used without value then stored as flag', function(t) {
const passedArgs = ['-f'];
const passedOptions = { f: { type: 'string' } };
const expected = { flags: { f: true }, values: { f: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('short option group behaves like multiple short options', function(t) {
const passedArgs = ['-rf'];
const passedOptions = { };
const expected = { flags: { r: true, f: true }, values: { r: undefined, f: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('short option group does not consume subsequent positional', function(t) {
const passedArgs = ['-rf', 'foo'];
const passedOptions = { };
const expected = { flags: { r: true, f: true }, values: { r: undefined, f: undefined }, positionals: ['foo'] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
// // See: Guideline 5 https://pubs.opengroup.org/onlinepubs/9699919799/basedefs/V1_chap12.html
test('if terminal of short-option group configured `type: "string"`, subsequent positional is stored', function(t) {
const passedArgs = ['-rvf', 'foo'];
const passedOptions = { f: { type: 'string' } };
const expected = { flags: { r: true, f: true, v: true }, values: { r: undefined, v: undefined, f: 'foo' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('handles short-option groups in conjunction with long-options', function(t) {
const passedArgs = ['-rf', '--foo', 'foo'];
const passedOptions = { foo: { type: 'string' } };
const expected = { flags: { r: true, f: true, foo: true }, values: { r: undefined, f: undefined, foo: 'foo' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('handles short-option groups with "short" alias configured', function(t) {
const passedArgs = ['-rf'];
const passedOptions = { remove: { short: 'r' } };
const expected = { flags: { remove: true, f: true }, values: { remove: undefined, f: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected);
t.end();
});
test('Everything after a bare `--` is considered a positional argument', function(t) {
const passedArgs = ['--', 'barepositionals', 'mopositionals'];
const expected = { flags: {}, values: {}, positionals: ['barepositionals', 'mopositionals'] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected, 'testing bare positionals');
t.end();
});
test('args are true', function(t) {
const passedArgs = ['--foo', '--bar'];
const expected = { flags: { foo: true, bar: true }, values: { foo: undefined, bar: undefined }, positionals: [] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected, 'args are true');
t.end();
});
test('arg is true and positional is identified', function(t) {
const passedArgs = ['--foo=a', '--foo', 'b'];
const expected = { flags: { foo: true }, values: { foo: undefined }, positionals: ['b'] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected, 'arg is true and positional is identified');
t.end();
});
test('args equals are passed `type: "string"`', function(t) {
const passedArgs = ['--so=wat'];
const passedOptions = { so: { type: 'string' } };
const expected = { flags: { so: true }, values: { so: 'wat' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'arg value is passed');
t.end();
});
test('when args include single dash then result stores dash as positional', function(t) {
const passedArgs = ['-'];
const expected = { flags: { }, values: { }, positionals: ['-'] };
const args = parseArgs({ args: passedArgs });
t.deepEqual(args, expected);
t.end();
});
test('zero config args equals are parsed as if `type: "string"`', function(t) {
const passedArgs = ['--so=wat'];
const passedOptions = { };
const expected = { flags: { so: true }, values: { so: 'wat' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'arg value is passed');
t.end();
});
test('same arg is passed twice `type: "string"` and last value is recorded', function(t) {
const passedArgs = ['--foo=a', '--foo', 'b'];
const passedOptions = { foo: { type: 'string' } };
const expected = { flags: { foo: true }, values: { foo: 'b' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'last arg value is passed');
t.end();
});
test('args equals pass string including more equals', function(t) {
const passedArgs = ['--so=wat=bing'];
const passedOptions = { so: { type: 'string' } };
const expected = { flags: { so: true }, values: { so: 'wat=bing' }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'arg value is passed');
t.end();
});
test('first arg passed for `type: "string"` and "multiple" is in array', function(t) {
const passedArgs = ['--foo=a'];
const passedOptions = { foo: { type: 'string', multiple: true } };
const expected = { flags: { foo: true }, values: { foo: ['a'] }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'first multiple in array');
t.end();
});
test('args are passed `type: "string"` and "multiple"', function(t) {
const passedArgs = ['--foo=a', '--foo', 'b'];
const passedOptions = {
foo: {
type: 'string',
multiple: true,
},
};
const expected = { flags: { foo: true }, values: { foo: ['a', 'b'] }, positionals: [] };
const args = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(args, expected, 'both arg values are passed');
t.end();
});
test('order of option and positional does not matter (per README)', function(t) {
const passedArgs1 = ['--foo=bar', 'baz'];
const passedArgs2 = ['baz', '--foo=bar'];
const passedOptions = { foo: { type: 'string' } };
const expected = { flags: { foo: true }, values: { foo: 'bar' }, positionals: ['baz'] };
t.deepEqual(parseArgs({ args: passedArgs1, options: passedOptions }), expected, 'option then positional');
t.deepEqual(parseArgs({ args: passedArgs2, options: passedOptions }), expected, 'positional then option');
t.end();
});
test('correct default args when use node -p', function(t) {
const holdArgv = process.argv;
process.argv = [process.argv0, '--foo'];
const holdExecArgv = process.execArgv;
process.execArgv = ['-p', '0'];
const result = parseArgs();
const expected = { flags: { foo: true },
values: { foo: undefined },
positionals: [] };
t.deepEqual(result, expected);
t.end();
process.argv = holdArgv;
process.execArgv = holdExecArgv;
});
test('correct default args when use node --print', function(t) {
const holdArgv = process.argv;
process.argv = [process.argv0, '--foo'];
const holdExecArgv = process.execArgv;
process.execArgv = ['--print', '0'];
const result = parseArgs();
const expected = { flags: { foo: true },
values: { foo: undefined },
positionals: [] };
t.deepEqual(result, expected);
t.end();
process.argv = holdArgv;
process.execArgv = holdExecArgv;
});
test('correct default args when use node -e', function(t) {
const holdArgv = process.argv;
process.argv = [process.argv0, '--foo'];
const holdExecArgv = process.execArgv;
process.execArgv = ['-e', '0'];
const result = parseArgs();
const expected = { flags: { foo: true },
values: { foo: undefined },
positionals: [] };
t.deepEqual(result, expected);
t.end();
process.argv = holdArgv;
process.execArgv = holdExecArgv;
});
test('correct default args when use node --eval', function(t) {
const holdArgv = process.argv;
process.argv = [process.argv0, '--foo'];
const holdExecArgv = process.execArgv;
process.execArgv = ['--eval', '0'];
const result = parseArgs();
const expected = { flags: { foo: true },
values: { foo: undefined },
positionals: [] };
t.deepEqual(result, expected);
t.end();
process.argv = holdArgv;
process.execArgv = holdExecArgv;
});
test('correct default args when normal arguments', function(t) {
const holdArgv = process.argv;
process.argv = [process.argv0, 'script.js', '--foo'];
const holdExecArgv = process.execArgv;
process.execArgv = [];
const result = parseArgs();
const expected = { flags: { foo: true },
values: { foo: undefined },
positionals: [] };
t.deepEqual(result, expected);
t.end();
process.argv = holdArgv;
process.execArgv = holdExecArgv;
});
test('excess leading dashes on options are retained', function(t) {
// Enforce a design decision for an edge case.
const passedArgs = ['---triple'];
const passedOptions = { };
const expected = {
flags: { '-triple': true },
values: { '-triple': undefined },
positionals: []
};
const result = parseArgs({ args: passedArgs, options: passedOptions });
t.deepEqual(result, expected, 'excess option dashes are retained');
t.end();
});
// Test bad inputs
test('invalid argument passed for options', function(t) {
const passedArgs = ['--so=wat'];
const passedOptions = 'bad value';
t.throws(function() { parseArgs({ args: passedArgs, options: passedOptions }); }, {
code: 'ERR_INVALID_ARG_TYPE'
});
t.end();
});
test('boolean passed to "type" option', function(t) {
const passedArgs = ['--so=wat'];
const passedOptions = { foo: { type: true } };
t.throws(function() { parseArgs({ args: passedArgs, options: passedOptions }); }, {
code: 'ERR_INVALID_ARG_TYPE'
});
t.end();
});
test('invalid union value passed to "type" option', function(t) {
const passedArgs = ['--so=wat'];
const passedOptions = { foo: { type: 'str' } };
t.throws(function() { parseArgs({ args: passedArgs, options: passedOptions }); }, {
code: 'ERR_INVALID_ARG_TYPE'
});
t.end();
});
test('invalid short option length', function(t) {
const passedArgs = [];
const passedOptions = { foo: { short: 'fo' } };
t.throws(function() { parseArgs({ args: passedArgs, options: passedOptions }); });
t.end();
});