forked from mirrors/gecko-dev
MozReview-Commit-ID: EWo7Gv9oSsz --HG-- extra : rebase_source : e83863f0ef55f04be32a5062d0f672c8a3d88cb5
314 lines
11 KiB
JavaScript
314 lines
11 KiB
JavaScript
/* Any copyright is dedicated to the Public Domain.
|
|
* http://creativecommons.org/publicdomain/zero/1.0/ */
|
|
|
|
"use strict";
|
|
|
|
Cu.import("resource://testing-common/httpd.js");
|
|
Cu.import("resource://gre/modules/TelemetryLog.jsm");
|
|
var bsp = Cu.import("resource:///modules/experiments/Experiments.jsm");
|
|
|
|
|
|
const FILE_MANIFEST = "experiments.manifest";
|
|
const MANIFEST_HANDLER = "manifests/handler";
|
|
|
|
const SEC_IN_ONE_DAY = 24 * 60 * 60;
|
|
const MS_IN_ONE_DAY = SEC_IN_ONE_DAY * 1000;
|
|
|
|
|
|
var gProfileDir = null;
|
|
var gHttpServer = null;
|
|
var gHttpRoot = null;
|
|
var gDataRoot = null;
|
|
var gPolicy = null;
|
|
var gManifestObject = null;
|
|
var gManifestHandlerURI = null;
|
|
|
|
const TLOG = bsp.TELEMETRY_LOG;
|
|
|
|
function checkEvent(event, id, data)
|
|
{
|
|
do_print("Checking message " + id);
|
|
Assert.equal(event[0], id, "id should match");
|
|
Assert.ok(event[1] > 0, "timestamp should be greater than 0");
|
|
|
|
if (data === undefined) {
|
|
Assert.equal(event.length, 2, "event array should have 2 entries");
|
|
} else {
|
|
Assert.equal(event.length, data.length + 2, "event entry count should match expected count");
|
|
for (var i = 0; i < data.length; ++i) {
|
|
Assert.equal(typeof(event[i + 2]), "string", "event entry should be a string");
|
|
Assert.equal(event[i + 2], data[i], "event entry should match expected entry");
|
|
}
|
|
}
|
|
}
|
|
|
|
function run_test() {
|
|
run_next_test();
|
|
}
|
|
|
|
add_task(function* test_setup() {
|
|
loadAddonManager();
|
|
gProfileDir = do_get_profile();
|
|
|
|
gHttpServer = new HttpServer();
|
|
gHttpServer.start(-1);
|
|
let port = gHttpServer.identity.primaryPort;
|
|
gHttpRoot = "http://localhost:" + port + "/";
|
|
gDataRoot = gHttpRoot + "data/";
|
|
gManifestHandlerURI = gHttpRoot + MANIFEST_HANDLER;
|
|
gHttpServer.registerDirectory("/data/", do_get_cwd());
|
|
gHttpServer.registerPathHandler("/" + MANIFEST_HANDLER, (request, response) => {
|
|
response.setStatusLine(null, 200, "OK");
|
|
response.write(JSON.stringify(gManifestObject));
|
|
response.processAsync();
|
|
response.finish();
|
|
});
|
|
do_register_cleanup(() => gHttpServer.stop(() => {}));
|
|
|
|
Services.prefs.setBoolPref(PREF_EXPERIMENTS_ENABLED, true);
|
|
Services.prefs.setIntPref(PREF_LOGGING_LEVEL, 0);
|
|
Services.prefs.setBoolPref(PREF_LOGGING_DUMP, true);
|
|
Services.prefs.setCharPref(PREF_MANIFEST_URI, gManifestHandlerURI);
|
|
Services.prefs.setIntPref(PREF_FETCHINTERVAL, 0);
|
|
|
|
gPolicy = new Experiments.Policy();
|
|
let dummyTimer = { cancel: () => {}, clear: () => {} };
|
|
patchPolicy(gPolicy, {
|
|
updatechannel: () => "nightly",
|
|
oneshotTimer: (callback, timeout, thisObj, name) => dummyTimer,
|
|
});
|
|
|
|
yield removeCacheFile();
|
|
});
|
|
|
|
// Test basic starting and stopping of experiments.
|
|
|
|
add_task(function* test_telemetryBasics() {
|
|
// Check TelemetryLog instead of TelemetrySession.getPayload().log because
|
|
// TelemetrySession gets Experiments.instance() and side-effects log entries.
|
|
|
|
const OBSERVER_TOPIC = "experiments-changed";
|
|
let observerFireCount = 0;
|
|
let expectedLogLength = 0;
|
|
|
|
// Dates the following tests are based on.
|
|
|
|
let baseDate = new Date(2014, 5, 1, 12);
|
|
let startDate1 = futureDate(baseDate, 50 * MS_IN_ONE_DAY);
|
|
let endDate1 = futureDate(baseDate, 100 * MS_IN_ONE_DAY);
|
|
let startDate2 = futureDate(baseDate, 150 * MS_IN_ONE_DAY);
|
|
let endDate2 = futureDate(baseDate, 200 * MS_IN_ONE_DAY);
|
|
|
|
// The manifest data we test with.
|
|
|
|
gManifestObject = {
|
|
"version": 1,
|
|
experiments: [
|
|
{
|
|
id: EXPERIMENT1_ID,
|
|
xpiURL: gDataRoot + EXPERIMENT1_XPI_NAME,
|
|
xpiHash: EXPERIMENT1_XPI_SHA1,
|
|
startTime: dateToSeconds(startDate1),
|
|
endTime: dateToSeconds(endDate1),
|
|
maxActiveSeconds: 10 * SEC_IN_ONE_DAY,
|
|
appName: ["XPCShell"],
|
|
channel: ["nightly"],
|
|
},
|
|
{
|
|
id: EXPERIMENT2_ID,
|
|
xpiURL: gDataRoot + EXPERIMENT2_XPI_NAME,
|
|
xpiHash: EXPERIMENT2_XPI_SHA1,
|
|
startTime: dateToSeconds(startDate2),
|
|
endTime: dateToSeconds(endDate2),
|
|
maxActiveSeconds: 10 * SEC_IN_ONE_DAY,
|
|
appName: ["XPCShell"],
|
|
channel: ["nightly"],
|
|
},
|
|
],
|
|
};
|
|
|
|
// Data to compare the result of Experiments.getExperiments() against.
|
|
|
|
let experimentListData = [
|
|
{
|
|
id: EXPERIMENT2_ID,
|
|
name: "Test experiment 2",
|
|
description: "And yet another experiment that experiments experimentally.",
|
|
},
|
|
{
|
|
id: EXPERIMENT1_ID,
|
|
name: EXPERIMENT1_NAME,
|
|
description: "Yet another experiment that experiments experimentally.",
|
|
},
|
|
];
|
|
|
|
let experiments = new Experiments.Experiments(gPolicy);
|
|
|
|
// Trigger update, clock set to before any activation.
|
|
// Use updateManifest() to provide for coverage of that path.
|
|
|
|
let now = baseDate;
|
|
defineNow(gPolicy, now);
|
|
|
|
yield experiments.updateManifest();
|
|
let list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 0, "Experiment list should be empty.");
|
|
|
|
expectedLogLength += 2;
|
|
let log = TelemetryLog.entries();
|
|
do_print("Telemetry log: " + JSON.stringify(log));
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-2], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.REJECTED, EXPERIMENT1_ID, "startTime"]);
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.REJECTED, EXPERIMENT2_ID, "startTime"]);
|
|
|
|
// Trigger update, clock set for experiment 1 to start.
|
|
|
|
now = futureDate(startDate1, 5 * MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 1, "Experiment list should have 1 entry now.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries. Got " + log.toSource());
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.ACTIVATED, EXPERIMENT1_ID]);
|
|
|
|
// Trigger update, clock set for experiment 1 to stop.
|
|
|
|
now = futureDate(endDate1, 1000);
|
|
defineNow(gPolicy, now);
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 1, "Experiment list should have 1 entry.");
|
|
|
|
expectedLogLength += 2;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-2], TLOG.TERMINATION_KEY,
|
|
[TLOG.TERMINATION.EXPIRED, EXPERIMENT1_ID]);
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.REJECTED, EXPERIMENT2_ID, "startTime"]);
|
|
|
|
// Trigger update, clock set for experiment 2 to start with invalid hash.
|
|
|
|
now = startDate2;
|
|
defineNow(gPolicy, now);
|
|
gManifestObject.experiments[1].xpiHash = "sha1:0000000000000000000000000000000000000000";
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 1, "Experiment list should have 1 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.INSTALL_FAILURE, EXPERIMENT2_ID]);
|
|
|
|
// Trigger update, clock set for experiment 2 to properly start now.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
gManifestObject.experiments[1].xpiHash = EXPERIMENT2_XPI_SHA1;
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 2, "Experiment list should have 2 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.ACTIVATED, EXPERIMENT2_ID]);
|
|
|
|
// Fake user uninstall of experiment via add-on manager.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
|
|
yield experiments.disableExperiment(TLOG.TERMINATION.ADDON_UNINSTALLED);
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 2, "Experiment list should have 2 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.TERMINATION_KEY,
|
|
[TLOG.TERMINATION.ADDON_UNINSTALLED, EXPERIMENT2_ID]);
|
|
|
|
// Trigger update with experiment 1a ready to start.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
gManifestObject.experiments[0].id = EXPERIMENT3_ID;
|
|
gManifestObject.experiments[0].endTime = dateToSeconds(futureDate(now, 50 * MS_IN_ONE_DAY));
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 3, "Experiment list should have 3 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.ACTIVATED, EXPERIMENT3_ID]);
|
|
|
|
// Trigger disable of an experiment via the API.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
|
|
yield experiments.disableExperiment(TLOG.TERMINATION.FROM_API);
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 3, "Experiment list should have 3 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.TERMINATION_KEY,
|
|
[TLOG.TERMINATION.FROM_API, EXPERIMENT3_ID]);
|
|
|
|
// Trigger update with experiment 1a ready to start.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
gManifestObject.experiments[0].id = EXPERIMENT4_ID;
|
|
gManifestObject.experiments[0].endTime = dateToSeconds(futureDate(now, 50 * MS_IN_ONE_DAY));
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 4, "Experiment list should have 4 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.ACTIVATION_KEY,
|
|
[TLOG.ACTIVATION.ACTIVATED, EXPERIMENT4_ID]);
|
|
|
|
// Trigger experiment termination by something other than expiry via the manifest.
|
|
|
|
now = futureDate(now, MS_IN_ONE_DAY);
|
|
defineNow(gPolicy, now);
|
|
gManifestObject.experiments[0].os = "Plan9";
|
|
|
|
yield experiments.updateManifest();
|
|
list = yield experiments.getExperiments();
|
|
Assert.equal(list.length, 4, "Experiment list should have 4 entries.");
|
|
|
|
expectedLogLength += 1;
|
|
log = TelemetryLog.entries();
|
|
Assert.equal(log.length, expectedLogLength, "Telemetry log should have " + expectedLogLength + " entries.");
|
|
checkEvent(log[log.length-1], TLOG.TERMINATION_KEY,
|
|
[TLOG.TERMINATION.RECHECK, EXPERIMENT4_ID, "os"]);
|
|
|
|
// Cleanup.
|
|
|
|
yield promiseRestartManager();
|
|
yield removeCacheFile();
|
|
});
|