"use strict"; const { ExperimentFakes } = ChromeUtils.import( "resource://testing-common/MSTestUtils.jsm" ); const { CleanupManager } = ChromeUtils.import( "resource://normandy/lib/CleanupManager.jsm" ); const { ExperimentManager } = ChromeUtils.import( "resource://messaging-system/experiments/ExperimentManager.jsm" ); const { RemoteSettingsExperimentLoader } = ChromeUtils.import( "resource://messaging-system/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(); 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 .updateRecipes"); }); 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" ); });