2014-02-21 20:25:31 -05:00
|
|
|
var cp = require('child_process'),
|
|
|
|
_ = require('lodash'),
|
|
|
|
fs = require('fs'),
|
|
|
|
url = require('url'),
|
|
|
|
net = require('net'),
|
2014-08-17 06:17:23 +00:00
|
|
|
Promise = require('bluebird'),
|
2014-02-21 20:25:31 -05:00
|
|
|
path = require('path'),
|
|
|
|
config = require('../../server/config');
|
|
|
|
|
|
|
|
function findFreePort(port) {
|
2014-08-17 06:17:23 +00:00
|
|
|
return new Promise(function (resolve, reject) {
|
|
|
|
if (typeof port === 'string') {
|
|
|
|
port = parseInt(port);
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
2014-08-17 06:17:23 +00:00
|
|
|
|
|
|
|
if (typeof port !== 'number') {
|
|
|
|
port = 2368;
|
|
|
|
}
|
|
|
|
|
|
|
|
port = port + 1;
|
|
|
|
|
|
|
|
var server = net.createServer();
|
|
|
|
|
|
|
|
server.on('error', function(e) {
|
|
|
|
if (e.code === 'EADDRINUSE') {
|
|
|
|
resolve(findFreePort(port));
|
|
|
|
} else {
|
|
|
|
reject(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
server.listen(port, function() {
|
|
|
|
var listenPort = server.address().port;
|
|
|
|
server.close(function() {
|
|
|
|
resolve(listenPort);
|
|
|
|
});
|
2014-02-21 20:25:31 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get a copy of current config object from file, to be modified before
|
|
|
|
// passing to forkGhost() method
|
|
|
|
function forkConfig() {
|
|
|
|
// require caches values, and we want to read it fresh from the file
|
2014-07-17 10:33:21 -04:00
|
|
|
delete require.cache[config.paths.config];
|
|
|
|
return _.cloneDeep(require(config.paths.config)[process.env.NODE_ENV]);
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Creates a new fork of Ghost process with a given config
|
|
|
|
// Useful for tests that want to verify certain config options
|
|
|
|
function forkGhost(newConfig, envName) {
|
|
|
|
envName = envName || 'forked';
|
2014-08-17 06:17:23 +00:00
|
|
|
|
|
|
|
return findFreePort(newConfig.server ? newConfig.server.port : undefined)
|
2014-02-21 20:25:31 -05:00
|
|
|
.then(function(port) {
|
|
|
|
newConfig.server = newConfig.server || {};
|
|
|
|
newConfig.server.port = port;
|
|
|
|
newConfig.url = url.format(_.extend(url.parse(newConfig.url), {port: port, host: null}));
|
2014-08-17 06:17:23 +00:00
|
|
|
|
2014-07-17 10:33:21 -04:00
|
|
|
var newConfigFile = path.join(config.paths.appRoot, 'config.test' + port + '.js');
|
2014-08-17 06:17:23 +00:00
|
|
|
|
|
|
|
return new Promise(function (resolve, reject) {
|
|
|
|
fs.writeFile(newConfigFile, 'module.exports = {' + envName + ': ' + JSON.stringify(newConfig) + '}', function(err) {
|
|
|
|
if (err) {
|
|
|
|
return reject(err);
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
2014-08-17 06:17:23 +00:00
|
|
|
|
|
|
|
// setup process environment for the forked Ghost to use the new config file
|
|
|
|
var env = _.clone(process.env);
|
|
|
|
env['GHOST_CONFIG'] = newConfigFile;
|
|
|
|
env['NODE_ENV'] = envName;
|
|
|
|
var child = cp.fork(path.join(config.paths.appRoot, 'index.js'), {env: env});
|
|
|
|
|
|
|
|
var pingTries = 0;
|
|
|
|
var pingCheck;
|
|
|
|
var pingStop = function() {
|
|
|
|
if (pingCheck) {
|
|
|
|
clearInterval(pingCheck);
|
|
|
|
pingCheck = undefined;
|
|
|
|
return true;
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
2014-08-17 06:17:23 +00:00
|
|
|
return false;
|
|
|
|
};
|
|
|
|
// periodic check until forked Ghost is running and is listening on the port
|
|
|
|
pingCheck = setInterval(function() {
|
|
|
|
var socket = net.connect(port);
|
|
|
|
socket.on('connect', function() {
|
|
|
|
socket.end();
|
|
|
|
if (pingStop()) {
|
|
|
|
resolve(child);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
socket.on('error', function(err) {
|
|
|
|
// continue checking
|
|
|
|
if (++pingTries >= 20 && pingStop()) {
|
|
|
|
reject(new Error("Timed out waiting for child process"));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, 200);
|
|
|
|
|
|
|
|
child.on('exit', function(code, signal) {
|
|
|
|
child.exited = true;
|
|
|
|
if (pingStop()) {
|
|
|
|
reject(new Error("Child process exit code: " + code));
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
2014-08-17 06:17:23 +00:00
|
|
|
// cleanup the temporary config file
|
|
|
|
fs.unlink(newConfigFile);
|
2014-02-21 20:25:31 -05:00
|
|
|
});
|
|
|
|
|
2014-08-17 06:17:23 +00:00
|
|
|
// override kill() to have an async callback
|
|
|
|
var baseKill = child.kill;
|
|
|
|
child.kill = function(signal, cb) {
|
|
|
|
if (typeof signal === 'function') {
|
|
|
|
cb = signal;
|
|
|
|
signal = undefined;
|
|
|
|
}
|
2014-02-21 20:25:31 -05:00
|
|
|
|
2014-08-17 06:17:23 +00:00
|
|
|
if (cb) {
|
|
|
|
child.on('exit', function() {
|
|
|
|
cb();
|
|
|
|
});
|
|
|
|
}
|
2014-02-21 20:25:31 -05:00
|
|
|
|
2014-08-17 06:17:23 +00:00
|
|
|
if (child.exited) {
|
|
|
|
process.nextTick(cb);
|
|
|
|
} else {
|
|
|
|
baseKill.apply(child, [signal]);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2014-02-21 20:25:31 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports.ghost = forkGhost;
|
|
|
|
module.exports.config = forkConfig;
|