node/lib/child_process.js

207 lines
5.2 KiB
JavaScript
Raw Normal View History

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;
var spawn = exports.spawn = function (path, args, env, customFds) {
2010-03-18 05:00:17 +08:00
var child = new ChildProcess();
child.spawn(path, args, env, customFds);
2010-03-18 05:00:17 +08:00
return child;
};
exports.exec = function (command /*, options, callback */) {
if (arguments.length < 3) {
return exports.execFile("/bin/sh", ["-c", command], arguments[1]);
2010-07-09 21:05:54 +08:00
} else if (arguments.length < 4) {
return exports.execFile("/bin/sh", ["-c", command], arguments[1], arguments[2]);
2010-07-09 21:05:54 +08:00
} else {
return exports.execFile("/bin/sh", ["-c", command], arguments[1], arguments[2], arguments[3]);
}
};
2010-03-18 05:00:17 +08:00
// execFile("something.sh", { env: ENV }, funciton() { })
exports.execFile = function (file /* args, options, callback */) {
var options = { encoding: 'utf8'
, timeout: 0
, maxBuffer: 200*1024
, killSignal: 'SIGKILL'
, env: null
};
var args, optionArg, callback;
// Parse the parameters.
if (typeof arguments[arguments.length-1] === "function") {
callback = arguments[arguments.length-1];
}
if (Array.isArray(arguments[1])) {
args = arguments[1];
if (typeof arguments[2] === 'object') optionArg = arguments[2];
} else {
args = [];
if (typeof arguments[1] === 'object') optionArg = arguments[1];
}
// Merge optionArg into options
if (optionArg) {
var keys = Object.keys(options);
for (var i = 0; i < keys.length; i++) {
var k = keys[i];
if (optionArg[k] !== undefined) options[k] = optionArg[k];
}
}
var child = spawn(file, args, options.env);
2010-03-18 05:00:17 +08:00
var stdout = "";
var stderr = "";
var killed = false;
2010-03-18 05:00:17 +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
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
child.addListener("exit", function (code, signal) {
if (timeoutId) clearTimeout(timeoutId);
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);
e.killed = killed;
2010-03-18 05:00:17 +08:00
e.code = code;
e.signal = signal;
2010-03-18 05:00:17 +08:00
if (callback) callback(e, stdout, stderr);
}
});
2010-07-13 05:18:09 +08:00
return child;
2010-03-18 05:00:17 +08:00
};
function ChildProcess () {
EventEmitter.call(this);
2010-03-18 05:00:17 +08:00
var self = this;
var gotCHLD = false;
var exitCode;
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
var stderrClosed = false;
var stdoutClosed = false;
stderr.addListener('close', function () {
stderrClosed = true;
if (gotCHLD && (!self.stdout || stdoutClosed)) {
self.emit('exit', exitCode, termSignal);
2010-03-18 05:00:17 +08:00
}
});
stdout.addListener('close', function () {
stdoutClosed = true;
if (gotCHLD && (!self.stderr || stderrClosed)) {
self.emit('exit', exitCode, termSignal);
}
});
2010-03-18 05:00:17 +08:00
internal.onexit = function (code, signal) {
2010-03-18 05:00:17 +08:00
gotCHLD = true;
exitCode = code;
termSignal = signal;
if (self.stdin) {
self.stdin.end();
}
if ( (!self.stdout || !self.stdout.readable)
&& (!self.stderr || !self.stderr.readable)) {
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);
};
ChildProcess.prototype.spawn = function (path, args, env, customFds) {
2010-03-18 05:00:17 +08:00
args = args || [];
env = env || process.env;
var envPairs = [];
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
}
customFds = customFds || [-1, -1, -1];
var fds = this.fds = this._internal.spawn(path, args, envPairs, customFds);
2010-03-18 05:00:17 +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
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
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
};