forked from readmeio/rdme
-
Notifications
You must be signed in to change notification settings - Fork 0
/
cli.js
120 lines (99 loc) · 3.69 KB
/
cli.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
/* eslint-disable no-underscore-dangle */
const cliArgs = require('command-line-args');
const path = require('path');
// We have to do this otherwise `require('config')` loads
// from the cwd where the user is running `rdme` which
// wont be what we want
//
// This is a little sketchy overwriting environment variables
// but since this is only supposed to be a cli and not
// requireable, i think this is okay
const configDir = process.env.NODE_CONFIG_DIR;
process.env.NODE_CONFIG_DIR = path.join(__dirname, '/config');
const config = require('config');
process.env.NODE_CONFIG_DIR = configDir;
const { version } = require('./package.json');
const configStore = require('./lib/configstore');
const help = require('./lib/help');
const commands = require('./lib/commands');
/**
* @param {Array} processArgv - An array of arguments from the current process. Can be used to mock
* fake CLI calls.
* @return {Promise}
*/
module.exports = processArgv => {
const mainArgs = [
{ name: 'help', alias: 'h', type: Boolean, description: 'Display this usage guide' },
{
name: 'version',
alias: 'v',
type: Boolean,
description: `Show the current ${config.cli} version`,
},
{ name: 'command', type: String, defaultOption: true },
];
const argv = cliArgs(mainArgs, { partial: true, argv: processArgv });
const cmd = argv.command || false;
// Add support for `-V` as an additional `--version` alias.
if (typeof argv._unknown !== 'undefined') {
if (argv._unknown.indexOf('-V') !== -1) {
argv.version = true;
}
}
if (argv.version && (!cmd || cmd === 'help')) return Promise.resolve(version);
let command = cmd || '';
if (!command) {
command = 'help';
}
if (command === 'help') {
argv.help = true;
}
try {
let cmdArgv;
let bin;
// Handling for `rdme help` and `rdme help <command>` cases.
if (command === 'help') {
if ((argv._unknown || []).length === 0) {
return Promise.resolve(help.globalUsage(mainArgs));
}
if (argv._unknown.indexOf('-H') !== -1) {
return Promise.resolve(help.globalUsage(mainArgs));
}
cmdArgv = cliArgs([{ name: 'subcommand', type: String, defaultOption: true }], {
argv: argv._unknown,
});
if (!cmdArgv.subcommand) {
return Promise.resolve(help.globalUsage(mainArgs));
}
bin = commands.load(cmdArgv.subcommand);
return Promise.resolve(help.commandUsage(bin));
}
bin = commands.load(command);
// Handling for `rdme <command> --help`.
if (argv.help) {
return Promise.resolve(help.commandUsage(bin));
}
try {
cmdArgv = cliArgs(bin.args, { argv: argv._unknown || [] });
} catch (e) {
// If we have a command that has its own `--version` argument to accept data, that argument,
// if supplied in the `--version VERSION_STRING` format instead of `--version=VERSION_STRING`,
// will collide with the global version argument because their types differ and the argument
// parser gets confused.
//
// Instead of failing out to the user with an undecipherable "Unknown value: ..." error, let's
// try to parse their request again but a tad less eager.
if ((e.name !== 'UNKNOWN_VALUE' || (e.name === 'UNKNOWN_VALUE' && !argv.version)) && argv.command !== 'oas') {
throw e;
}
cmdArgv = cliArgs(bin.args, { partial: true, argv: processArgv.slice(1) });
}
cmdArgv = { key: configStore.get('apiKey'), ...cmdArgv };
return bin.run(cmdArgv);
} catch (e) {
if (e.message === 'Command not found.') {
e.description = `Type \`${`${config.cli} help`.yellow}\` ${`to see all commands`.red}`;
}
return Promise.reject(e);
}
};