2014-02-07 23:21:24 +02:00
|
|
|
/*
|
2015-09-04 21:12:20 +03:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2016-01-28 23:30:11 +02:00
|
|
|
* Copyright 2016 Joyent, Inc.
|
2014-02-07 23:21:24 +02:00
|
|
|
*
|
2015-08-25 22:14:16 +03:00
|
|
|
* The `triton` CLI class.
|
2014-02-07 23:21:24 +02:00
|
|
|
*/
|
|
|
|
|
2015-08-25 22:14:16 +03:00
|
|
|
var assert = require('assert-plus');
|
|
|
|
var bunyan = require('bunyan');
|
2014-02-07 23:21:24 +02:00
|
|
|
var child_process = require('child_process'),
|
|
|
|
spawn = child_process.spawn,
|
|
|
|
exec = child_process.exec;
|
|
|
|
var cmdln = require('cmdln'),
|
|
|
|
Cmdln = cmdln.Cmdln;
|
2015-09-04 01:12:08 +03:00
|
|
|
var mkdirp = require('mkdirp');
|
2015-08-25 22:14:16 +03:00
|
|
|
var util = require('util'),
|
|
|
|
format = util.format;
|
2015-08-26 19:59:12 +03:00
|
|
|
var path = require('path');
|
2015-08-25 22:14:16 +03:00
|
|
|
var vasync = require('vasync');
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
var common = require('./common');
|
2015-09-08 19:55:48 +03:00
|
|
|
var mod_config = require('./config');
|
2014-02-08 10:15:26 +02:00
|
|
|
var errors = require('./errors');
|
2015-09-30 01:13:34 +03:00
|
|
|
var tritonapi = require('./tritonapi');
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//---- globals
|
|
|
|
|
|
|
|
var pkg = require('../package.json');
|
2015-09-08 19:55:48 +03:00
|
|
|
|
2016-01-28 23:30:11 +02:00
|
|
|
var CONFIG_DIR;
|
|
|
|
if (process.platform === 'win32') {
|
|
|
|
/*
|
|
|
|
* For better or worse we are using APPDATA (i.e. the *Roaming* AppData
|
|
|
|
* dir) over LOCALAPPDATA (non-roaming). The former is meant for "user"
|
|
|
|
* data, the latter for "machine" data.
|
|
|
|
*
|
|
|
|
* TODO: We should likely separate out the *cache* subdir to
|
|
|
|
* machine-specific data dir.
|
|
|
|
*/
|
|
|
|
CONFIG_DIR = path.resolve(process.env.APPDATA, 'Joyent', 'Triton');
|
|
|
|
} else {
|
|
|
|
CONFIG_DIR = path.resolve(process.env.HOME, '.triton');
|
|
|
|
}
|
|
|
|
|
2014-02-07 23:21:24 +02:00
|
|
|
|
2015-08-27 03:21:27 +03:00
|
|
|
var OPTIONS = [
|
|
|
|
{
|
|
|
|
names: ['help', 'h'],
|
|
|
|
type: 'bool',
|
|
|
|
help: 'Print this help and exit.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'version',
|
|
|
|
type: 'bool',
|
|
|
|
help: 'Print version and exit.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
names: ['verbose', 'v'],
|
|
|
|
type: 'bool',
|
|
|
|
help: 'Verbose/debug output.'
|
|
|
|
},
|
|
|
|
|
2015-09-08 19:55:48 +03:00
|
|
|
{
|
|
|
|
names: ['profile', 'p'],
|
|
|
|
type: 'string',
|
2016-01-08 21:07:43 +02:00
|
|
|
completionType: 'tritonprofile',
|
2015-09-08 19:55:48 +03:00
|
|
|
env: 'TRITON_PROFILE',
|
|
|
|
helpArg: 'NAME',
|
2015-09-25 22:19:29 +03:00
|
|
|
help: 'Triton client profile to use.'
|
2015-09-08 19:55:48 +03:00
|
|
|
},
|
2015-08-27 03:21:27 +03:00
|
|
|
|
|
|
|
{
|
2015-09-21 22:34:37 +03:00
|
|
|
group: 'CloudAPI Options'
|
2015-08-27 03:21:27 +03:00
|
|
|
},
|
2015-09-21 22:34:37 +03:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Environment variable integration.
|
|
|
|
*
|
|
|
|
* While dashdash supports integrated envvar parsing with options
|
|
|
|
* we don't use that with `triton` because (a) we want to apply *option*
|
|
|
|
* usage (but not envvars) to profiles other than the default 'env'
|
|
|
|
* profile, and (b) we want to support `TRITON_*` *and* `SDC_*` envvars,
|
|
|
|
* which dashdash doesn't support.
|
|
|
|
*
|
|
|
|
* See <https://github.com/joyent/node-triton/issues/28> for some details.
|
|
|
|
*/
|
2015-08-27 03:21:27 +03:00
|
|
|
{
|
|
|
|
names: ['account', 'a'],
|
|
|
|
type: 'string',
|
2015-09-21 22:34:37 +03:00
|
|
|
help: 'Account (login name). Environment: TRITON_ACCOUNT=ACCOUNT ' +
|
|
|
|
'or SDC_ACCOUNT=ACCOUNT.',
|
2015-08-27 03:21:27 +03:00
|
|
|
helpArg: 'ACCOUNT'
|
|
|
|
},
|
2015-12-02 20:52:47 +02:00
|
|
|
{
|
|
|
|
names: ['act-as'],
|
|
|
|
type: 'string',
|
|
|
|
help: 'Masquerade as the given account login name. This can only ' +
|
|
|
|
'succeed for operator accounts. Note that accesses like these ' +
|
|
|
|
'audited on the CloudAPI server side.',
|
|
|
|
helpArg: 'ACCOUNT',
|
|
|
|
hidden: true
|
|
|
|
},
|
2015-11-04 01:40:59 +02:00
|
|
|
{
|
|
|
|
names: ['user', 'u'],
|
|
|
|
type: 'string',
|
|
|
|
help: 'RBAC user (login name). Environment: TRITON_USER=USER ' +
|
|
|
|
'or SDC_USER=USER.',
|
|
|
|
helpArg: 'USER'
|
|
|
|
},
|
|
|
|
// TODO: full rbac support
|
2015-08-27 03:21:27 +03:00
|
|
|
//{
|
|
|
|
// names: ['role'],
|
|
|
|
// type: 'arrayOfString',
|
|
|
|
// env: 'MANTA_ROLE',
|
|
|
|
// help: 'Assume a role. Use multiple times or once with a list',
|
|
|
|
// helpArg: 'ROLE,ROLE,...'
|
|
|
|
//},
|
|
|
|
{
|
|
|
|
names: ['keyId', 'k'],
|
|
|
|
type: 'string',
|
2015-09-25 20:22:58 +03:00
|
|
|
help: 'SSH key fingerprint. Environment: TRITON_KEY_ID=FINGERPRINT ' +
|
|
|
|
'or SDC_KEY_ID=FINGERPRINT.',
|
2015-11-25 21:04:44 +02:00
|
|
|
helpArg: 'FP'
|
2015-08-27 03:21:27 +03:00
|
|
|
},
|
|
|
|
{
|
2015-11-04 01:40:59 +02:00
|
|
|
names: ['url', 'U'],
|
2015-08-27 03:21:27 +03:00
|
|
|
type: 'string',
|
2015-09-21 22:34:37 +03:00
|
|
|
help: 'CloudAPI URL. Environment: TRITON_URL=URL or SDC_URL=URL.',
|
2015-08-27 03:21:27 +03:00
|
|
|
helpArg: 'URL'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
names: ['J'],
|
|
|
|
type: 'string',
|
|
|
|
hidden: true,
|
|
|
|
help: 'Joyent Public Cloud (JPC) datacenter name. This is ' +
|
|
|
|
'a shortcut to the "https://$dc.api.joyent.com" ' +
|
|
|
|
'cloudapi URL.'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
names: ['insecure', 'i'],
|
|
|
|
type: 'bool',
|
2015-09-21 22:34:37 +03:00
|
|
|
help: 'Do not validate the CloudAPI SSL certificate. Environment: ' +
|
|
|
|
'TRITON_TLS_INSECURE=1, SDC_TLS_INSECURE=1 (or the deprecated ' +
|
|
|
|
'SDC_TESTING=1).',
|
2015-09-21 22:37:59 +03:00
|
|
|
'default': false
|
2015-11-25 21:04:44 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
names: ['accept-version'],
|
|
|
|
type: 'string',
|
|
|
|
helpArg: 'VER',
|
|
|
|
help: 'A cloudapi API version, or semver range, to attempt to use. ' +
|
|
|
|
'This is passed in the "Accept-Version" header. ' +
|
|
|
|
'See `triton cloudapi /--ping` to list supported versions. ' +
|
|
|
|
'The default is "' + tritonapi.CLOUDAPI_ACCEPT_VERSION + '". ' +
|
|
|
|
'*This is intended for development use only. It could cause ' +
|
|
|
|
'`triton` processing of responses to break.*',
|
|
|
|
hidden: true
|
2015-08-27 03:21:27 +03:00
|
|
|
}
|
|
|
|
];
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
|
2015-12-31 19:55:31 +02:00
|
|
|
|
|
|
|
// ---- other support stuff
|
|
|
|
|
|
|
|
function parseCommaSepStringNoEmpties(option, optstr, arg) {
|
|
|
|
// JSSTYLED
|
|
|
|
return arg.trim().split(/\s*,\s*/g)
|
|
|
|
.filter(function (part) { return part; });
|
|
|
|
}
|
|
|
|
|
|
|
|
cmdln.dashdash.addOptionType({
|
|
|
|
name: 'commaSepString',
|
|
|
|
takesArg: true,
|
|
|
|
helpArg: 'STRING',
|
|
|
|
parseArg: parseCommaSepStringNoEmpties
|
|
|
|
});
|
|
|
|
|
|
|
|
cmdln.dashdash.addOptionType({
|
|
|
|
name: 'arrayOfCommaSepString',
|
|
|
|
takesArg: true,
|
|
|
|
helpArg: 'STRING',
|
|
|
|
parseArg: parseCommaSepStringNoEmpties,
|
|
|
|
array: true,
|
|
|
|
arrayFlatten: true
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
|
2014-02-07 23:21:24 +02:00
|
|
|
//---- CLI class
|
|
|
|
|
|
|
|
function CLI() {
|
|
|
|
Cmdln.call(this, {
|
2015-09-08 19:55:48 +03:00
|
|
|
name: 'triton',
|
2014-02-07 23:21:24 +02:00
|
|
|
desc: pkg.description,
|
2015-08-27 03:21:27 +03:00
|
|
|
options: OPTIONS,
|
2014-02-07 23:21:24 +02:00
|
|
|
helpOpts: {
|
|
|
|
includeEnv: true,
|
2015-08-26 07:34:47 +03:00
|
|
|
minHelpCol: 30
|
2015-08-26 01:10:13 +03:00
|
|
|
},
|
|
|
|
helpSubcmds: [
|
|
|
|
'help',
|
2016-01-14 19:22:13 +02:00
|
|
|
'completion',
|
2015-09-25 22:19:29 +03:00
|
|
|
'profile',
|
2015-08-26 01:15:02 +03:00
|
|
|
{ group: 'Instances (aka VMs/Machines/Containers)' },
|
2015-08-26 03:27:46 +03:00
|
|
|
'instance',
|
2016-01-04 19:05:53 +02:00
|
|
|
'instances',
|
|
|
|
'create',
|
2016-01-04 23:08:16 +02:00
|
|
|
'delete',
|
|
|
|
'start',
|
|
|
|
'stop',
|
|
|
|
'reboot',
|
2015-08-26 06:25:00 +03:00
|
|
|
'ssh',
|
2015-12-30 02:47:03 +02:00
|
|
|
{ group: 'Images, Packages, Networks' },
|
2015-08-26 01:47:29 +03:00
|
|
|
'image',
|
2015-08-26 23:40:50 +03:00
|
|
|
'package',
|
2015-12-30 02:47:03 +02:00
|
|
|
'network',
|
|
|
|
{ group: 'Other Commands' },
|
|
|
|
'info',
|
|
|
|
'account',
|
2015-12-31 04:43:46 +02:00
|
|
|
'key',
|
2015-12-30 02:47:03 +02:00
|
|
|
'services',
|
|
|
|
'datacenters'
|
2015-10-07 09:09:52 +03:00
|
|
|
],
|
|
|
|
helpBody: [
|
|
|
|
/* BEGIN JSSTYLED */
|
|
|
|
'Exit Status:',
|
|
|
|
' 0 Successful completion.',
|
|
|
|
' 1 An error occurred.',
|
|
|
|
' 2 Usage error.',
|
|
|
|
' 3 "ResourceNotFound" error. Returned when an instance, image,',
|
|
|
|
' package, etc. with the given name or id is not found.'
|
|
|
|
/* END JSSTYLED */
|
|
|
|
].join('\n')
|
2014-02-07 23:21:24 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
util.inherits(CLI, Cmdln);
|
|
|
|
|
|
|
|
CLI.prototype.init = function (opts, args, callback) {
|
|
|
|
var self = this;
|
2015-09-21 22:34:37 +03:00
|
|
|
this.opts = opts;
|
2014-02-07 23:21:24 +02:00
|
|
|
|
2015-09-08 19:55:48 +03:00
|
|
|
this.log = bunyan.createLogger({
|
|
|
|
name: this.name,
|
|
|
|
serializers: bunyan.stdSerializers,
|
|
|
|
stream: process.stderr,
|
|
|
|
level: 'warn'
|
|
|
|
});
|
2014-02-07 23:21:24 +02:00
|
|
|
if (opts.verbose) {
|
2015-09-08 19:55:48 +03:00
|
|
|
this.log.level('trace');
|
|
|
|
this.log.src = true;
|
2015-08-27 03:21:27 +03:00
|
|
|
this.showErrStack = true;
|
2014-02-07 23:21:24 +02:00
|
|
|
}
|
|
|
|
|
2015-12-08 21:59:45 +02:00
|
|
|
if (opts.version) {
|
|
|
|
console.log(this.name, pkg.version);
|
|
|
|
callback(false);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-21 22:34:37 +03:00
|
|
|
if (opts.url && opts.J) {
|
|
|
|
callback(new errors.UsageError(
|
|
|
|
'cannot use both "--url" and "-J" options'));
|
|
|
|
} else if (opts.J) {
|
2015-09-08 19:55:48 +03:00
|
|
|
opts.url = format('https://%s.api.joyent.com', opts.J);
|
|
|
|
}
|
2015-09-21 22:34:37 +03:00
|
|
|
|
2015-09-08 19:55:48 +03:00
|
|
|
this.configDir = CONFIG_DIR;
|
2015-08-26 19:59:12 +03:00
|
|
|
|
2015-09-08 19:55:48 +03:00
|
|
|
this.__defineGetter__('tritonapi', function () {
|
2015-11-04 00:39:18 +02:00
|
|
|
if (self._tritonapi === undefined) {
|
2015-09-08 19:55:48 +03:00
|
|
|
var config = mod_config.loadConfig({
|
|
|
|
configDir: self.configDir
|
|
|
|
});
|
|
|
|
self.log.trace({config: config}, 'loaded config');
|
2015-09-21 22:34:37 +03:00
|
|
|
|
2015-09-25 20:24:12 +03:00
|
|
|
var profileName = opts.profile || config.profile || 'env';
|
2015-09-21 22:34:37 +03:00
|
|
|
var profile = mod_config.loadProfile({
|
|
|
|
configDir: self.configDir,
|
|
|
|
name: profileName
|
|
|
|
});
|
|
|
|
self._applyProfileOverrides(profile);
|
2015-09-08 19:55:48 +03:00
|
|
|
self.log.trace({profile: profile}, 'loaded profile');
|
2015-08-27 03:21:27 +03:00
|
|
|
|
2015-09-30 01:13:34 +03:00
|
|
|
self._tritonapi = tritonapi.createClient({
|
2015-09-08 19:55:48 +03:00
|
|
|
log: self.log,
|
|
|
|
profile: profile,
|
|
|
|
config: config
|
2015-08-26 19:59:12 +03:00
|
|
|
});
|
2015-08-25 23:11:40 +03:00
|
|
|
}
|
2015-09-04 21:04:45 +03:00
|
|
|
return self._tritonapi;
|
2015-08-25 23:11:40 +03:00
|
|
|
});
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
// Cmdln class handles `opts.help`.
|
|
|
|
Cmdln.prototype.init.apply(this, arguments);
|
|
|
|
};
|
|
|
|
|
2015-08-25 23:11:40 +03:00
|
|
|
|
2015-12-08 21:59:45 +02:00
|
|
|
CLI.prototype.fini = function fini(subcmd, err, cb) {
|
|
|
|
this.log.trace({err: err, subcmd: subcmd}, 'cli fini');
|
|
|
|
if (this._tritonapi) {
|
|
|
|
this._tritonapi.close();
|
|
|
|
delete this._tritonapi;
|
|
|
|
}
|
|
|
|
cb(err, subcmd);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-09-21 22:34:37 +03:00
|
|
|
/*
|
|
|
|
* Apply overrides from CLI options to the given profile object *in place*.
|
|
|
|
*/
|
|
|
|
CLI.prototype._applyProfileOverrides =
|
|
|
|
function _applyProfileOverrides(profile) {
|
|
|
|
var self = this;
|
2015-11-25 21:04:44 +02:00
|
|
|
[
|
|
|
|
{oname: 'account', pname: 'account'},
|
|
|
|
{oname: 'user', pname: 'user'},
|
|
|
|
{oname: 'url', pname: 'url'},
|
|
|
|
{oname: 'keyId', pname: 'keyId'},
|
|
|
|
{oname: 'insecure', pname: 'insecure'},
|
2015-12-02 20:52:47 +02:00
|
|
|
{oname: 'accept_version', pname: 'acceptVersion'},
|
|
|
|
{oname: 'act_as', pname: 'actAsAccount'}
|
2015-11-25 21:04:44 +02:00
|
|
|
].forEach(function (field) {
|
2015-09-21 23:57:10 +03:00
|
|
|
// We need to check `opts._order` to know if boolean opts
|
|
|
|
// were specified.
|
|
|
|
var specified = self.opts._order.filter(
|
2015-11-25 21:04:44 +02:00
|
|
|
function (opt) { return opt.key === field.oname; }).length > 0;
|
2015-09-21 23:57:10 +03:00
|
|
|
if (specified) {
|
2015-11-25 21:04:44 +02:00
|
|
|
profile[field.pname] = self.opts[field.oname];
|
2015-09-21 22:34:37 +03:00
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-08-25 23:11:40 +03:00
|
|
|
|
2015-09-04 10:09:19 +03:00
|
|
|
// Meta
|
|
|
|
CLI.prototype.do_completion = require('./do_completion');
|
2015-09-08 19:55:48 +03:00
|
|
|
CLI.prototype.do_profiles = require('./do_profiles');
|
2015-09-25 22:19:29 +03:00
|
|
|
CLI.prototype.do_profile = require('./do_profile');
|
2015-11-25 02:40:17 +02:00
|
|
|
CLI.prototype.do_env = require('./do_env');
|
2014-02-07 23:21:24 +02:00
|
|
|
|
2015-09-04 10:09:19 +03:00
|
|
|
// Other
|
2015-08-26 06:44:08 +03:00
|
|
|
CLI.prototype.do_account = require('./do_account');
|
2015-08-27 02:56:18 +03:00
|
|
|
CLI.prototype.do_services = require('./do_services');
|
2015-08-27 02:59:28 +03:00
|
|
|
CLI.prototype.do_datacenters = require('./do_datacenters');
|
2015-08-26 07:16:41 +03:00
|
|
|
CLI.prototype.do_info = require('./do_info');
|
2015-12-31 04:43:46 +02:00
|
|
|
|
|
|
|
// Account keys
|
|
|
|
CLI.prototype.do_key = require('./do_key');
|
2015-08-26 07:40:32 +03:00
|
|
|
CLI.prototype.do_keys = require('./do_keys');
|
2015-08-26 06:44:08 +03:00
|
|
|
|
2015-08-26 01:10:13 +03:00
|
|
|
// Images
|
2015-08-26 00:25:30 +03:00
|
|
|
CLI.prototype.do_images = require('./do_images');
|
2015-08-26 01:47:29 +03:00
|
|
|
CLI.prototype.do_image = require('./do_image');
|
2015-08-26 00:25:30 +03:00
|
|
|
|
2015-08-26 01:15:02 +03:00
|
|
|
// Instances (aka VMs/containers/machines)
|
2015-08-26 03:27:46 +03:00
|
|
|
CLI.prototype.do_instance = require('./do_instance');
|
2015-08-26 01:15:02 +03:00
|
|
|
CLI.prototype.do_instances = require('./do_instances');
|
2016-01-04 19:05:53 +02:00
|
|
|
CLI.prototype.do_create = require('./do_create');
|
2016-01-04 23:08:16 +02:00
|
|
|
CLI.prototype.do_delete = require('./do_delete');
|
|
|
|
CLI.prototype.do_start = require('./do_start');
|
|
|
|
CLI.prototype.do_stop = require('./do_stop');
|
|
|
|
CLI.prototype.do_reboot = require('./do_reboot');
|
2015-08-26 06:25:00 +03:00
|
|
|
CLI.prototype.do_ssh = require('./do_ssh');
|
2014-02-07 23:21:24 +02:00
|
|
|
|
2015-08-26 01:30:25 +03:00
|
|
|
// Packages
|
|
|
|
CLI.prototype.do_packages = require('./do_packages');
|
2015-08-26 02:12:35 +03:00
|
|
|
CLI.prototype.do_package = require('./do_package');
|
2015-08-26 01:30:25 +03:00
|
|
|
|
2015-08-26 23:40:50 +03:00
|
|
|
// Networks
|
|
|
|
CLI.prototype.do_networks = require('./do_networks');
|
2015-08-27 00:09:50 +03:00
|
|
|
CLI.prototype.do_network = require('./do_network');
|
2015-08-26 23:40:50 +03:00
|
|
|
|
2015-08-26 02:12:35 +03:00
|
|
|
// Hidden commands
|
2015-08-26 01:30:25 +03:00
|
|
|
CLI.prototype.do_cloudapi = require('./do_cloudapi');
|
2015-08-26 02:12:35 +03:00
|
|
|
CLI.prototype.do_badger = require('./do_badger');
|
2015-11-04 01:40:59 +02:00
|
|
|
CLI.prototype.do_rbac = require('./do_rbac');
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
|
2015-07-26 08:45:20 +03:00
|
|
|
|
2015-08-25 22:14:16 +03:00
|
|
|
//---- mainline
|
|
|
|
|
2015-09-02 11:04:20 +03:00
|
|
|
function main(argv) {
|
|
|
|
if (!argv) {
|
|
|
|
argv = process.argv;
|
|
|
|
}
|
|
|
|
|
2015-08-25 22:14:16 +03:00
|
|
|
var cli = new CLI();
|
2015-09-02 11:04:20 +03:00
|
|
|
cli.main(argv, function (err, subcmd) {
|
|
|
|
var exitStatus = (err ? err.exitStatus || 1 : 0);
|
|
|
|
var showErr = (cli.showErr !== undefined ? cli.showErr : true);
|
|
|
|
|
|
|
|
if (err && showErr) {
|
2015-09-24 07:08:26 +03:00
|
|
|
var code = (err.body ? err.body.code : err.code) || err.restCode;
|
2015-09-02 11:04:20 +03:00
|
|
|
if (code === 'NoCommand') {
|
|
|
|
/* jsl:pass */
|
|
|
|
} else if (err.message !== undefined) {
|
2016-02-12 21:09:55 +02:00
|
|
|
/*
|
|
|
|
* If the err has `body.errors` (as some Triton/SDC APIs do per
|
|
|
|
* // JSSTYLED
|
|
|
|
* https://github.com/joyent/eng/blob/master/docs/index.md#error-handling
|
|
|
|
* then append a one-line summary for each error object.
|
|
|
|
*/
|
|
|
|
var bodyErrors = '';
|
|
|
|
if (err.body && err.body.errors) {
|
|
|
|
err.body.errors.forEach(function (e) {
|
|
|
|
bodyErrors += format('\n %s: %s', e.field, e.code);
|
|
|
|
if (e.message) {
|
|
|
|
bodyErrors += ': ' + e.message;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
console.error('%s%s: error%s: %s%s',
|
2015-09-02 11:04:20 +03:00
|
|
|
cli.name,
|
|
|
|
(subcmd ? ' ' + subcmd : ''),
|
|
|
|
(code ? format(' (%s)', code) : ''),
|
2016-02-12 21:09:55 +02:00
|
|
|
(cli.showErrStack ? err.stack : err.message),
|
|
|
|
bodyErrors);
|
2015-09-02 11:04:20 +03:00
|
|
|
|
|
|
|
// If this is a usage error, attempt to show some usage info.
|
|
|
|
if (['Usage', 'Option'].indexOf(code) !== -1 && subcmd) {
|
|
|
|
var help = cli.helpFromSubcmd(subcmd);
|
2015-11-04 00:39:37 +02:00
|
|
|
if (help && typeof (help) === 'string') {
|
2015-09-02 11:04:20 +03:00
|
|
|
// Would like a shorter synopsis. Attempt to
|
2015-11-04 00:39:37 +02:00
|
|
|
// parse it down, somewhat generally. Unfortunately this
|
|
|
|
// doesn't work for multi-level subcmds, like
|
|
|
|
// `triton rbac subcmd ...`.
|
2015-09-02 11:04:20 +03:00
|
|
|
var usageIdx = help.indexOf('\nUsage:');
|
|
|
|
if (usageIdx !== -1) {
|
|
|
|
help = help.slice(usageIdx);
|
|
|
|
}
|
|
|
|
console.error(help);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-08 21:59:45 +02:00
|
|
|
/*
|
|
|
|
* We'd like to NOT use `process.exit` because that doesn't always
|
|
|
|
* allow std handles to flush (e.g. all logging to complete). However
|
|
|
|
* I don't know of another way to exit non-zero.
|
|
|
|
*/
|
|
|
|
if (exitStatus !== 0) {
|
|
|
|
process.exit(exitStatus);
|
|
|
|
}
|
2015-09-02 11:04:20 +03:00
|
|
|
});
|
2015-08-25 22:14:16 +03:00
|
|
|
}
|
2014-02-07 23:21:24 +02:00
|
|
|
|
|
|
|
//---- exports
|
|
|
|
|
2015-09-02 11:04:20 +03:00
|
|
|
module.exports = {
|
2015-09-08 19:55:48 +03:00
|
|
|
CONFIG_DIR: CONFIG_DIR,
|
2015-09-02 11:04:20 +03:00
|
|
|
CLI: CLI,
|
|
|
|
main: main
|
|
|
|
};
|