mirror of
https://github.com/meteor/meteor.git
synced 2026-05-02 03:01:46 -04:00
Velocity has visited the mirror sometimes before it is ready. With this new check that is already used in the testing frameworks this no longer happens. This commit also handles the upcoming breaking change regarding the rootUrl of the mirror. See: https://github.com/meteor-velocity/velocity/issues/260
183 lines
6.3 KiB
JavaScript
183 lines
6.3 KiB
JavaScript
var Console = require('./console.js').Console;
|
|
var isopackets = require("./isopackets.js");
|
|
|
|
var phantomjs = require('phantomjs');
|
|
var child_process = require('child_process');
|
|
var _ = require('underscore');
|
|
|
|
// XXX this could really use a self-test!
|
|
|
|
// XXX would be nice be nice if this didn't have to be in core. Perhaps
|
|
// at some point we'll have an API for packages to register commands in
|
|
// the tool.
|
|
|
|
// 1. Establish a DDP connection to Meteor
|
|
// 2. Subscribe to the Velocity subscriptions that tell us
|
|
// which tests pass/fail and when all tests have completed.
|
|
// 3. Open the app server with PhantomJS to run client side tests.
|
|
// 4. Print the results and exit with the appropriate exit code.
|
|
var runVelocity = function (url) {
|
|
var unipackages = isopackets.load('ddp');
|
|
var DDP = unipackages.ddp.DDP;
|
|
|
|
// XXX maybe a startup message so users know the tests are running.
|
|
|
|
// All running browser processes that visit the mirror pages
|
|
var browserProcesses = [];
|
|
// Maps mirror id to url
|
|
var mirrorUrls = {};
|
|
var ddpConnection = DDP.connect(url);
|
|
|
|
var killBrowserProcesses = function () {
|
|
browserProcesses.forEach(function (browserProcess) {
|
|
browserProcess.kill('SIGINT');
|
|
});
|
|
browserProcesses = [];
|
|
};
|
|
|
|
var interval = setInterval(function () {
|
|
if (ddpConnection.status().status === "connected") {
|
|
clearInterval(interval);
|
|
|
|
ddpConnection.subscribe("VelocityTestReports", {
|
|
onError: function () {
|
|
Console.error("failed to subscribe to VelocityTestReports " +
|
|
"subscription");
|
|
// XXX tell user to add velocity:core
|
|
// XXX these also fire if the user turns on autopublish
|
|
}, onReady: function () {
|
|
this.connection.registerStore("velocityTestReports", {
|
|
update: function (msg) {
|
|
if (msg.msg === "added") {
|
|
var testDesc = msg.fields.framework + " : " +
|
|
msg.fields.ancestors.join(":") + " => " + msg.fields.name;
|
|
if (msg.fields.result === "passed") {
|
|
console.log("PASSED", testDesc);
|
|
} else if (msg.fields.result === "failed") {
|
|
console.error("FAILED", testDesc);
|
|
console.log(msg.fields.failureStackTrace);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
var reports = {};
|
|
function updateReport(msg) {
|
|
var report = reports[msg.id];
|
|
if (! report) {
|
|
reports[msg.id] = msg.fields;
|
|
} else {
|
|
_.extend(report, msg.fields);
|
|
}
|
|
}
|
|
var aggregateResult = null;
|
|
var isFinished = false;
|
|
ddpConnection.subscribe("VelocityAggregateReports", {
|
|
onError: function () {
|
|
Console.error("failed to subscribe to " +
|
|
"VelocityAggregateReports subscription");
|
|
}, onReady: function () {
|
|
this.connection.registerStore("velocityAggregateReports", {
|
|
update: function (msg) {
|
|
if (msg.msg === "added" || msg.msg === "changed") {
|
|
updateReport(msg);
|
|
var report = reports[msg.id];
|
|
|
|
if (report.name === "aggregateResult") {
|
|
aggregateResult = report.result;
|
|
}
|
|
|
|
if (report.name === "aggregateComplete" &&
|
|
report.result === "completed") {
|
|
setTimeout(function () {
|
|
killBrowserProcesses();
|
|
if (aggregateResult === "passed") {
|
|
console.log("TESTS RAN SUCCESSFULLY");
|
|
// XXX XXX this is not great. We shouldn't be
|
|
// exiting from deep within code like this. Better
|
|
// would be to integrate with run --once, and
|
|
// signal the inner process to exit cleanly on
|
|
// test completion.
|
|
process.exit(0);
|
|
}
|
|
if (aggregateResult === "failed") {
|
|
console.log("FAILURE");
|
|
process.exit(1);
|
|
}
|
|
}, 2000);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
function visitWithPhantom (url) {
|
|
var phantomScript = "require('webpage').create().open('" + url + "');";
|
|
var browserProcess = child_process.execFile(
|
|
'/bin/bash',
|
|
['-c',
|
|
("exec " + phantomjs.path + " /dev/stdin <<'END'\n" +
|
|
phantomScript + "END\n")]);
|
|
browserProcesses.push(browserProcess);
|
|
}
|
|
|
|
ddpConnection.subscribe("VelocityMirrors", {
|
|
onError: function (err) {
|
|
Console.error("failed to subscribe to VelocityMirrors " +
|
|
"subscription", err);
|
|
}, onReady: function () {
|
|
this.connection.registerStore("velocityMirrors", {
|
|
update: function (msg) {
|
|
if (isMirrorUrlMessage(msg)) {
|
|
mirrorUrls[msg.id] = generateMirrorUrl(
|
|
msg.fields.rootUrl,
|
|
msg.fields.rootUrlPath
|
|
);
|
|
}
|
|
if (isMirrorReadyMessage(msg)) {
|
|
var mirrorUrl = mirrorUrls[msg.id];
|
|
if (mirrorUrl) {
|
|
visitWithPhantom(mirrorUrl);
|
|
} else {
|
|
Console.error(
|
|
"Could not find URL of mirror " +
|
|
"with the MongoDB ID " + msg.id
|
|
);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|
|
});
|
|
|
|
function isUpdateMessage(msg) {
|
|
return msg.msg === "added" || msg.msg === "changed";
|
|
}
|
|
|
|
function isMirrorUrlMessage(msg) {
|
|
return isUpdateMessage(msg) && msg.fields.rootUrl;
|
|
}
|
|
|
|
function isMirrorReadyMessage(msg) {
|
|
return isUpdateMessage(msg) && msg.fields.state === "ready";
|
|
}
|
|
|
|
function generateMirrorUrl(rootUrl, rootUrlPath) {
|
|
var mirrorUrl = rootUrl;
|
|
// Handle the breaking change in velocity:core 0.5
|
|
// See: https://github.com/meteor-velocity/velocity/issues/260
|
|
if (rootUrlPath && mirrorUrl.indexOf(rootUrlPath) === -1) {
|
|
mirrorUrl += rootUrlPath;
|
|
}
|
|
|
|
return mirrorUrl;
|
|
}
|
|
}
|
|
}, 2000);
|
|
};
|
|
|
|
exports.runVelocity = runVelocity;
|