mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-11-09 12:51:09 +02:00
217 lines
6.4 KiB
JavaScript
217 lines
6.4 KiB
JavaScript
"use strict";
|
|
|
|
const { ExperimentFakes } = ChromeUtils.import(
|
|
"resource://testing-common/NimbusTestUtils.jsm"
|
|
);
|
|
const { CleanupManager } = ChromeUtils.import(
|
|
"resource://normandy/lib/CleanupManager.jsm"
|
|
);
|
|
|
|
const { ExperimentManager } = ChromeUtils.import(
|
|
"resource://nimbus/lib/ExperimentManager.jsm"
|
|
);
|
|
|
|
const {
|
|
RemoteSettingsExperimentLoader,
|
|
RemoteDefaultsLoader,
|
|
} = ChromeUtils.import(
|
|
"resource://nimbus/lib/RemoteSettingsExperimentLoader.jsm"
|
|
);
|
|
|
|
const ENABLED_PREF = "messaging-system.rsexperimentloader.enabled";
|
|
const RUN_INTERVAL_PREF = "app.normandy.run_interval_seconds";
|
|
const STUDIES_OPT_OUT_PREF = "app.shield.optoutstudies.enabled";
|
|
|
|
add_task(async function test_real_exp_manager() {
|
|
equal(
|
|
RemoteSettingsExperimentLoader.manager,
|
|
ExperimentManager,
|
|
"should reference ExperimentManager singleton by default"
|
|
);
|
|
});
|
|
|
|
add_task(async function test_lazy_pref_getters() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
sinon.stub(loader, "updateRecipes").resolves();
|
|
|
|
Services.prefs.setIntPref(RUN_INTERVAL_PREF, 123456);
|
|
equal(
|
|
loader.intervalInSeconds,
|
|
123456,
|
|
`should set intervalInSeconds to the value of ${RUN_INTERVAL_PREF}`
|
|
);
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
equal(
|
|
loader.enabled,
|
|
true,
|
|
`should set enabled to the value of ${ENABLED_PREF}`
|
|
);
|
|
Services.prefs.setBoolPref(ENABLED_PREF, false);
|
|
equal(loader.enabled, false);
|
|
|
|
Services.prefs.clearUserPref(RUN_INTERVAL_PREF);
|
|
Services.prefs.clearUserPref(ENABLED_PREF);
|
|
});
|
|
|
|
add_task(async function test_init() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
sinon.stub(loader, "setTimer");
|
|
sinon.stub(loader, "updateRecipes").resolves();
|
|
sinon.stub(RemoteDefaultsLoader, "syncRemoteDefaults");
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, false);
|
|
await loader.init();
|
|
equal(
|
|
loader.setTimer.callCount,
|
|
0,
|
|
`should not initialize if ${ENABLED_PREF} pref is false`
|
|
);
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
await loader.init();
|
|
ok(loader.setTimer.calledOnce, "should call .setTimer");
|
|
ok(loader.updateRecipes.calledOnce, "should call .updatpickeRecipes");
|
|
ok(
|
|
RemoteDefaultsLoader.syncRemoteDefaults,
|
|
"initialized remote defaults loader"
|
|
);
|
|
});
|
|
|
|
add_task(async function test_init_with_opt_in() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
sinon.stub(loader, "setTimer");
|
|
sinon.stub(loader, "updateRecipes").resolves();
|
|
|
|
Services.prefs.setBoolPref(STUDIES_OPT_OUT_PREF, false);
|
|
await loader.init();
|
|
equal(
|
|
loader.setTimer.callCount,
|
|
0,
|
|
`should not initialize if ${STUDIES_OPT_OUT_PREF} pref is false`
|
|
);
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, false);
|
|
await loader.init();
|
|
equal(
|
|
loader.setTimer.callCount,
|
|
0,
|
|
`should not initialize if ${ENABLED_PREF} pref is false`
|
|
);
|
|
|
|
Services.prefs.setBoolPref(STUDIES_OPT_OUT_PREF, true);
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
await loader.init();
|
|
ok(loader.setTimer.calledOnce, "should call .setTimer");
|
|
ok(loader.updateRecipes.calledOnce, "should call .updateRecipes");
|
|
});
|
|
|
|
add_task(async function test_updateRecipes() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
|
|
const PASS_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting: "true",
|
|
});
|
|
const FAIL_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting: "false",
|
|
});
|
|
sinon.stub(loader, "setTimer");
|
|
sinon.spy(loader, "updateRecipes");
|
|
|
|
sinon
|
|
.stub(loader.remoteSettingsClient, "get")
|
|
.resolves([PASS_FILTER_RECIPE, FAIL_FILTER_RECIPE]);
|
|
sinon.stub(loader.manager, "onRecipe").resolves();
|
|
sinon.stub(loader.manager, "onFinalize");
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
await loader.init();
|
|
ok(loader.updateRecipes.calledOnce, "should call .updateRecipes");
|
|
equal(
|
|
loader.manager.onRecipe.callCount,
|
|
1,
|
|
"should call .onRecipe only for recipes that pass"
|
|
);
|
|
ok(
|
|
loader.manager.onRecipe.calledWith(PASS_FILTER_RECIPE, "rs-loader"),
|
|
"should call .onRecipe with argument data"
|
|
);
|
|
});
|
|
|
|
add_task(async function test_updateRecipes_forFirstStartup() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
const PASS_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting: "isFirstStartup",
|
|
});
|
|
sinon.stub(loader.remoteSettingsClient, "get").resolves([PASS_FILTER_RECIPE]);
|
|
sinon.stub(loader.manager, "onRecipe").resolves();
|
|
sinon.stub(loader.manager, "onFinalize");
|
|
sinon
|
|
.stub(loader.manager, "createTargetingContext")
|
|
.returns({ isFirstStartup: true });
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
await loader.init({ isFirstStartup: true });
|
|
|
|
ok(loader.manager.onRecipe.calledOnce, "should pass the targeting filter");
|
|
});
|
|
|
|
add_task(async function test_updateRecipes_forNoneFirstStartup() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
const PASS_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting: "isFirstStartup",
|
|
});
|
|
sinon.stub(loader.remoteSettingsClient, "get").resolves([PASS_FILTER_RECIPE]);
|
|
sinon.stub(loader.manager, "onRecipe").resolves();
|
|
sinon.stub(loader.manager, "onFinalize");
|
|
sinon
|
|
.stub(loader.manager, "createTargetingContext")
|
|
.returns({ isFirstStartup: false });
|
|
|
|
Services.prefs.setBoolPref(ENABLED_PREF, true);
|
|
await loader.init({ isFirstStartup: true });
|
|
|
|
ok(loader.manager.onRecipe.notCalled, "should not pass the targeting filter");
|
|
});
|
|
|
|
add_task(async function test_checkTargeting() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
equal(
|
|
await loader.checkTargeting({}),
|
|
true,
|
|
"should return true if .targeting is not defined"
|
|
);
|
|
equal(
|
|
await loader.checkTargeting({ targeting: "'foo'" }),
|
|
true,
|
|
"should return true for truthy expression"
|
|
);
|
|
equal(
|
|
await loader.checkTargeting({ targeting: "aPropertyThatDoesNotExist" }),
|
|
false,
|
|
"should return false for falsey expression"
|
|
);
|
|
});
|
|
|
|
add_task(async function test_checkExperimentSelfReference() {
|
|
const loader = ExperimentFakes.rsLoader();
|
|
const PASS_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting:
|
|
"experiment.slug == 'foo' && experiment.branches[0].slug == 'control'",
|
|
});
|
|
|
|
const FAIL_FILTER_RECIPE = ExperimentFakes.recipe("foo", {
|
|
targeting: "experiment.slug == 'bar'",
|
|
});
|
|
|
|
equal(
|
|
await loader.checkTargeting(PASS_FILTER_RECIPE),
|
|
true,
|
|
"Should return true for matching on slug name and branch"
|
|
);
|
|
equal(
|
|
await loader.checkTargeting(FAIL_FILTER_RECIPE),
|
|
false,
|
|
"Should fail targeting"
|
|
);
|
|
});
|