firmament-yargs
Version:
Typescript classes for building CLI node applications
239 lines (222 loc) • 8.32 kB
text/typescript
import {injectable, inject} from "inversify";
import {Spawn} from '../interfaces/spawn';
import {ChildProcess, SpawnSyncReturns, spawn, spawnSync} from 'child_process';
import {CommandUtil} from '../interfaces/command-util';
import {ForceErrorImpl} from "./force-error-impl";
import {SpawnOptions2} from "../custom-typings";
const readlineSync = require('readline-sync');
const inpathSync = require('inpath').sync;
const pidof = require('pidof');
const psTree = require('ps-tree');
//noinspection JSUnusedGlobalSymbols
()
export class SpawnImpl extends ForceErrorImpl implements Spawn {
cachedPassword: string;
commandUtil: CommandUtil;
constructor(('CommandUtil')_commandUtil: CommandUtil) {
super();
this.commandUtil = _commandUtil;
}
spawnShellCommandPipelineAsync(cmdArray: string[][],
options?: SpawnOptions2,
cbStatusOrFinal?: (err: Error, result: string)=>void,
cbFinal?: (err: Error, result: string)=>void): ChildProcess {
let lclCmdArray = cmdArray.slice(0);
let cmd = lclCmdArray.pop().slice(0);
let childProcess = this.spawnShellCommandAsync(cmd, options, cbStatusOrFinal, cbFinal);
let proc = childProcess;
while (cmd = lclCmdArray.pop()) {
cmd = cmd.slice(0);
let prevStdin = proc.stdin;
proc = this.spawnShellCommandAsync(cmd);
//proc = spawn(cmd.shift(), cmd);
proc.stdout.pipe(prevStdin);
}
return childProcess;
}
sudoSpawnPipelineAsync(cmdArray: string[][],
options?: SpawnOptions2,
cbStatusOrFinal?: (err: Error, result: string)=>void,
cbFinal?: (err: Error, result: string)=>void): ChildProcess {
return undefined;
}
spawnShellCommandAsync(cmd: string[],
options: SpawnOptions2 = null,
cbStatusOrFinal: (err: Error, result: string)=>void = null,
cbFinal: (err: Error, result: string)=>void = null) {
let me = this;
let childProcess: ChildProcess;
//Do some callback juggling. If they don't specify cbFinal the cbStatus will be the
//presumptive cbFinal. If neither callback is specified the user will need to deal
//with the returned child process object to get results from the process.
if (typeof cbFinal !== 'function') {
if (typeof cbStatusOrFinal !== 'function') {
cbFinal = cbStatusOrFinal = null;
} else {
cbFinal = cbStatusOrFinal;
cbStatusOrFinal = null;
}
} else {
if (typeof cbStatusOrFinal !== 'function') {
cbStatusOrFinal = null;
}
}
if (me.checkForceError('spawnShellCommandAsync', cbFinal)) {
return;
}
try {
cmd = cmd || [];
cmd = cmd.slice(0);
options = options || {};
options.preSpawnMessage = options.preSpawnMessage || '';
options.postSpawnMessage = options.postSpawnMessage || '';
options.showDiagnostics = options.showDiagnostics || false;
options.suppressOutput = options.suppressOutput || false;
options.stdio = options.stdio || 'pipe';
options.cwd = options.cwd || process.cwd();
if (options.showDiagnostics) {
//Meager diagnostics for now. Maybe bolster later.
me.commandUtil.log(`Running '${cmd}' @ '${options.cwd}'`);
}
if (options.preSpawnMessage && cbStatusOrFinal) {
cbStatusOrFinal(null, options.preSpawnMessage);
}
let command = cmd.shift();
let result = '';
childProcess = spawn(command, cmd, options);
//Don't attach events if there is no final callback
if (cbFinal) {
childProcess.stdout.on('data', function (data) {
result += data.toString();
if (cbStatusOrFinal && !options.suppressOutput) {
cbStatusOrFinal(null, data.toString());
}
});
childProcess.on('exit', function (code: number, signal: string) {
if (options.postSpawnMessage && cbStatusOrFinal) {
cbStatusOrFinal(null, options.postSpawnMessage);
cbStatusOrFinal = null;
}
cbFinal = me.childCloseOrExit(code, signal, result, cbFinal);
});
childProcess.on('close', function (code: number, signal: string) {
if (options.postSpawnMessage && cbStatusOrFinal) {
cbStatusOrFinal(null, options.postSpawnMessage);
cbStatusOrFinal = null;
}
cbFinal = me.childCloseOrExit(code, signal, result, cbFinal);
});
childProcess.on('error', function (code: any) {
if (cbFinal) {
cbFinal(new Error(`spawn error: exit code ${code.code}`), null);
cbFinal = null;
}
});
}
} catch (err) {
cbFinal(err, null);
}
return childProcess;
}
private childCloseOrExit(code: number,
signal: string,
result: string,
cbFinal: (err: Error, result: string)=>void): (err: Error, result: string)=>void {
if (cbFinal) {
if (code !== null) {
if (code !== 0) {
cbFinal(new Error(`spawn error: exit code ${code}`), result);
}
else {
cbFinal(null, result);
}
} else {
cbFinal(new Error(`child process received signal: ${signal}`), result);
}
cbFinal = null;
}
return cbFinal;
}
sudoSpawnAsync(cmd: string[],
options: SpawnOptions2 = null,
cbStatusOrFinal: (err: Error, result: string)=>void = null,
cbFinal: (err: Error, result: string)=>void = null) {
let me = this;
let prompt = '#node-sudo-passwd#';
let prompts = 0;
let args = ['-S', '-p', prompt];
cmd = cmd || [];
cmd = cmd.slice(0);
[].push.apply(args, cmd);
let path = process.env['PATH'].split(':');
let sudoBin = inpathSync('sudo', path);
args.unshift(sudoBin);
//TODO: Set default options
let child: ChildProcess = me.spawnShellCommandAsync(args, options, cbStatusOrFinal, cbFinal);
if (!child) {
//In this case spawnShellCommandAsync should handle the error callbacks
return;
}
function waitForStartup(err, children: any[]) {
if (err) {
throw new Error(`Error spawning process`);
}
if (children && children.length) {
child.stderr.removeAllListeners();
} else {
setTimeout(function () {
psTree(child.pid, waitForStartup);
}, 100);
}
}
psTree(child.pid, waitForStartup);
child.stderr.on('data', function (data) {
let lines = data.toString().trim().split('\n');
lines.forEach(function (line) {
if (line === prompt) {
if (++prompts > 1) {
// The previous entry must have been incorrect, since sudo asks again.
me.cachedPassword = null;
}
let username = require('username').sync();
let loginMessage = (prompts > 1)
? `Sorry, try again.\n[sudo] password for ${username}: `
: `[sudo] password for ${username}: `;
if (me.cachedPassword) {
child.stdin.write(me.cachedPassword + '\n');
} else {
me.cachedPassword = readlineSync.question(loginMessage, {hideEchoBack: true});
child.stdin.write(me.cachedPassword + '\n');
}
}
});
});
return child;
}
}
/*
public spawnShellCommandSync(cmd: string[],
options: SpawnOptions2 = null,
cb: (err: Error,
spawnSyncReturns: SpawnSyncReturns<Buffer>)=>void = null): SpawnSyncReturns<Buffer> {
cb = this.checkCallback(cb);
let child: SpawnSyncReturns<Buffer>;
if (this.checkForceError('spawnShellCommandSync', cb)) {
return;
}
try {
cmd = cmd || [];
cmd = cmd.slice(0);
options = options || {preSpawnMessage: '', postSpawnMessage: '', showDiagnostics: false};
options.stdio = options.stdio || 'inherit';
options.cwd = process.cwd() || __dirname;
this.commandUtil.log(`Running '${cmd}' @ '${options.cwd}'`);
let command = cmd.shift();
child = spawnSync(command, cmd, options);
cb(child.error, child);
} catch (err) {
cb(err, null);
}
return child;
}
*/