2017-08-04 19:51:01 +02:00
|
|
|
import yargParser from 'yargs-parser';
|
2017-08-04 18:50:12 +02:00
|
|
|
import { _ } from 'lib/locale.js';
|
|
|
|
import { time } from 'lib/time-utils.js';
|
2017-08-03 19:48:14 +02:00
|
|
|
const stringPadding = require('string-padding');
|
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
const cliUtils = {};
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-10-26 23:57:49 +02:00
|
|
|
cliUtils.splitCommandString = function(command) {
|
|
|
|
let args = [];
|
|
|
|
let state = "start"
|
|
|
|
let current = ""
|
|
|
|
let quote = "\""
|
|
|
|
let escapeNext = false;
|
|
|
|
for (let i = 0; i < command.length; i++) {
|
|
|
|
let c = command[i]
|
|
|
|
|
|
|
|
if (state == "quotes") {
|
|
|
|
if (c != quote) {
|
|
|
|
current += c
|
|
|
|
} else {
|
|
|
|
args.push(current)
|
|
|
|
current = ""
|
|
|
|
state = "start"
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if (escapeNext) {
|
|
|
|
current += c;
|
|
|
|
escapeNext = false;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == "\\") {
|
|
|
|
escapeNext = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c == '"' || c == '\'') {
|
|
|
|
state = "quotes"
|
|
|
|
quote = c
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state == "arg") {
|
|
|
|
if (c == ' ' || c == '\t') {
|
|
|
|
args.push(current)
|
|
|
|
current = ""
|
|
|
|
state = "start"
|
|
|
|
} else {
|
|
|
|
current += c
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if (c != ' ' && c != "\t") {
|
|
|
|
state = "arg"
|
|
|
|
current += c
|
|
|
|
}
|
2017-10-06 19:38:17 +02:00
|
|
|
}
|
2017-10-26 23:57:49 +02:00
|
|
|
|
|
|
|
if (state == "quotes") {
|
|
|
|
throw new Error("Unclosed quote in command line: " + command)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (current != "") {
|
|
|
|
args.push(current)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args.length <= 0) {
|
|
|
|
throw new Error("Empty command line")
|
|
|
|
}
|
|
|
|
|
|
|
|
return args;
|
2017-10-06 19:38:17 +02:00
|
|
|
}
|
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
cliUtils.printArray = function(logFunction, rows, headers = null) {
|
|
|
|
if (!rows.length) return '';
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
const ALIGN_LEFT = 0;
|
|
|
|
const ALIGN_RIGHT = 1;
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
let colWidths = [];
|
|
|
|
let colAligns = [];
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
for (let i = 0; i < rows.length; i++) {
|
|
|
|
let row = rows[i];
|
|
|
|
|
|
|
|
for (let j = 0; j < row.length; j++) {
|
|
|
|
let item = row[j];
|
|
|
|
let width = item ? item.toString().length : 0;
|
|
|
|
let align = typeof item == 'number' ? ALIGN_RIGHT : ALIGN_LEFT;
|
|
|
|
if (!colWidths[j] || colWidths[j] < width) colWidths[j] = width;
|
|
|
|
if (colAligns.length <= j) colAligns[j] = align;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let lines = [];
|
|
|
|
for (let row = 0; row < rows.length; row++) {
|
|
|
|
let line = [];
|
|
|
|
for (let col = 0; col < colWidths.length; col++) {
|
|
|
|
let item = rows[row][col];
|
|
|
|
let width = colWidths[col];
|
|
|
|
let dir = colAligns[col] == ALIGN_LEFT ? stringPadding.RIGHT : stringPadding.LEFT;
|
|
|
|
line.push(stringPadding(item, width, ' ', dir));
|
2017-08-03 19:48:14 +02:00
|
|
|
}
|
2017-08-04 19:51:01 +02:00
|
|
|
logFunction(line.join(' '));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cliUtils.parseFlags = function(flags) {
|
|
|
|
let output = {};
|
|
|
|
flags = flags.split(',');
|
|
|
|
for (let i = 0; i < flags.length; i++) {
|
|
|
|
let f = flags[i].trim();
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
if (f.substr(0, 2) == '--') {
|
|
|
|
f = f.split(' ');
|
|
|
|
output.long = f[0].substr(2).trim();
|
|
|
|
if (f.length == 2) {
|
|
|
|
output.arg = cliUtils.parseCommandArg(f[1].trim());
|
2017-08-03 19:48:14 +02:00
|
|
|
}
|
2017-08-04 19:51:01 +02:00
|
|
|
} else if (f.substr(0, 1) == '-') {
|
|
|
|
output.short = f.substr(1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
cliUtils.parseCommandArg = function(arg) {
|
|
|
|
if (arg.length <= 2) throw new Error('Invalid command arg: ' + arg);
|
|
|
|
|
|
|
|
const c1 = arg[0];
|
|
|
|
const c2 = arg[arg.length - 1];
|
|
|
|
const name = arg.substr(1, arg.length - 2);
|
|
|
|
|
|
|
|
if (c1 == '<' && c2 == '>') {
|
|
|
|
return { required: true, name: name };
|
|
|
|
} else if (c1 == '[' && c2 == ']') {
|
|
|
|
return { required: false, name: name };
|
|
|
|
} else {
|
|
|
|
throw new Error('Invalid command arg: ' + arg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
cliUtils.makeCommandArgs = function(cmd, argv) {
|
|
|
|
let cmdUsage = cmd.usage();
|
|
|
|
cmdUsage = yargParser(cmdUsage);
|
|
|
|
let output = {};
|
|
|
|
|
|
|
|
let options = cmd.options();
|
|
|
|
let booleanFlags = [];
|
|
|
|
let aliases = {};
|
|
|
|
for (let i = 0; i < options.length; i++) {
|
|
|
|
if (options[i].length != 2) throw new Error('Invalid options: ' + options[i]);
|
|
|
|
let flags = options[i][0];
|
|
|
|
let text = options[i][1];
|
|
|
|
|
|
|
|
flags = cliUtils.parseFlags(flags);
|
|
|
|
|
|
|
|
if (!flags.arg) {
|
|
|
|
booleanFlags.push(flags.short);
|
|
|
|
if (flags.long) booleanFlags.push(flags.long);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags.short && flags.long) {
|
|
|
|
aliases[flags.long] = [flags.short];
|
2017-08-03 19:48:14 +02:00
|
|
|
}
|
2017-08-04 19:51:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
let args = yargParser(argv, {
|
|
|
|
boolean: booleanFlags,
|
|
|
|
alias: aliases,
|
2017-08-21 19:56:40 +02:00
|
|
|
string: ['_'],
|
2017-08-04 19:51:01 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
for (let i = 1; i < cmdUsage['_'].length; i++) {
|
|
|
|
const a = cliUtils.parseCommandArg(cmdUsage['_'][i]);
|
2017-08-04 18:50:12 +02:00
|
|
|
if (a.required && !args['_'][i]) throw new Error(_('Missing required argument: %s', a.name));
|
2017-08-04 19:51:01 +02:00
|
|
|
if (i >= a.length) {
|
|
|
|
output[a.name] = null;
|
|
|
|
} else {
|
|
|
|
output[a.name] = args['_'][i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let argOptions = {};
|
|
|
|
for (let key in args) {
|
|
|
|
if (!args.hasOwnProperty(key)) continue;
|
|
|
|
if (key == '_') continue;
|
|
|
|
argOptions[key] = args[key];
|
|
|
|
}
|
|
|
|
|
|
|
|
output.options = argOptions;
|
2017-08-03 19:48:14 +02:00
|
|
|
|
2017-08-04 19:51:01 +02:00
|
|
|
return output;
|
2017-08-03 19:48:14 +02:00
|
|
|
}
|
|
|
|
|
2017-09-10 19:32:04 +02:00
|
|
|
cliUtils.promptMcq = function(message, answers) {
|
|
|
|
const readline = require('readline');
|
|
|
|
|
|
|
|
const rl = readline.createInterface({
|
|
|
|
input: process.stdin,
|
|
|
|
output: process.stdout
|
|
|
|
});
|
|
|
|
|
|
|
|
message += "\n\n";
|
|
|
|
for (let n in answers) {
|
|
|
|
if (!answers.hasOwnProperty(n)) continue;
|
|
|
|
message += _('%s: %s', n, answers[n]) + "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
message += "\n";
|
|
|
|
message += _('Your choice: ');
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
rl.question(message, (answer) => {
|
|
|
|
rl.close();
|
|
|
|
|
|
|
|
if (!(answer in answers)) {
|
|
|
|
reject(new Error(_('Invalid answer: %s', answer)));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
resolve(answer);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-04 18:50:12 +02:00
|
|
|
cliUtils.promptConfirm = function(message, answers = null) {
|
|
|
|
if (!answers) answers = [_('Y'), _('n')];
|
|
|
|
const readline = require('readline');
|
|
|
|
|
|
|
|
const rl = readline.createInterface({
|
|
|
|
input: process.stdin,
|
|
|
|
output: process.stdout
|
|
|
|
});
|
|
|
|
|
|
|
|
message += ' (' + answers.join('/') + ')';
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
rl.question(message + ' ', (answer) => {
|
|
|
|
const ok = !answer || answer.toLowerCase() == answers[0].toLowerCase();
|
|
|
|
rl.close();
|
|
|
|
resolve(ok);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-22 19:57:35 +02:00
|
|
|
cliUtils.promptInput = function(message) {
|
|
|
|
const readline = require('readline');
|
|
|
|
|
|
|
|
const rl = readline.createInterface({
|
|
|
|
input: process.stdin,
|
|
|
|
output: process.stdout
|
|
|
|
});
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
rl.question(message + ' ', (answer) => {
|
|
|
|
rl.close();
|
|
|
|
resolve(answer);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-10-20 00:02:13 +02:00
|
|
|
// Note: initialText is there to have the same signature as statusBar.prompt() so that
|
|
|
|
// it can be a drop-in replacement, however initialText is not used (and cannot be
|
|
|
|
// with readline.question?).
|
|
|
|
cliUtils.prompt = function(initialText = '', promptString = ':') {
|
|
|
|
const readline = require('readline');
|
|
|
|
|
|
|
|
const rl = readline.createInterface({
|
|
|
|
input: process.stdin,
|
|
|
|
output: process.stdout
|
|
|
|
});
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
rl.question(promptString, (answer) => {
|
|
|
|
rl.close();
|
|
|
|
resolve(answer);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-08-04 18:50:12 +02:00
|
|
|
let redrawStarted_ = false;
|
|
|
|
let redrawLastLog_ = null;
|
|
|
|
let redrawLastUpdateTime_ = 0;
|
|
|
|
|
2017-10-14 20:03:23 +02:00
|
|
|
cliUtils.setStdout = function(v) {
|
|
|
|
this.stdout_ = v;
|
|
|
|
}
|
|
|
|
|
2017-08-04 18:50:12 +02:00
|
|
|
cliUtils.redraw = function(s) {
|
|
|
|
const now = time.unixMs();
|
|
|
|
|
|
|
|
if (now - redrawLastUpdateTime_ > 4000) {
|
2017-10-25 19:41:36 +02:00
|
|
|
this.stdout_(s);
|
2017-08-04 18:50:12 +02:00
|
|
|
redrawLastUpdateTime_ = now;
|
|
|
|
redrawLastLog_ = null;
|
|
|
|
} else {
|
|
|
|
redrawLastLog_ = s;
|
|
|
|
}
|
|
|
|
|
|
|
|
redrawStarted_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
cliUtils.redrawDone = function() {
|
|
|
|
if (!redrawStarted_) return;
|
|
|
|
|
|
|
|
if (redrawLastLog_) {
|
2017-10-14 20:03:23 +02:00
|
|
|
this.stdout_(redrawLastLog_);
|
2017-08-04 18:50:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
redrawLastLog_ = null;
|
|
|
|
redrawStarted_ = false;
|
|
|
|
}
|
|
|
|
|
2017-08-03 19:48:14 +02:00
|
|
|
export { cliUtils };
|