2010-03-18 05:00:17 +08:00
|
|
|
var inherits = require('sys').inherits;
|
|
|
|
var EventEmitter = require('events').EventEmitter;
|
2010-03-18 07:31:24 +08:00
|
|
|
var Stream = require('net').Stream;
|
2010-03-18 05:00:17 +08:00
|
|
|
var InternalChildProcess = process.binding('child_process').ChildProcess;
|
|
|
|
|
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
var spawn = exports.spawn = function (path, args, env, customFds) {
|
2010-03-18 05:00:17 +08:00
|
|
|
var child = new ChildProcess();
|
2010-06-01 15:56:08 +08:00
|
|
|
child.spawn(path, args, env, customFds);
|
2010-03-18 05:00:17 +08:00
|
|
|
return child;
|
|
|
|
};
|
|
|
|
|
2010-04-15 09:50:41 +08:00
|
|
|
exports.exec = function (command /*, options, callback */) {
|
|
|
|
if (arguments.length < 3) {
|
|
|
|
return exports.execFile("/bin/sh", ["-c", command], arguments[1]);
|
|
|
|
} else {
|
|
|
|
return exports.execFile("/bin/sh", ["-c", command], arguments[1], arguments[2]);
|
|
|
|
}
|
|
|
|
};
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-04-15 09:50:41 +08:00
|
|
|
exports.execFile = function (file, args /*, options, callback */) {
|
2010-04-15 02:59:24 +08:00
|
|
|
var options = { encoding: 'utf8'
|
|
|
|
, timeout: 0
|
|
|
|
, maxBuffer: 200*1024
|
|
|
|
, killSignal: 'SIGKILL'
|
|
|
|
};
|
|
|
|
|
|
|
|
var callback = arguments[arguments.length-1];
|
2010-04-15 09:50:41 +08:00
|
|
|
|
|
|
|
if (typeof arguments[2] == 'object') {
|
2010-04-15 02:59:24 +08:00
|
|
|
var keys = Object.keys(options);
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
var k = keys[i];
|
2010-04-15 09:50:41 +08:00
|
|
|
if (arguments[2][k] !== undefined) options[k] = arguments[2][k];
|
2010-04-15 02:59:24 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-04-15 09:50:41 +08:00
|
|
|
var child = spawn(file, args);
|
2010-03-18 05:00:17 +08:00
|
|
|
var stdout = "";
|
|
|
|
var stderr = "";
|
2010-04-15 02:59:24 +08:00
|
|
|
var killed = false;
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-04-15 02:59:24 +08:00
|
|
|
var timeoutId;
|
|
|
|
if (options.timeout > 0) {
|
|
|
|
timeoutId = setTimeout(function () {
|
|
|
|
if (!killed) {
|
|
|
|
child.kill(options.killSignal);
|
|
|
|
killed = true;
|
|
|
|
timeoutId = null;
|
|
|
|
}
|
|
|
|
}, options.timeout);
|
|
|
|
}
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-04-15 02:59:24 +08:00
|
|
|
child.stdout.setEncoding(options.encoding);
|
|
|
|
child.stderr.setEncoding(options.encoding);
|
|
|
|
|
|
|
|
child.stdout.addListener("data", function (chunk) {
|
|
|
|
stdout += chunk;
|
|
|
|
if (!killed && stdout.length > options.maxBuffer) {
|
|
|
|
child.kill(options.killSignal);
|
|
|
|
killed = true;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
child.stderr.addListener("data", function (chunk) {
|
|
|
|
stderr += chunk;
|
|
|
|
if (!killed && stderr.length > options.maxBuffer) {
|
|
|
|
child.kill(options.killSignal);
|
|
|
|
killed = true
|
|
|
|
}
|
|
|
|
});
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-04-28 21:04:08 +08:00
|
|
|
child.addListener("exit", function (code, signal) {
|
2010-04-15 02:59:24 +08:00
|
|
|
if (timeoutId) clearTimeout(timeoutId);
|
2010-04-28 21:04:08 +08:00
|
|
|
if (code === 0 && signal === null) {
|
2010-03-18 05:00:17 +08:00
|
|
|
if (callback) callback(null, stdout, stderr);
|
|
|
|
} else {
|
|
|
|
var e = new Error("Command failed: " + stderr);
|
2010-04-15 02:59:24 +08:00
|
|
|
e.killed = killed;
|
2010-03-18 05:00:17 +08:00
|
|
|
e.code = code;
|
2010-04-28 21:04:08 +08:00
|
|
|
e.signal = signal;
|
2010-03-18 05:00:17 +08:00
|
|
|
if (callback) callback(e, stdout, stderr);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
function ChildProcess () {
|
2010-05-09 13:11:55 +08:00
|
|
|
EventEmitter.call(this);
|
2010-03-18 05:00:17 +08:00
|
|
|
|
|
|
|
var self = this;
|
|
|
|
|
|
|
|
var gotCHLD = false;
|
|
|
|
var exitCode;
|
2010-04-28 21:04:08 +08:00
|
|
|
var termSignal;
|
2010-03-18 05:00:17 +08:00
|
|
|
var internal = this._internal = new InternalChildProcess();
|
|
|
|
|
2010-03-18 07:31:24 +08:00
|
|
|
var stdin = this.stdin = new Stream();
|
|
|
|
var stdout = this.stdout = new Stream();
|
|
|
|
var stderr = this.stderr = new Stream();
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-05-09 14:28:26 +08:00
|
|
|
var stderrClosed = false;
|
|
|
|
var stdoutClosed = false;
|
|
|
|
|
|
|
|
stderr.addListener('close', function () {
|
|
|
|
stderrClosed = true;
|
2010-06-01 15:56:08 +08:00
|
|
|
if (gotCHLD && (!self.stdout || stdoutClosed)) {
|
2010-04-28 21:04:08 +08:00
|
|
|
self.emit('exit', exitCode, termSignal);
|
2010-03-18 05:00:17 +08:00
|
|
|
}
|
2010-05-09 14:28:26 +08:00
|
|
|
});
|
2010-05-09 13:11:55 +08:00
|
|
|
|
2010-05-09 14:28:26 +08:00
|
|
|
stdout.addListener('close', function () {
|
|
|
|
stdoutClosed = true;
|
2010-06-01 15:56:08 +08:00
|
|
|
if (gotCHLD && (!self.stderr || stderrClosed)) {
|
2010-05-09 14:28:26 +08:00
|
|
|
self.emit('exit', exitCode, termSignal);
|
|
|
|
}
|
|
|
|
});
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-04-28 21:04:08 +08:00
|
|
|
internal.onexit = function (code, signal) {
|
2010-03-18 05:00:17 +08:00
|
|
|
gotCHLD = true;
|
|
|
|
exitCode = code;
|
2010-04-28 21:04:08 +08:00
|
|
|
termSignal = signal;
|
2010-06-01 15:56:08 +08:00
|
|
|
if (self.stdin) {
|
|
|
|
self.stdin.end();
|
|
|
|
}
|
|
|
|
if ( (!self.stdout || !self.stdout.readable)
|
|
|
|
&& (!self.stderr || !self.stderr.readable)) {
|
2010-04-28 21:04:08 +08:00
|
|
|
self.emit('exit', exitCode, termSignal);
|
2010-03-18 05:00:17 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
this.__defineGetter__('pid', function () { return internal.pid; });
|
|
|
|
}
|
|
|
|
inherits(ChildProcess, EventEmitter);
|
|
|
|
|
|
|
|
|
|
|
|
ChildProcess.prototype.kill = function (sig) {
|
|
|
|
return this._internal.kill(sig);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
ChildProcess.prototype.spawn = function (path, args, env, customFds) {
|
2010-03-18 05:00:17 +08:00
|
|
|
args = args || [];
|
|
|
|
env = env || process.env;
|
|
|
|
var envPairs = [];
|
2010-04-13 00:57:24 +08:00
|
|
|
var keys = Object.keys(env);
|
|
|
|
for (var index = 0, keysLength = keys.length; index < keysLength; index++) {
|
|
|
|
var key = keys[index];
|
|
|
|
envPairs.push(key + "=" + env[key]);
|
2010-03-18 05:00:17 +08:00
|
|
|
}
|
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
customFds = customFds || [-1, -1, -1];
|
|
|
|
var fds = this.fds = this._internal.spawn(path, args, envPairs, customFds);
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
if (customFds[0] === -1 || customFds[0] === undefined) {
|
|
|
|
this.stdin.open(fds[0]);
|
|
|
|
this.stdin.writable = true;
|
|
|
|
this.stdin.readable = false;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.stdin = null;
|
|
|
|
}
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
if (customFds[1] === -1 || customFds[1] === undefined) {
|
|
|
|
this.stdout.open(fds[1]);
|
|
|
|
this.stdout.writable = false;
|
|
|
|
this.stdout.readable = true;
|
|
|
|
this.stdout.resume();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.stdout = null;
|
|
|
|
}
|
2010-03-18 05:00:17 +08:00
|
|
|
|
2010-06-01 15:56:08 +08:00
|
|
|
if (customFds[2] === -1 || customFds[2] === undefined) {
|
|
|
|
this.stderr.open(fds[2]);
|
|
|
|
this.stderr.writable = false;
|
|
|
|
this.stderr.readable = true;
|
|
|
|
this.stderr.resume();
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
this.stderr = null;
|
|
|
|
}
|
2010-03-18 05:00:17 +08:00
|
|
|
};
|
|
|
|
|