#!/usr/bin/env node
/**
* @copyright Copyright 2016 Kevin Locke <kevin@kevinlocke.name>
* @license MIT
*/
'use strict';
// Set NODE_DEBUG for request before importing it
if (require.main === module
&& (process.argv.includes('--debug')
|| process.argv.includes('--debug-http'))) {
const nodeDebug = process.env.NODE_DEBUG;
if (!nodeDebug) {
process.env.NODE_DEBUG = 'request';
} else if (!/\brequest\b/.test(nodeDebug)) {
process.env.NODE_DEBUG = `${nodeDebug},request`;
}
}
const Chalk = require('chalk').Instance;
const { Command } = require('commander');
const util = require('util');
const packageJson = require('../package.json');
const stateInfo = require('../lib/state-info');
const travisStatus = require('..');
const debug = util.debuglog('travis-status');
/** Options for command entry points.
*
* @typedef {{
* in: (stream.Readable|undefined),
* out: (stream.Writable|undefined),
* err: (stream.Writable|undefined)
* }} CommandOptions
* @property {stream.Readable=} in Stream from which input is read. (default:
* <code>process.stdin</code>)
* @property {stream.Writable=} out Stream to which output is written.
* (default: <code>process.stdout</code>)
* @property {stream.Writable=} err Stream to which errors (and non-output
* status messages) are written. (default: <code>process.stderr</code>)
*/
// var CommandOptions;
/** Entry point for this command.
*
* @param {!Array<string>} args Command-line arguments.
* @param {CommandOptions=} options Options.
* @param {?function(Error, number=)=}
* callback Callback for the exit code or an <code>Error</code>.
* @return {Promise<number>|undefined} If <code>callback</code> is not given,
* a <code>Promise</code> with the exit code or <code>Error</code>.
*/
function travisStatusCmd(args, options, callback) {
if (!callback && typeof options === 'function') {
callback = options;
options = null;
}
if (!callback) {
return new Promise((resolve, reject) => {
travisStatusCmd(args, options, (err, result) => {
if (err) { reject(err); } else { resolve(result); }
});
});
}
if (typeof callback !== 'function') {
throw new TypeError('callback must be a function');
}
try {
if (args === undefined || args === null || args.length === 0) {
// Fake args to keep Commander.js happy
args = [
process.execPath,
__filename,
];
} else if (typeof args !== 'object'
|| Math.floor(args.length) !== args.length) {
throw new TypeError('args must be Array-like');
} else if (args.length < 2) {
throw new RangeError('non-empty args must have at least 2 elements');
} else {
args = Array.prototype.map.call(args, String);
}
if (options && typeof options !== 'object') {
throw new TypeError('options must be an object');
}
options = {
in: process.stdin,
out: process.stdout,
err: process.stderr,
...options,
};
if (!options.in || typeof options.in.read !== 'function') {
throw new TypeError('options.in must be a stream.Readable');
}
if (!options.out || typeof options.out.write !== 'function') {
throw new TypeError('options.out must be a stream.Writable');
}
if (!options.err || typeof options.err.write !== 'function') {
throw new TypeError('options.err must be a stream.Writable');
}
} catch (err) {
process.nextTick(() => {
callback(err);
});
return undefined;
}
const command = new Command()
.description('Checks status of the latest build.')
// Note: Option order matches travis.rb with new ones at bottom
.option('-i, --interactive', 'be interactive and colorful')
.option('-E, --explode', 'ignored for compatibility with travis.rb')
.option('--skip-version-check', 'ignored for compatibility with travis.rb')
.option('--skip-completion-check',
'ignored for compatibility with travis.rb')
.option('-I, --insecure', 'do not verify SSL certificate of API endpoint')
.option('-e, --api-endpoint <URL>', 'Travis API server to talk to')
.option('--pro',
`short-cut for --api-endpoint '${travisStatus.PRO_URI}'`)
.on('option:pro', function() {
this.apiEndpoint = travisStatus.PRO_URI;
})
.option('--org',
`short-cut for --api-endpoint '${travisStatus.ORG_URI}'`)
.on('option:org', function() {
this.apiEndpoint = travisStatus.ORG_URI;
})
.option('--staging', 'talks to staging system')
.on('option:staging', function() {
this.apiEndpoint = (this.apiEndpoint || travisStatus.ORG_URI)
.replace(/api/g, 'api-staging');
})
.option('-t, --token <ACCESS_TOKEN>', 'access token to use')
.option('--debug', 'show API requests')
.option('--debug-http', 'show HTTP(S) exchange')
.option('-r, --repo <SLUG>',
'repository to use (will try to detect from current git clone)')
.option('-R, --store-repo <SLUG>',
'like --repo, but remembers value for current directory')
.on('option:store-repo', function() {
this.repo = this.storeRepo;
})
.option('-x, --exit-code', 'sets the exit code to 1 if the build failed')
.option('-q, --quiet', 'does not print anything')
.option('-p, --fail-pending',
'sets the status code to 1 if the build is pending')
.option('-b, --branch [BRANCH]',
'query latest build for a branch (default: current)')
.option('-c, --commit [COMMIT]',
'require build to be for a specific commit (default: HEAD)')
.option('-w, --wait [TIMEOUT]',
'wait if build is pending (timeout in seconds)')
.version(packageJson.version);
// Patch stdout, stderr, and exit for Commander
// See: https://github.com/tj/commander.js/pull/444
const exitDesc = Object.getOwnPropertyDescriptor(process, 'exit');
const stdoutDesc = Object.getOwnPropertyDescriptor(process, 'stdout');
const stderrDesc = Object.getOwnPropertyDescriptor(process, 'stderr');
const consoleDesc = Object.getOwnPropertyDescriptor(global, 'console');
const errExit = new Error('process.exit() called');
process.exit = function throwOnExit(code) {
errExit.code = code;
throw errExit;
};
if (options.out) {
Object.defineProperty(
process,
'stdout',
{ configurable: true, enumerable: true, value: options.out },
);
}
if (options.err) {
Object.defineProperty(
process,
'stderr',
{ configurable: true, enumerable: true, value: options.err },
);
}
if (options.out || options.err) {
Object.defineProperty(
global,
'console',
{
configurable: true,
enumerable: true,
// eslint-disable-next-line no-console
value: new console.Console(process.stdout, process.stderr),
},
);
}
try {
command.parse(args);
} catch (errParse) {
const exitCode = errParse === errExit ? errExit.code || 0 : null;
process.nextTick(() => {
if (exitCode !== null) {
callback(null, exitCode);
} else {
callback(errParse);
}
});
return undefined;
} finally {
Object.defineProperty(process, 'exit', exitDesc);
Object.defineProperty(process, 'stdout', stdoutDesc);
Object.defineProperty(process, 'stderr', stderrDesc);
Object.defineProperty(global, 'console', consoleDesc);
}
if (command.commit === true) {
command.commit = 'HEAD';
}
if (typeof command.interactive === 'undefined') {
// Note: Same default as travis.rb
// Need cast to Boolean so undefined becomes false to disable Chalk
command.interactive = Boolean(options.out.isTTY);
}
if (command.wait === true) {
command.wait = Infinity;
}
const chalk = new Chalk({
level: command.interactive ? 1 : 0,
});
if (command.args.length > 0) {
options.err.write(`${chalk.red('too many arguments')}\n${
command.helpInformation()}`);
process.nextTick(() => { callback(null, 1); });
return undefined;
}
if (hasOwnProperty.call(command, 'wait')) {
const wait = Number(command.wait);
if (Number.isNaN(wait)) {
const waitErr = chalk.red(`invalid wait time "${command.wait}"`);
options.err.write(`${waitErr}\n`);
process.nextTick(() => { callback(null, 1); });
return undefined;
}
command.wait = wait * 1000;
}
// Pass through options
command.in = options.in;
command.out = options.out;
command.err = options.err;
// Use HTTP keep-alive to avoid unnecessary reconnections
command.requestOpts = {
forever: true,
};
if (command.insecure) {
command.requestOpts.strictSSL = false;
}
travisStatus(command, (err, build) => {
if (err && err.name === 'SlugDetectionError') {
debug('Error detecting repo slug', err);
options.err.write(chalk.red(
'Can\'t figure out GitHub repo name. '
+ 'Ensure you\'re in the repo directory, or specify the repo name via '
+ 'the -r option (e.g. travis-status -r <owner>/<repo>)\n',
));
callback(null, 1);
return;
}
if (err) {
options.err.write(`${chalk.red(err.message)}\n`);
callback(null, 1);
return;
}
const state = build.repo ? build.repo.last_build_state : build.branch.state;
if (!command.quiet) {
const color = stateInfo.colors[state] || 'yellow';
const number =
build.repo ? build.repo.last_build_number : build.branch.number;
options.out.write(`build #${number} ${chalk[color](state)
}\n`);
}
let code = 0;
if ((command.exitCode && stateInfo.isUnsuccessful[state])
|| (command.failPending && stateInfo.isPending[state])) {
code = 1;
}
callback(null, code);
});
return undefined;
}
module.exports = travisStatusCmd;
if (require.main === module) {
// This file was invoked directly.
/* eslint-disable no-process-exit */
const mainOptions = {
in: process.stdin,
out: process.stdout,
err: process.stderr,
};
travisStatusCmd(process.argv, mainOptions, (err, code) => {
if (err) {
if (err.stdout) { process.stdout.write(err.stdout); }
if (err.stderr) { process.stderr.write(err.stderr); }
process.stderr.write(`${err.name}: ${err.message}\n`);
code = typeof err.code === 'number' ? err.code : 1;
}
process.exit(code);
});
}