fune/browser/components/newtab/test/unit/asrouter/ASRouter.test.js
Mike Conley 9415a9f13d Bug 1866802 - Move ASRouterAdmin tool to about:asrouter and its own component folder. r=pdahiya,Gijs,desktop-theme-reviewers,dao
This tries to maintain stylistic continuity, while also trying to decouple from
newtab as much as possible. This is a first foray, and future patches will
further this decoupling.

This also modifies about:asrouter to show an error message if the ASRouter devtools
pref is not set to true.

Differential Revision: https://phabricator.services.mozilla.com/D194811
2023-12-14 18:46:55 +00:00

2870 lines
95 KiB
JavaScript

import { _ASRouter, MessageLoaderUtils } from "lib/ASRouter.jsm";
import { QueryCache } from "lib/ASRouterTargeting.jsm";
import {
FAKE_LOCAL_MESSAGES,
FAKE_LOCAL_PROVIDER,
FAKE_LOCAL_PROVIDERS,
FAKE_REMOTE_MESSAGES,
FAKE_REMOTE_PROVIDER,
FAKE_REMOTE_SETTINGS_PROVIDER,
} from "./constants";
import {
ASRouterPreferences,
TARGETING_PREFERENCES,
} from "lib/ASRouterPreferences.jsm";
import { ASRouterTriggerListeners } from "lib/ASRouterTriggerListeners.jsm";
import { CFRPageActions } from "lib/CFRPageActions.jsm";
import { GlobalOverrider } from "test/unit/utils";
import { PanelTestProvider } from "lib/PanelTestProvider.sys.mjs";
import ProviderResponseSchema from "content-src/asrouter/schemas/provider-response.schema.json";
const MESSAGE_PROVIDER_PREF_NAME =
"browser.newtabpage.activity-stream.asrouter.providers.cfr";
const FAKE_PROVIDERS = [
FAKE_LOCAL_PROVIDER,
FAKE_REMOTE_PROVIDER,
FAKE_REMOTE_SETTINGS_PROVIDER,
];
const ONE_DAY_IN_MS = 24 * 60 * 60 * 1000;
const FAKE_RESPONSE_HEADERS = { get() {} };
const FAKE_BUNDLE = [FAKE_LOCAL_MESSAGES[1], FAKE_LOCAL_MESSAGES[2]];
const USE_REMOTE_L10N_PREF =
"browser.newtabpage.activity-stream.asrouter.useRemoteL10n";
// eslint-disable-next-line max-statements
describe("ASRouter", () => {
let Router;
let globals;
let sandbox;
let initParams;
let messageBlockList;
let providerBlockList;
let messageImpressions;
let groupImpressions;
let previousSessionEnd;
let fetchStub;
let clock;
let fakeAttributionCode;
let fakeTargetingContext;
let FakeToolbarBadgeHub;
let FakeToolbarPanelHub;
let FakeMomentsPageHub;
let ASRouterTargeting;
let screenImpressions;
function setMessageProviderPref(value) {
sandbox.stub(ASRouterPreferences, "providers").get(() => value);
}
function initASRouter(router) {
const getStub = sandbox.stub();
getStub.returns(Promise.resolve());
getStub
.withArgs("messageBlockList")
.returns(Promise.resolve(messageBlockList));
getStub
.withArgs("providerBlockList")
.returns(Promise.resolve(providerBlockList));
getStub
.withArgs("messageImpressions")
.returns(Promise.resolve(messageImpressions));
getStub.withArgs("groupImpressions").resolves(groupImpressions);
getStub
.withArgs("previousSessionEnd")
.returns(Promise.resolve(previousSessionEnd));
getStub
.withArgs("screenImpressions")
.returns(Promise.resolve(screenImpressions));
initParams = {
storage: {
get: getStub,
set: sandbox.stub().returns(Promise.resolve()),
},
sendTelemetry: sandbox.stub().resolves(),
clearChildMessages: sandbox.stub().resolves(),
clearChildProviders: sandbox.stub().resolves(),
updateAdminState: sandbox.stub().resolves(),
dispatchCFRAction: sandbox.stub().resolves(),
};
sandbox.stub(router, "loadMessagesFromAllProviders").callThrough();
return router.init(initParams);
}
async function createRouterAndInit(providers = FAKE_PROVIDERS) {
setMessageProviderPref(providers);
// `.freeze` to catch any attempts at modifying the object
Router = new _ASRouter(Object.freeze(FAKE_LOCAL_PROVIDERS));
await initASRouter(Router);
}
beforeEach(async () => {
globals = new GlobalOverrider();
messageBlockList = [];
providerBlockList = [];
messageImpressions = {};
groupImpressions = {};
previousSessionEnd = 100;
screenImpressions = {};
sandbox = sinon.createSandbox();
ASRouterTargeting = {
isMatch: sandbox.stub(),
findMatchingMessage: sandbox.stub(),
Environment: {
locale: "en-US",
localeLanguageCode: "en",
browserSettings: {
update: {
channel: "default",
enabled: true,
autoDownload: true,
},
},
attributionData: {},
currentDate: "2000-01-01T10:00:0.001Z",
profileAgeCreated: {},
profileAgeReset: {},
usesFirefoxSync: false,
isFxAEnabled: true,
isFxASignedIn: false,
sync: {
desktopDevices: 0,
mobileDevices: 0,
totalDevices: 0,
},
xpinstallEnabled: true,
addonsInfo: {},
searchEngines: {},
isDefaultBrowser: false,
devToolsOpenedCount: 5,
topFrecentSites: {},
recentBookmarks: {},
pinnedSites: [
{
url: "https://amazon.com",
host: "amazon.com",
searchTopSite: true,
},
],
providerCohorts: {
onboarding: "",
cfr: "",
"message-groups": "",
"messaging-experiments": "",
"whats-new-panel": "",
},
totalBookmarksCount: {},
firefoxVersion: 80,
region: "US",
needsUpdate: {},
hasPinnedTabs: false,
hasAccessedFxAPanel: false,
isWhatsNewPanelEnabled: true,
userPrefs: {
cfrFeatures: true,
cfrAddons: true,
},
totalBlockedCount: {},
blockedCountByType: {},
attachedFxAOAuthClients: [],
platformName: "macosx",
scores: {},
scoreThreshold: 5000,
isChinaRepack: false,
userId: "adsf",
},
};
ASRouterPreferences.specialConditions = {
someCondition: true,
};
sandbox.spy(ASRouterPreferences, "init");
sandbox.spy(ASRouterPreferences, "uninit");
sandbox.spy(ASRouterPreferences, "addListener");
sandbox.spy(ASRouterPreferences, "removeListener");
clock = sandbox.useFakeTimers();
fetchStub = sandbox
.stub(global, "fetch")
.withArgs("http://fake.com/endpoint")
.resolves({
ok: true,
status: 200,
json: () => Promise.resolve({ messages: FAKE_REMOTE_MESSAGES }),
headers: FAKE_RESPONSE_HEADERS,
});
sandbox.stub(global.Services.prefs, "getStringPref");
fakeAttributionCode = {
allowedCodeKeys: ["foo", "bar", "baz"],
_clearCache: () => sinon.stub(),
getAttrDataAsync: () => Promise.resolve({ content: "addonID" }),
deleteFileAsync: () => Promise.resolve(),
writeAttributionFile: () => Promise.resolve(),
getCachedAttributionData: sinon.stub(),
};
FakeToolbarPanelHub = {
init: sandbox.stub(),
uninit: sandbox.stub(),
forceShowMessage: sandbox.stub(),
enableToolbarButton: sandbox.stub(),
};
FakeToolbarBadgeHub = {
init: sandbox.stub(),
uninit: sandbox.stub(),
registerBadgeNotificationListener: sandbox.stub(),
};
FakeMomentsPageHub = {
init: sandbox.stub(),
uninit: sandbox.stub(),
executeAction: sandbox.stub(),
};
fakeTargetingContext = {
combineContexts: sandbox.stub(),
evalWithDefault: sandbox.stub().resolves(),
};
let fakeNimbusFeatures = [
"cfr",
"infobar",
"spotlight",
"moments-page",
"pbNewtab",
].reduce((features, featureId) => {
features[featureId] = {
getAllVariables: sandbox.stub().returns(null),
recordExposureEvent: sandbox.stub(),
};
return features;
}, {});
globals.set({
// Testing framework doesn't know how to `defineLazyModuleGetters` so we're
// importing these modules into the global scope ourselves.
GroupsConfigurationProvider: { getMessages: () => Promise.resolve([]) },
ASRouterPreferences,
TARGETING_PREFERENCES,
ASRouterTargeting,
ASRouterTriggerListeners,
QueryCache,
gBrowser: { selectedBrowser: {} },
gURLBar: {},
isSeparateAboutWelcome: true,
AttributionCode: fakeAttributionCode,
PanelTestProvider,
MacAttribution: { applicationPath: "" },
ToolbarBadgeHub: FakeToolbarBadgeHub,
ToolbarPanelHub: FakeToolbarPanelHub,
MomentsPageHub: FakeMomentsPageHub,
KintoHttpClient: class {
bucket() {
return this;
}
collection() {
return this;
}
getRecord() {
return Promise.resolve({ data: {} });
}
},
Downloader: class {
download() {
return Promise.resolve("/path/to/download");
}
},
NimbusFeatures: fakeNimbusFeatures,
ExperimentAPI: {
getExperimentMetaData: sandbox.stub().returns({
slug: "experiment-slug",
active: true,
branch: { slug: "experiment-branch-slug" },
}),
getExperiment: sandbox.stub().returns({
branch: {
slug: "unit-slug",
feature: {
featureId: "foo",
value: { id: "test-message" },
},
},
}),
getAllBranches: sandbox.stub().resolves([]),
ready: sandbox.stub().resolves(),
},
SpecialMessageActions: {
handleAction: sandbox.stub(),
},
TargetingContext: class {
static combineContexts(...args) {
return fakeTargetingContext.combineContexts.apply(sandbox, args);
}
evalWithDefault(expr) {
return fakeTargetingContext.evalWithDefault(expr);
}
},
RemoteL10n: {
// This is just a subset of supported locales that happen to be used in
// the test.
isLocaleSupported: locale => ["en-US", "ja-JP-mac"].includes(locale),
},
});
await createRouterAndInit();
});
afterEach(() => {
Router.uninit();
ASRouterPreferences.uninit();
sandbox.restore();
globals.restore();
});
describe(".state", () => {
it("should throw if an attempt to set .state was made", () => {
assert.throws(() => {
Router.state = {};
});
});
});
describe("#init", () => {
it("should only be called once", async () => {
Router = new _ASRouter();
let state = await initASRouter(Router);
assert.equal(state, Router.state);
assert.isNull(await Router.init({}));
});
it("should only be called once", async () => {
Router = new _ASRouter();
initASRouter(Router);
let secondCall = await Router.init({});
assert.isNull(
secondCall,
"Should not init twice, it should exit early with null"
);
});
it("should set state.messageBlockList to the block list in persistent storage", async () => {
messageBlockList = ["foo"];
Router = new _ASRouter();
await initASRouter(Router);
assert.deepEqual(Router.state.messageBlockList, ["foo"]);
});
it("should initialize all the hub providers", async () => {
// ASRouter init called in `beforeEach` block above
assert.calledOnce(FakeToolbarBadgeHub.init);
assert.calledOnce(FakeToolbarPanelHub.init);
assert.calledOnce(FakeMomentsPageHub.init);
assert.calledWithExactly(
FakeToolbarBadgeHub.init,
Router.waitForInitialized,
{
handleMessageRequest: Router.handleMessageRequest,
addImpression: Router.addImpression,
blockMessageById: Router.blockMessageById,
sendTelemetry: Router.sendTelemetry,
unblockMessageById: Router.unblockMessageById,
}
);
assert.calledWithExactly(
FakeToolbarPanelHub.init,
Router.waitForInitialized,
{
getMessages: Router.handleMessageRequest,
sendTelemetry: Router.sendTelemetry,
}
);
assert.calledWithExactly(
FakeMomentsPageHub.init,
Router.waitForInitialized,
{
handleMessageRequest: Router.handleMessageRequest,
addImpression: Router.addImpression,
blockMessageById: Router.blockMessageById,
sendTelemetry: Router.sendTelemetry,
}
);
});
it("should set state.messageImpressions to the messageImpressions object in persistent storage", async () => {
// Note that messageImpressions are only kept if a message exists in router and has a .frequency property,
// otherwise they will be cleaned up by .cleanupImpressions()
const testMessage = { id: "foo", frequency: { lifetimeCap: 10 } };
messageImpressions = { foo: [0, 1, 2] };
setMessageProviderPref([
{ id: "onboarding", type: "local", messages: [testMessage] },
]);
Router = new _ASRouter();
await initASRouter(Router);
assert.deepEqual(Router.state.messageImpressions, messageImpressions);
});
it("should set state.screenImpressions to the screenImpressions object in persistent storage", async () => {
screenImpressions = { test: 123 };
Router = new _ASRouter();
await initASRouter(Router);
assert.deepEqual(Router.state.screenImpressions, screenImpressions);
});
it("should clear impressions for groups that are not active", async () => {
groupImpressions = { foo: [0, 1, 2] };
Router = new _ASRouter();
await initASRouter(Router);
assert.notProperty(Router.state.groupImpressions, "foo");
});
it("should keep impressions for groups that are active", async () => {
Router = new _ASRouter();
await initASRouter(Router);
await Router.setState(() => {
return {
groups: [
{
id: "foo",
enabled: true,
frequency: {
custom: [{ period: ONE_DAY_IN_MS, cap: 10 }],
lifetime: Infinity,
},
},
],
groupImpressions: { foo: [Date.now()] },
};
});
Router.cleanupImpressions();
assert.property(Router.state.groupImpressions, "foo");
assert.lengthOf(Router.state.groupImpressions.foo, 1);
});
it("should remove old impressions for a group", async () => {
Router = new _ASRouter();
await initASRouter(Router);
await Router.setState(() => {
return {
groups: [
{
id: "foo",
enabled: true,
frequency: {
custom: [{ period: ONE_DAY_IN_MS, cap: 10 }],
},
},
],
groupImpressions: {
foo: [Date.now() - ONE_DAY_IN_MS - 1, Date.now()],
},
};
});
Router.cleanupImpressions();
assert.property(Router.state.groupImpressions, "foo");
assert.lengthOf(Router.state.groupImpressions.foo, 1);
});
it("should await .loadMessagesFromAllProviders() and add messages from providers to state.messages", async () => {
Router = new _ASRouter(Object.freeze(FAKE_LOCAL_PROVIDERS));
await initASRouter(Router);
assert.calledOnce(Router.loadMessagesFromAllProviders);
assert.isArray(Router.state.messages);
assert.lengthOf(
Router.state.messages,
FAKE_LOCAL_MESSAGES.length + FAKE_REMOTE_MESSAGES.length
);
});
it("should set state.previousSessionEnd from IndexedDB", async () => {
previousSessionEnd = 200;
await createRouterAndInit();
assert.equal(Router.state.previousSessionEnd, previousSessionEnd);
});
it("should assign ASRouterPreferences.specialConditions to state", async () => {
assert.isTrue(ASRouterPreferences.specialConditions.someCondition);
assert.isTrue(Router.state.someCondition);
});
it("should add observer for `intl:app-locales-changed`", async () => {
sandbox.spy(global.Services.obs, "addObserver");
await createRouterAndInit();
assert.calledWithExactly(
global.Services.obs.addObserver,
Router._onLocaleChanged,
"intl:app-locales-changed"
);
});
it("should add a pref observer", async () => {
sandbox.spy(global.Services.prefs, "addObserver");
await createRouterAndInit();
assert.calledOnce(global.Services.prefs.addObserver);
assert.calledWithExactly(
global.Services.prefs.addObserver,
USE_REMOTE_L10N_PREF,
Router
);
});
describe("lazily loading local test providers", () => {
afterEach(() => {
Router.uninit();
});
it("should add the local test providers on init if devtools are enabled", async () => {
sandbox.stub(ASRouterPreferences, "devtoolsEnabled").get(() => true);
await createRouterAndInit();
assert.property(Router._localProviders, "PanelTestProvider");
});
it("should not add the local test providers on init if devtools are disabled", async () => {
sandbox.stub(ASRouterPreferences, "devtoolsEnabled").get(() => false);
await createRouterAndInit();
assert.notProperty(Router._localProviders, "PanelTestProvider");
});
});
});
describe("preference changes", () => {
it("should call ASRouterPreferences.init and add a listener on init", () => {
assert.calledOnce(ASRouterPreferences.init);
assert.calledWith(ASRouterPreferences.addListener, Router.onPrefChange);
});
it("should call ASRouterPreferences.uninit and remove the listener on uninit", () => {
Router.uninit();
assert.calledOnce(ASRouterPreferences.uninit);
assert.calledWith(
ASRouterPreferences.removeListener,
Router.onPrefChange
);
});
it("should send a AS_ROUTER_TARGETING_UPDATE message", async () => {
const messageTargeted = {
id: "1",
campaign: "foocampaign",
targeting: "true",
groups: ["cfr"],
provider: "cfr",
};
const messageNotTargeted = {
id: "2",
campaign: "foocampaign",
groups: ["cfr"],
provider: "cfr",
};
await Router.setState({
messages: [messageTargeted, messageNotTargeted],
providers: [{ id: "cfr" }],
});
fakeTargetingContext.evalWithDefault.resolves(false);
await Router.onPrefChange("services.sync.username");
assert.calledOnce(initParams.clearChildMessages);
assert.calledWith(initParams.clearChildMessages, [messageTargeted.id]);
});
it("should call loadMessagesFromAllProviders on pref change", () => {
ASRouterPreferences.observe(null, null, MESSAGE_PROVIDER_PREF_NAME);
assert.calledOnce(Router.loadMessagesFromAllProviders);
});
it("should update groups state if a user pref changes", async () => {
await Router.setState({
groups: [{ id: "foo", userPreferences: ["bar"], enabled: true }],
});
sandbox.stub(ASRouterPreferences, "getUserPreference");
await Router.onPrefChange(MESSAGE_PROVIDER_PREF_NAME);
assert.calledWithExactly(ASRouterPreferences.getUserPreference, "bar");
});
it("should update the list of providers on pref change", async () => {
const modifiedRemoteProvider = Object.assign({}, FAKE_REMOTE_PROVIDER, {
url: "baz.com",
});
setMessageProviderPref([
FAKE_LOCAL_PROVIDER,
modifiedRemoteProvider,
FAKE_REMOTE_SETTINGS_PROVIDER,
]);
const { length } = Router.state.providers;
ASRouterPreferences.observe(null, null, MESSAGE_PROVIDER_PREF_NAME);
await Router._updateMessageProviders();
const provider = Router.state.providers.find(p => p.url === "baz.com");
assert.lengthOf(Router.state.providers, length);
assert.isDefined(provider);
});
it("should clear disabled providers on pref change", async () => {
const TEST_PROVIDER_ID = "some_provider_id";
await Router.setState({
providers: [{ id: TEST_PROVIDER_ID }],
});
const modifiedRemoteProvider = Object.assign({}, FAKE_REMOTE_PROVIDER, {
id: TEST_PROVIDER_ID,
enabled: false,
});
setMessageProviderPref([
FAKE_LOCAL_PROVIDER,
modifiedRemoteProvider,
FAKE_REMOTE_SETTINGS_PROVIDER,
]);
await Router.onPrefChange(MESSAGE_PROVIDER_PREF_NAME);
assert.calledOnce(initParams.clearChildProviders);
assert.calledWith(initParams.clearChildProviders, [TEST_PROVIDER_ID]);
});
});
describe("setState", () => {
it("should broadcast a message to update the admin tool on a state change if the asrouter.devtoolsEnabled pref is", async () => {
sandbox.stub(ASRouterPreferences, "devtoolsEnabled").get(() => true);
sandbox.stub(Router, "getTargetingParameters").resolves({});
const state = await Router.setState({ foo: 123 });
assert.calledOnce(initParams.updateAdminState);
assert.deepEqual(state.providerPrefs, ASRouterPreferences.providers);
assert.deepEqual(
state.userPrefs,
ASRouterPreferences.getAllUserPreferences()
);
assert.deepEqual(state.targetingParameters, {});
assert.deepEqual(state.errors, Router.errors);
});
it("should not send a message on a state change asrouter.devtoolsEnabled pref is on", async () => {
sandbox.stub(ASRouterPreferences, "devtoolsEnabled").get(() => false);
await Router.setState({ foo: 123 });
assert.notCalled(initParams.updateAdminState);
});
});
describe("getTargetingParameters", () => {
it("should return the targeting parameters", async () => {
const stub = sandbox.stub().resolves("foo");
const obj = { foo: 1 };
sandbox.stub(obj, "foo").get(stub);
const result = await Router.getTargetingParameters(obj, obj);
assert.calledTwice(stub);
assert.propertyVal(result, "foo", "foo");
});
});
describe("evaluateExpression", () => {
it("should call ASRouterTargeting to evaluate", async () => {
fakeTargetingContext.evalWithDefault.resolves("foo");
const response = await Router.evaluateExpression({});
assert.equal(response.evaluationStatus.result, "foo");
assert.isTrue(response.evaluationStatus.success);
});
it("should catch evaluation errors", async () => {
fakeTargetingContext.evalWithDefault.returns(
Promise.reject(new Error("fake error"))
);
const response = await Router.evaluateExpression({});
assert.isFalse(response.evaluationStatus.success);
});
});
describe("#routeCFRMessage", () => {
let browser;
beforeEach(() => {
sandbox.stub(CFRPageActions, "forceRecommendation");
sandbox.stub(CFRPageActions, "addRecommendation");
browser = {};
});
it("should route whatsnew_panel_message message to the right hub", () => {
Router.routeCFRMessage(
{ template: "whatsnew_panel_message" },
browser,
"",
true
);
assert.calledOnce(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route moments messages to the right hub", () => {
Router.routeCFRMessage({ template: "update_action" }, browser, "", true);
assert.calledOnce(FakeMomentsPageHub.executeAction);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(CFRPageActions.forceRecommendation);
});
it("should route toolbar_badge message to the right hub", () => {
Router.routeCFRMessage({ template: "toolbar_badge" }, browser);
assert.calledOnce(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route milestone_message to the right hub", () => {
Router.routeCFRMessage(
{ template: "milestone_message" },
browser,
"",
false
);
assert.calledOnce(CFRPageActions.addRecommendation);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route cfr_doorhanger message to the right hub force = false", () => {
Router.routeCFRMessage(
{ template: "cfr_doorhanger" },
browser,
{ param: {} },
false
);
assert.calledOnce(CFRPageActions.addRecommendation);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route cfr_doorhanger message to the right hub force = true", () => {
Router.routeCFRMessage({ template: "cfr_doorhanger" }, browser, {}, true);
assert.calledOnce(CFRPageActions.forceRecommendation);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route cfr_urlbar_chiclet message to the right hub force = false", () => {
Router.routeCFRMessage(
{ template: "cfr_urlbar_chiclet" },
browser,
{ param: {} },
false
);
assert.calledOnce(CFRPageActions.addRecommendation);
const { args } = CFRPageActions.addRecommendation.firstCall;
// Host should be null
assert.isNull(args[1]);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route cfr_urlbar_chiclet message to the right hub force = true", () => {
Router.routeCFRMessage(
{ template: "cfr_urlbar_chiclet" },
browser,
{},
true
);
assert.calledOnce(CFRPageActions.forceRecommendation);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
it("should route default to sending to content", () => {
Router.routeCFRMessage(
{ template: "some_other_template" },
browser,
{},
true
);
assert.notCalled(FakeToolbarPanelHub.forceShowMessage);
assert.notCalled(CFRPageActions.forceRecommendation);
assert.notCalled(CFRPageActions.addRecommendation);
assert.notCalled(FakeToolbarBadgeHub.registerBadgeNotificationListener);
assert.notCalled(FakeMomentsPageHub.executeAction);
});
});
describe("#loadMessagesFromAllProviders", () => {
function assertRouterContainsMessages(messages) {
const messageIdsInRouter = Router.state.messages.map(m => m.id);
for (const message of messages) {
assert.include(messageIdsInRouter, message.id);
}
}
it("should not trigger an update if not enough time has passed for a provider", async () => {
await createRouterAndInit([
{
id: "remotey",
type: "remote",
enabled: true,
url: "http://fake.com/endpoint",
updateCycleInMs: 300,
},
]);
const previousState = Router.state;
// Since we've previously gotten messages during init and we haven't advanced our fake timer,
// no updates should be triggered.
await Router.loadMessagesFromAllProviders();
assert.deepEqual(Router.state, previousState);
});
it("should not trigger an update if we only have local providers", async () => {
await createRouterAndInit([
{
id: "foo",
type: "local",
enabled: true,
messages: FAKE_LOCAL_MESSAGES,
},
]);
const previousState = Router.state;
const stub = sandbox.stub(MessageLoaderUtils, "loadMessagesForProvider");
clock.tick(300);
await Router.loadMessagesFromAllProviders();
assert.deepEqual(Router.state, previousState);
assert.notCalled(stub);
});
it("should update messages for a provider if enough time has passed, without removing messages for other providers", async () => {
const NEW_MESSAGES = [{ id: "new_123" }];
await createRouterAndInit([
{
id: "remotey",
type: "remote",
url: "http://fake.com/endpoint",
enabled: true,
updateCycleInMs: 300,
},
{
id: "alocalprovider",
type: "local",
enabled: true,
messages: FAKE_LOCAL_MESSAGES,
},
]);
fetchStub.withArgs("http://fake.com/endpoint").resolves({
ok: true,
status: 200,
json: () => Promise.resolve({ messages: NEW_MESSAGES }),
headers: FAKE_RESPONSE_HEADERS,
});
clock.tick(301);
await Router.loadMessagesFromAllProviders();
// These are the new messages
assertRouterContainsMessages(NEW_MESSAGES);
// These are the local messages that should not have been deleted
assertRouterContainsMessages(FAKE_LOCAL_MESSAGES);
});
it("should parse the triggers in the messages and register the trigger listeners", async () => {
sandbox.spy(
ASRouterTriggerListeners.get("openURL"),
"init"
); /* eslint-disable object-property-newline */
/* eslint-disable object-curly-newline */ await createRouterAndInit([
{
id: "foo",
type: "local",
enabled: true,
messages: [
{
id: "foo",
template: "simple_template",
trigger: { id: "firstRun" },
content: { title: "Foo", body: "Foo123" },
},
{
id: "bar1",
template: "simple_template",
trigger: {
id: "openURL",
params: ["www.mozilla.org", "www.mozilla.com"],
},
content: { title: "Bar1", body: "Bar123" },
},
{
id: "bar2",
template: "simple_template",
trigger: { id: "openURL", params: ["www.example.com"] },
content: { title: "Bar2", body: "Bar123" },
},
],
},
]); /* eslint-enable object-property-newline */
/* eslint-enable object-curly-newline */ assert.calledTwice(
ASRouterTriggerListeners.get("openURL").init
);
assert.calledWithExactly(
ASRouterTriggerListeners.get("openURL").init,
Router._triggerHandler,
["www.mozilla.org", "www.mozilla.com"],
undefined
);
assert.calledWithExactly(
ASRouterTriggerListeners.get("openURL").init,
Router._triggerHandler,
["www.example.com"],
undefined
);
});
it("should parse the message's messagesLoaded trigger and immediately fire trigger", async () => {
setMessageProviderPref([
{
id: "foo",
type: "local",
enabled: true,
messages: [
{
id: "bar3",
template: "simple_template",
trigger: { id: "messagesLoaded" },
content: { title: "Bar3", body: "Bar123" },
},
],
},
]);
Router = new _ASRouter(Object.freeze(FAKE_LOCAL_PROVIDERS));
sandbox.spy(Router, "sendTriggerMessage");
await initASRouter(Router);
assert.calledOnce(Router.sendTriggerMessage);
assert.calledWith(
Router.sendTriggerMessage,
sandbox.match({ id: "messagesLoaded" }),
true
);
});
it("should gracefully handle messages loading before a window or browser exists", async () => {
sandbox.stub(global, "gBrowser").value(undefined);
sandbox
.stub(global.Services.wm, "getMostRecentBrowserWindow")
.returns(undefined);
setMessageProviderPref([
{
id: "foo",
type: "local",
enabled: true,
messages: [
"whatsnew_panel_message",
"cfr_doorhanger",
"toolbar_badge",
"update_action",
"infobar",
"spotlight",
"toast_notification",
].map((template, i) => {
return {
id: `foo${i}`,
template,
trigger: { id: "messagesLoaded" },
content: { title: `Foo${i}`, body: "Bar123" },
};
}),
},
]);
Router = new _ASRouter(Object.freeze(FAKE_LOCAL_PROVIDERS));
sandbox.spy(Router, "sendTriggerMessage");
await initASRouter(Router);
assert.calledWith(
Router.sendTriggerMessage,
sandbox.match({ id: "messagesLoaded" }),
true
);
});
it("should gracefully handle RemoteSettings blowing up and dispatch undesired event", async () => {
sandbox
.stub(MessageLoaderUtils, "_getRemoteSettingsMessages")
.rejects("fake error");
await createRouterAndInit();
assert.calledWith(initParams.dispatchCFRAction, {
data: {
action: "asrouter_undesired_event",
event: "ASR_RS_ERROR",
event_context: "remotey-settingsy",
message_id: "n/a",
},
meta: { from: "ActivityStream:Content", to: "ActivityStream:Main" },
type: "AS_ROUTER_TELEMETRY_USER_EVENT",
});
});
it("should dispatch undesired event if RemoteSettings returns no messages", async () => {
sandbox
.stub(MessageLoaderUtils, "_getRemoteSettingsMessages")
.resolves([]);
assert.calledWith(initParams.dispatchCFRAction, {
data: {
action: "asrouter_undesired_event",
event: "ASR_RS_NO_MESSAGES",
event_context: "remotey-settingsy",
message_id: "n/a",
},
meta: { from: "ActivityStream:Content", to: "ActivityStream:Main" },
type: "AS_ROUTER_TELEMETRY_USER_EVENT",
});
});
it("should download the attachment if RemoteSettings returns some messages", async () => {
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "en-US");
sandbox
.stub(MessageLoaderUtils, "_getRemoteSettingsMessages")
.resolves([{ id: "message_1" }]);
const spy = sandbox.spy();
global.Downloader.prototype.downloadToDisk = spy;
const provider = {
id: "cfr",
enabled: true,
type: "remote-settings",
collection: "cfr",
};
await createRouterAndInit([provider]);
assert.calledOnce(spy);
});
it("should dispatch undesired event if the ms-language-packs returns no messages", async () => {
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "en-US");
sandbox
.stub(MessageLoaderUtils, "_getRemoteSettingsMessages")
.resolves([{ id: "message_1" }]);
sandbox
.stub(global.KintoHttpClient.prototype, "getRecord")
.resolves(null);
const provider = {
id: "cfr",
enabled: true,
type: "remote-settings",
collection: "cfr",
};
await createRouterAndInit([provider]);
assert.calledWith(initParams.dispatchCFRAction, {
data: {
action: "asrouter_undesired_event",
event: "ASR_RS_NO_MESSAGES",
event_context: "ms-language-packs",
message_id: "n/a",
},
meta: { from: "ActivityStream:Content", to: "ActivityStream:Main" },
type: "AS_ROUTER_TELEMETRY_USER_EVENT",
});
});
});
describe("#_updateMessageProviders", () => {
it("should correctly replace %STARTPAGE_VERSION% in remote provider urls", async () => {
// If this test fails, you need to update the constant STARTPAGE_VERSION in
// ASRouter.jsm to match the `version` property of provider-response-schema.json
const expectedStartpageVersion = ProviderResponseSchema.version;
const provider = {
id: "foo",
enabled: true,
type: "remote",
url: "https://www.mozilla.org/%STARTPAGE_VERSION%/",
};
setMessageProviderPref([provider]);
await Router._updateMessageProviders();
assert.equal(
Router.state.providers[0].url,
`https://www.mozilla.org/${parseInt(expectedStartpageVersion, 10)}/`
);
});
it("should replace other params in remote provider urls by calling Services.urlFormater.formatURL", async () => {
const url = "https://www.example.com/";
const replacedUrl = "https://www.foo.bar/";
const stub = sandbox
.stub(global.Services.urlFormatter, "formatURL")
.withArgs(url)
.returns(replacedUrl);
const provider = { id: "foo", enabled: true, type: "remote", url };
setMessageProviderPref([provider]);
await Router._updateMessageProviders();
assert.calledOnce(stub);
assert.calledWithExactly(stub, url);
assert.equal(Router.state.providers[0].url, replacedUrl);
});
it("should only add the providers that are enabled", async () => {
const providers = [
{
id: "foo",
enabled: false,
type: "remote",
url: "https://www.foo.com/",
},
{
id: "bar",
enabled: true,
type: "remote",
url: "https://www.bar.com/",
},
];
setMessageProviderPref(providers);
await Router._updateMessageProviders();
assert.equal(Router.state.providers.length, 1);
assert.equal(Router.state.providers[0].id, providers[1].id);
});
});
describe("#handleMessageRequest", () => {
beforeEach(async () => {
await Router.setState(() => ({
providers: [{ id: "cfr" }, { id: "badge" }],
}));
});
it("should not return a blocked message", async () => {
// Block all messages except the first
await Router.setState(() => ({
messages: [
{ id: "foo", provider: "cfr", groups: ["cfr"] },
{ id: "bar", provider: "cfr", groups: ["cfr"] },
],
messageBlockList: ["foo"],
}));
await Router.handleMessageRequest({
provider: "cfr",
});
assert.calledWithMatch(ASRouterTargeting.findMatchingMessage, {
messages: [{ id: "bar", provider: "cfr", groups: ["cfr"] }],
});
});
it("should not return a message from a disabled group", async () => {
ASRouterTargeting.findMatchingMessage.callsFake(
({ messages }) => messages[0]
);
// Block all messages except the first
await Router.setState(() => ({
messages: [
{ id: "foo", provider: "cfr", groups: ["cfr"] },
{ id: "bar", provider: "cfr", groups: ["cfr"] },
],
groups: [{ id: "cfr", enabled: false }],
}));
const result = await Router.handleMessageRequest({
provider: "cfr",
});
assert.isNull(result);
});
it("should not return a message from a blocked campaign", async () => {
// Block all messages except the first
await Router.setState(() => ({
messages: [
{
id: "foo",
provider: "cfr",
campaign: "foocampaign",
groups: ["cfr"],
},
{ id: "bar", provider: "cfr", groups: ["cfr"] },
],
messageBlockList: ["foocampaign"],
}));
await Router.handleMessageRequest({
provider: "cfr",
});
assert.calledWithMatch(ASRouterTargeting.findMatchingMessage, {
messages: [{ id: "bar", provider: "cfr", groups: ["cfr"] }],
});
});
it("should not return a message excluded by the provider", async () => {
// There are only two providers; block the FAKE_LOCAL_PROVIDER, leaving
// only FAKE_REMOTE_PROVIDER unblocked, which provides only one message
await Router.setState(() => ({
providers: [{ id: "cfr", exclude: ["foo"] }],
}));
await Router.setState(() => ({
messages: [{ id: "foo", provider: "cfr" }],
messageBlockList: ["foocampaign"],
}));
const result = await Router.handleMessageRequest({
provider: "cfr",
});
assert.isNull(result);
});
it("should not return a message if the frequency cap has been hit", async () => {
sandbox.stub(Router, "isBelowFrequencyCaps").returns(false);
await Router.setState(() => ({
messages: [{ id: "foo", provider: "cfr" }],
}));
const result = await Router.handleMessageRequest({
provider: "cfr",
});
assert.isNull(result);
});
it("should get unblocked messages that match the trigger", async () => {
const message1 = {
id: "1",
campaign: "foocampaign",
trigger: { id: "foo" },
groups: ["cfr"],
provider: "cfr",
};
const message2 = {
id: "2",
campaign: "foocampaign",
trigger: { id: "bar" },
groups: ["cfr"],
provider: "cfr",
};
await Router.setState({ messages: [message2, message1] });
// Just return the first message provided as arg
ASRouterTargeting.findMatchingMessage.callsFake(
({ messages }) => messages[0]
);
const result = Router.handleMessageRequest({ triggerId: "foo" });
assert.deepEqual(result, message1);
});
it("should get unblocked messages that match trigger and template", async () => {
const message1 = {
id: "1",
campaign: "foocampaign",
template: "badge",
trigger: { id: "foo" },
groups: ["badge"],
provider: "badge",
};
const message2 = {
id: "2",
campaign: "foocampaign",
template: "test_template",
trigger: { id: "foo" },
groups: ["cfr"],
provider: "cfr",
};
await Router.setState({ messages: [message2, message1] });
// Just return the first message provided as arg
ASRouterTargeting.findMatchingMessage.callsFake(
({ messages }) => messages[0]
);
const result = Router.handleMessageRequest({
triggerId: "foo",
template: "badge",
});
assert.deepEqual(result, message1);
});
it("should have messageImpressions in the message context", () => {
assert.propertyVal(
Router._getMessagesContext(),
"messageImpressions",
Router.state.messageImpressions
);
});
it("should return all unblocked messages that match the template, trigger if returnAll=true", async () => {
const message1 = {
provider: "whats_new",
id: "1",
template: "whatsnew_panel_message",
trigger: { id: "whatsNewPanelOpened" },
groups: ["whats_new"],
};
const message2 = {
provider: "whats_new",
id: "2",
template: "whatsnew_panel_message",
trigger: { id: "whatsNewPanelOpened" },
groups: ["whats_new"],
};
const message3 = {
provider: "whats_new",
id: "3",
template: "badge",
groups: ["whats_new"],
};
ASRouterTargeting.findMatchingMessage.callsFake(() => [
message2,
message1,
]);
await Router.setState({
messages: [message3, message2, message1],
providers: [{ id: "whats_new" }],
});
const result = await Router.handleMessageRequest({
template: "whatsnew_panel_message",
triggerId: "whatsNewPanelOpened",
returnAll: true,
});
assert.deepEqual(result, [message2, message1]);
});
it("should forward trigger param info", async () => {
const trigger = {
triggerId: "foo",
triggerParam: "bar",
triggerContext: "context",
};
const message1 = {
id: "1",
campaign: "foocampaign",
trigger: { id: "foo" },
groups: ["cfr"],
provider: "cfr",
};
const message2 = {
id: "2",
campaign: "foocampaign",
trigger: { id: "bar" },
groups: ["badge"],
provider: "badge",
};
await Router.setState({ messages: [message2, message1] });
// Just return the first message provided as arg
Router.handleMessageRequest(trigger);
assert.calledOnce(ASRouterTargeting.findMatchingMessage);
const [options] = ASRouterTargeting.findMatchingMessage.firstCall.args;
assert.propertyVal(options.trigger, "id", trigger.triggerId);
assert.propertyVal(options.trigger, "param", trigger.triggerParam);
assert.propertyVal(options.trigger, "context", trigger.triggerContext);
});
it("should not cache badge messages", async () => {
const trigger = {
triggerId: "bar",
triggerParam: "bar",
triggerContext: "context",
};
const message1 = {
id: "1",
provider: "cfr",
campaign: "foocampaign",
trigger: { id: "foo" },
groups: ["cfr"],
};
const message2 = {
id: "2",
campaign: "foocampaign",
trigger: { id: "bar" },
groups: ["badge"],
provider: "badge",
};
await Router.setState({ messages: [message2, message1] });
// Just return the first message provided as arg
Router.handleMessageRequest(trigger);
assert.calledOnce(ASRouterTargeting.findMatchingMessage);
const [options] = ASRouterTargeting.findMatchingMessage.firstCall.args;
assert.propertyVal(options, "shouldCache", false);
});
it("should filter out messages without a trigger (or different) when a triggerId is defined", async () => {
const trigger = { triggerId: "foo" };
const message1 = {
id: "1",
campaign: "foocampaign",
trigger: { id: "foo" },
groups: ["cfr"],
provider: "cfr",
};
const message2 = {
id: "2",
campaign: "foocampaign",
trigger: { id: "bar" },
groups: ["cfr"],
provider: "cfr",
};
const message3 = {
id: "3",
campaign: "bazcampaign",
groups: ["cfr"],
provider: "cfr",
};
await Router.setState({
messages: [message2, message1, message3],
groups: [{ id: "cfr", enabled: true }],
});
// Just return the first message provided as arg
ASRouterTargeting.findMatchingMessage.callsFake(args => args.messages);
const result = Router.handleMessageRequest(trigger);
assert.lengthOf(result, 1);
assert.deepEqual(result[0], message1);
});
});
describe("#uninit", () => {
it("should unregister the trigger listeners", () => {
for (const listener of ASRouterTriggerListeners.values()) {
sandbox.spy(listener, "uninit");
}
Router.uninit();
for (const listener of ASRouterTriggerListeners.values()) {
assert.calledOnce(listener.uninit);
}
});
it("should set .dispatchCFRAction to null", () => {
Router.uninit();
assert.isNull(Router.dispatchCFRAction);
assert.isNull(Router.clearChildMessages);
assert.isNull(Router.sendTelemetry);
});
it("should save previousSessionEnd", () => {
Router.uninit();
assert.calledOnce(Router._storage.set);
assert.calledWithExactly(
Router._storage.set,
"previousSessionEnd",
sinon.match.number
);
});
it("should remove the observer for `intl:app-locales-changed`", () => {
sandbox.spy(global.Services.obs, "removeObserver");
Router.uninit();
assert.calledWithExactly(
global.Services.obs.removeObserver,
Router._onLocaleChanged,
"intl:app-locales-changed"
);
});
it("should remove the pref observer for `USE_REMOTE_L10N_PREF`", async () => {
sandbox.spy(global.Services.prefs, "removeObserver");
Router.uninit();
// Grab the last call as #uninit() also involves multiple calls of `Services.prefs.removeObserver`.
const call = global.Services.prefs.removeObserver.lastCall;
assert.calledWithExactly(call, USE_REMOTE_L10N_PREF, Router);
});
});
describe("#setMessageById", async () => {
it("should send an empty message if provided id did not resolve to a message", async () => {
let response = await Router.setMessageById({ id: -1 }, true, {});
assert.deepEqual(response.message, {});
});
});
describe("#isUnblockedMessage", () => {
it("should block a message if the group is blocked", async () => {
const msg = { id: "msg1", groups: ["foo"], provider: "unit-test" };
await Router.setState({
groups: [{ id: "foo", enabled: false }],
messages: [msg],
providers: [{ id: "unit-test" }],
});
assert.isFalse(Router.isUnblockedMessage(msg));
await Router.setState({ groups: [{ id: "foo", enabled: true }] });
assert.isTrue(Router.isUnblockedMessage(msg));
});
it("should block a message if at least one group is blocked", async () => {
const msg = {
id: "msg1",
groups: ["foo", "bar"],
provider: "unit-test",
};
await Router.setState({
groups: [
{ id: "foo", enabled: false },
{ id: "bar", enabled: false },
],
messages: [msg],
providers: [{ id: "unit-test" }],
});
assert.isFalse(Router.isUnblockedMessage(msg));
await Router.setState({
groups: [
{ id: "foo", enabled: true },
{ id: "bar", enabled: false },
],
});
assert.isFalse(Router.isUnblockedMessage(msg));
});
});
describe("#blockMessageById", () => {
it("should add the id to the messageBlockList", async () => {
await Router.blockMessageById("foo");
assert.isTrue(Router.state.messageBlockList.includes("foo"));
});
it("should add the campaign to the messageBlockList instead of id if .campaign is specified and not select messages of that campaign again", async () => {
await Router.setState({
messages: [
{ id: "1", campaign: "foocampaign" },
{ id: "2", campaign: "foocampaign" },
],
});
await Router.blockMessageById("1");
assert.isTrue(Router.state.messageBlockList.includes("foocampaign"));
assert.isEmpty(Router.state.messages.filter(Router.isUnblockedMessage));
});
it("should be able to add multiple items to the messageBlockList", async () => {
await await Router.blockMessageById(FAKE_BUNDLE.map(b => b.id));
assert.isTrue(Router.state.messageBlockList.includes(FAKE_BUNDLE[0].id));
assert.isTrue(Router.state.messageBlockList.includes(FAKE_BUNDLE[1].id));
});
it("should save the messageBlockList", async () => {
await await Router.blockMessageById(FAKE_BUNDLE.map(b => b.id));
assert.calledWithExactly(Router._storage.set, "messageBlockList", [
FAKE_BUNDLE[0].id,
FAKE_BUNDLE[1].id,
]);
});
});
describe("#unblockMessageById", () => {
it("should remove the id from the messageBlockList", async () => {
await Router.blockMessageById("foo");
assert.isTrue(Router.state.messageBlockList.includes("foo"));
await Router.unblockMessageById("foo");
assert.isFalse(Router.state.messageBlockList.includes("foo"));
});
it("should remove the campaign from the messageBlockList if it is defined", async () => {
await Router.setState({ messages: [{ id: "1", campaign: "foo" }] });
await Router.blockMessageById("1");
assert.isTrue(
Router.state.messageBlockList.includes("foo"),
"blocklist has campaign id"
);
await Router.unblockMessageById("1");
assert.isFalse(
Router.state.messageBlockList.includes("foo"),
"campaign id removed from blocklist"
);
});
it("should save the messageBlockList", async () => {
await Router.unblockMessageById("foo");
assert.calledWithExactly(Router._storage.set, "messageBlockList", []);
});
});
describe("#routeCFRMessage", () => {
it("should allow for echoing back message modifications", () => {
const message = { somekey: "some value" };
const data = { content: message };
const browser = {};
let msg = Router.routeCFRMessage(data.content, browser, data, false);
assert.deepEqual(msg.message, message);
});
it("should call CFRPageActions.forceRecommendation if the template is cfr_action and force is true", async () => {
sandbox.stub(CFRPageActions, "forceRecommendation");
const testMessage = { id: "foo", template: "cfr_doorhanger" };
await Router.setState({ messages: [testMessage] });
Router.routeCFRMessage(testMessage, {}, null, true);
assert.calledOnce(CFRPageActions.forceRecommendation);
});
it("should call CFRPageActions.addRecommendation if the template is cfr_action and force is false", async () => {
sandbox.stub(CFRPageActions, "addRecommendation");
const testMessage = { id: "foo", template: "cfr_doorhanger" };
await Router.setState({ messages: [testMessage] });
Router.routeCFRMessage(testMessage, {}, {}, false);
assert.calledOnce(CFRPageActions.addRecommendation);
});
});
describe("#updateTargetingParameters", () => {
it("should return an object containing the whole state", async () => {
sandbox.stub(Router, "getTargetingParameters").resolves({});
let msg = await Router.updateTargetingParameters();
let expected = Object.assign({}, Router.state, {
providerPrefs: ASRouterPreferences.providers,
userPrefs: ASRouterPreferences.getAllUserPreferences(),
targetingParameters: {},
errors: Router.errors,
devtoolsEnabled: ASRouterPreferences.devtoolsEnabled,
});
assert.deepEqual(msg, expected);
});
});
describe("#reachEvent", () => {
let experimentAPIStub;
let featureIds = ["cfr", "moments-page", "infobar", "spotlight"];
beforeEach(() => {
let getExperimentMetaDataStub = sandbox.stub();
let getAllBranchesStub = sandbox.stub();
featureIds.forEach(feature => {
global.NimbusFeatures[feature].getAllVariables.returns({
id: `message-${feature}`,
});
getExperimentMetaDataStub.withArgs({ featureId: feature }).returns({
slug: `slug-${feature}`,
branch: {
slug: `branch-${feature}`,
},
});
getAllBranchesStub.withArgs(`slug-${feature}`).resolves([
{
slug: `other-branch-${feature}`,
[feature]: { value: { trigger: "unit-test" } },
},
]);
});
experimentAPIStub = {
getExperimentMetaData: getExperimentMetaDataStub,
getAllBranches: getAllBranchesStub,
};
globals.set("ExperimentAPI", experimentAPIStub);
});
afterEach(() => {
sandbox.restore();
});
it("should tag `forReachEvent` for all the expected message types", async () => {
// This should match the `providers.messaging-experiments`
let response = await MessageLoaderUtils.loadMessagesForProvider({
type: "remote-experiments",
featureIds,
});
// 1 message for reach 1 for expose
assert.property(response, "messages");
assert.lengthOf(response.messages, featureIds.length * 2);
assert.lengthOf(
response.messages.filter(m => m.forReachEvent),
featureIds.length
);
});
});
describe("#sendTriggerMessage", () => {
it("should pass the trigger to ASRouterTargeting when sending trigger message", async () => {
await Router.setState({
messages: [
{
id: "foo1",
provider: "onboarding",
template: "onboarding",
trigger: { id: "firstRun" },
content: { title: "Foo1", body: "Foo123-1" },
groups: ["onboarding"],
},
],
providers: [{ id: "onboarding" }],
});
Router.loadMessagesFromAllProviders.resetHistory();
Router.loadMessagesFromAllProviders.onFirstCall().resolves();
await Router.sendTriggerMessage({
tabId: 0,
browser: {},
id: "firstRun",
});
assert.calledOnce(ASRouterTargeting.findMatchingMessage);
assert.deepEqual(
ASRouterTargeting.findMatchingMessage.firstCall.args[0].trigger,
{
id: "firstRun",
param: undefined,
context: undefined,
}
);
});
it("should record telemetry information", async () => {
const startTelemetryStopwatch = sandbox.stub(
global.TelemetryStopwatch,
"start"
);
const finishTelemetryStopwatch = sandbox.stub(
global.TelemetryStopwatch,
"finish"
);
const tabId = 123;
await Router.sendTriggerMessage({
tabId,
browser: {},
id: "firstRun",
});
assert.calledTwice(startTelemetryStopwatch);
assert.calledWithExactly(
startTelemetryStopwatch,
"MS_MESSAGE_REQUEST_TIME_MS",
{ tabId }
);
assert.calledTwice(finishTelemetryStopwatch);
assert.calledWithExactly(
finishTelemetryStopwatch,
"MS_MESSAGE_REQUEST_TIME_MS",
{ tabId }
);
});
it("should have previousSessionEnd in the message context", () => {
assert.propertyVal(
Router._getMessagesContext(),
"previousSessionEnd",
100
);
});
it("should record the Reach event if found any", async () => {
let messages = [
{
id: "foo1",
forReachEvent: { sent: false, group: "cfr" },
experimentSlug: "exp01",
branchSlug: "branch01",
template: "simple_template",
trigger: { id: "foo" },
content: { title: "Foo1", body: "Foo123-1" },
},
{
id: "foo2",
template: "simple_template",
trigger: { id: "bar" },
content: { title: "Foo2", body: "Foo123-2" },
provider: "onboarding",
},
{
id: "foo3",
forReachEvent: { sent: false, group: "cfr" },
experimentSlug: "exp02",
branchSlug: "branch02",
template: "simple_template",
trigger: { id: "foo" },
content: { title: "Foo1", body: "Foo123-1" },
},
];
sandbox.stub(Router, "handleMessageRequest").resolves(messages);
sandbox.spy(Services.telemetry, "recordEvent");
await Router.sendTriggerMessage({
tabId: 0,
browser: {},
id: "foo",
});
assert.calledTwice(Services.telemetry.recordEvent);
});
it("should not record the Reach event if it's already sent", async () => {
let messages = [
{
id: "foo1",
forReachEvent: { sent: true, group: "cfr" },
experimentSlug: "exp01",
branchSlug: "branch01",
template: "simple_template",
trigger: { id: "foo" },
content: { title: "Foo1", body: "Foo123-1" },
},
];
sandbox.stub(Router, "handleMessageRequest").resolves(messages);
sandbox.spy(Services.telemetry, "recordEvent");
await Router.sendTriggerMessage({
tabId: 0,
browser: {},
id: "foo",
});
assert.notCalled(Services.telemetry.recordEvent);
});
it("should record the Exposure event for each valid feature", async () => {
["cfr_doorhanger", "update_action", "infobar", "spotlight"].forEach(
async template => {
let featureMap = {
cfr_doorhanger: "cfr",
spotlight: "spotlight",
infobar: "infobar",
update_action: "moments-page",
};
assert.notCalled(
global.NimbusFeatures[featureMap[template]].recordExposureEvent
);
let messages = [
{
id: "foo1",
template,
trigger: { id: "foo" },
content: { title: "Foo1", body: "Foo123-1" },
},
];
sandbox.stub(Router, "handleMessageRequest").resolves(messages);
await Router.sendTriggerMessage({
tabId: 0,
browser: {},
id: "foo",
});
assert.calledOnce(
global.NimbusFeatures[featureMap[template]].recordExposureEvent
);
}
);
});
});
describe("forceAttribution", () => {
let setAttributionString;
beforeEach(() => {
setAttributionString = sandbox.spy(Router, "setAttributionString");
sandbox.stub(global.Services.env, "set");
});
afterEach(() => {
sandbox.reset();
});
it("should double encode on windows", async () => {
sandbox.stub(fakeAttributionCode, "writeAttributionFile");
Router.forceAttribution({ foo: "FOO!", eh: "NOPE", bar: "BAR?" });
assert.notCalled(setAttributionString);
assert.calledWithMatch(
fakeAttributionCode.writeAttributionFile,
"foo%3DFOO!%26bar%3DBAR%253F"
);
});
it("should set attribution string on mac", async () => {
sandbox.stub(global.AppConstants, "platform").value("macosx");
Router.forceAttribution({ foo: "FOO!", eh: "NOPE", bar: "BAR?" });
assert.calledOnce(setAttributionString);
assert.calledWithMatch(
setAttributionString,
"foo%3DFOO!%26bar%3DBAR%253F"
);
});
});
describe("#forceWNPanel", () => {
let browser = {
ownerGlobal: {
document: new Document(),
PanelUI: {
showSubView: sinon.stub(),
panel: {
setAttribute: sinon.stub(),
},
},
},
};
let fakePanel = {
setAttribute: sinon.stub(),
};
sinon
.stub(browser.ownerGlobal.document, "getElementById")
.returns(fakePanel);
it("should call enableToolbarButton", async () => {
await Router.forceWNPanel(browser);
assert.calledOnce(FakeToolbarPanelHub.enableToolbarButton);
assert.calledOnce(browser.ownerGlobal.PanelUI.showSubView);
assert.calledWith(fakePanel.setAttribute, "noautohide", true);
});
});
describe("_triggerHandler", () => {
it("should call #sendTriggerMessage with the correct trigger", () => {
const getter = sandbox.stub();
getter.returns(false);
sandbox.stub(global.BrowserHandler, "kiosk").get(getter);
sinon.spy(Router, "sendTriggerMessage");
const browser = {};
const trigger = { id: "FAKE_TRIGGER", param: "some fake param" };
Router._triggerHandler(browser, trigger);
assert.calledOnce(Router.sendTriggerMessage);
assert.calledWith(
Router.sendTriggerMessage,
sandbox.match({
id: "FAKE_TRIGGER",
param: "some fake param",
})
);
});
});
describe("_triggerHandler_kiosk", () => {
it("should not call #sendTriggerMessage", () => {
const getter = sandbox.stub();
getter.returns(true);
sandbox.stub(global.BrowserHandler, "kiosk").get(getter);
sinon.spy(Router, "sendTriggerMessage");
const browser = {};
const trigger = { id: "FAKE_TRIGGER", param: "some fake param" };
Router._triggerHandler(browser, trigger);
assert.notCalled(Router.sendTriggerMessage);
});
});
describe("valid preview endpoint", () => {
it("should report an error if url protocol is not https", () => {
sandbox.stub(console, "error");
assert.equal(false, Router._validPreviewEndpoint("http://foo.com"));
assert.calledTwice(console.error);
});
});
describe("impressions", () => {
describe("#addImpression for groups", () => {
it("should save an impression in each group-with-frequency in a message", async () => {
const fooMessageImpressions = [0];
const aGroupImpressions = [0, 1, 2];
const bGroupImpressions = [3, 4, 5];
const cGroupImpressions = [6, 7, 8];
const message = {
id: "foo",
provider: "bar",
groups: ["a", "b", "c"],
};
const groups = [
{ id: "a", frequency: { lifetime: 3 } },
{ id: "b", frequency: { lifetime: 4 } },
{ id: "c", frequency: { lifetime: 5 } },
];
await Router.setState(state => {
// Add provider
const providers = [...state.providers];
// Add fooMessageImpressions
// eslint-disable-next-line no-shadow
const messageImpressions = Object.assign(
{},
state.messageImpressions
);
let gImpressions = {};
gImpressions.a = aGroupImpressions;
gImpressions.b = bGroupImpressions;
gImpressions.c = cGroupImpressions;
messageImpressions.foo = fooMessageImpressions;
return {
providers,
messageImpressions,
groups,
groupImpressions: gImpressions,
};
});
await Router.addImpression(message);
assert.deepEqual(
Router.state.groupImpressions.a,
[0, 1, 2, 0],
"a impressions"
);
assert.deepEqual(
Router.state.groupImpressions.b,
[3, 4, 5, 0],
"b impressions"
);
assert.deepEqual(
Router.state.groupImpressions.c,
[6, 7, 8, 0],
"c impressions"
);
});
});
describe("#isBelowFrequencyCaps", () => {
it("should call #_isBelowItemFrequencyCap for the message and for the provider with the correct impressions and arguments", async () => {
sinon.spy(Router, "_isBelowItemFrequencyCap");
const MAX_MESSAGE_LIFETIME_CAP = 100; // Defined in ASRouter
const fooMessageImpressions = [0, 1];
const barGroupImpressions = [0, 1, 2];
const message = {
id: "foo",
provider: "bar",
groups: ["bar"],
frequency: { lifetime: 3 },
};
const groups = [{ id: "bar", frequency: { lifetime: 5 } }];
await Router.setState(state => {
// Add provider
const providers = [...state.providers];
// Add fooMessageImpressions
// eslint-disable-next-line no-shadow
const messageImpressions = Object.assign(
{},
state.messageImpressions
);
let gImpressions = {};
gImpressions.bar = barGroupImpressions;
messageImpressions.foo = fooMessageImpressions;
return {
providers,
messageImpressions,
groups,
groupImpressions: gImpressions,
};
});
await Router.isBelowFrequencyCaps(message);
assert.calledTwice(Router._isBelowItemFrequencyCap);
assert.calledWithExactly(
Router._isBelowItemFrequencyCap,
message,
fooMessageImpressions,
MAX_MESSAGE_LIFETIME_CAP
);
assert.calledWithExactly(
Router._isBelowItemFrequencyCap,
groups[0],
barGroupImpressions
);
});
});
describe("#_isBelowItemFrequencyCap", () => {
it("should return false if the # of impressions exceeds the maxLifetimeCap", () => {
const item = { id: "foo", frequency: { lifetime: 5 } };
const impressions = [0, 1];
const maxLifetimeCap = 1;
const result = Router._isBelowItemFrequencyCap(
item,
impressions,
maxLifetimeCap
);
assert.isFalse(result);
});
describe("lifetime frequency caps", () => {
it("should return true if .frequency is not defined on the item", () => {
const item = { id: "foo" };
const impressions = [0, 1];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isTrue(result);
});
it("should return true if there are no impressions", () => {
const item = {
id: "foo",
frequency: {
lifetime: 10,
custom: [{ period: ONE_DAY_IN_MS, cap: 2 }],
},
};
const impressions = [];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isTrue(result);
});
it("should return true if the # of impressions is less than .frequency.lifetime of the item", () => {
const item = { id: "foo", frequency: { lifetime: 3 } };
const impressions = [0, 1];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isTrue(result);
});
it("should return false if the # of impressions is equal to .frequency.lifetime of the item", async () => {
const item = { id: "foo", frequency: { lifetime: 3 } };
const impressions = [0, 1, 2];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isFalse(result);
});
it("should return false if the # of impressions is greater than .frequency.lifetime of the item", async () => {
const item = { id: "foo", frequency: { lifetime: 3 } };
const impressions = [0, 1, 2, 3];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isFalse(result);
});
});
describe("custom frequency caps", () => {
it("should return true if impressions in the time period < the cap and total impressions < the lifetime cap", () => {
clock.tick(ONE_DAY_IN_MS + 10);
const item = {
id: "foo",
frequency: {
custom: [{ period: ONE_DAY_IN_MS, cap: 2 }],
lifetime: 3,
},
};
const impressions = [0, ONE_DAY_IN_MS + 1];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isTrue(result);
});
it("should return false if impressions in the time period > the cap and total impressions < the lifetime cap", () => {
clock.tick(200);
const item = {
id: "msg1",
frequency: { custom: [{ period: 100, cap: 2 }], lifetime: 3 },
};
const impressions = [0, 160, 161];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isFalse(result);
});
it("should return false if impressions in one of the time periods > the cap and total impressions < the lifetime cap", () => {
clock.tick(ONE_DAY_IN_MS + 200);
const itemTrue = {
id: "msg2",
frequency: { custom: [{ period: 100, cap: 2 }] },
};
const itemFalse = {
id: "msg1",
frequency: {
custom: [
{ period: 100, cap: 2 },
{ period: ONE_DAY_IN_MS, cap: 3 },
],
},
};
const impressions = [
0,
ONE_DAY_IN_MS + 160,
ONE_DAY_IN_MS - 100,
ONE_DAY_IN_MS - 200,
];
assert.isTrue(Router._isBelowItemFrequencyCap(itemTrue, impressions));
assert.isFalse(
Router._isBelowItemFrequencyCap(itemFalse, impressions)
);
});
it("should return false if impressions in the time period < the cap and total impressions > the lifetime cap", () => {
clock.tick(ONE_DAY_IN_MS + 10);
const item = {
id: "msg1",
frequency: {
custom: [{ period: ONE_DAY_IN_MS, cap: 2 }],
lifetime: 3,
},
};
const impressions = [0, 1, 2, 3, ONE_DAY_IN_MS + 1];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isFalse(result);
});
it("should return true if daily impressions < the daily cap and there is no lifetime cap", () => {
clock.tick(ONE_DAY_IN_MS + 10);
const item = {
id: "msg1",
frequency: { custom: [{ period: ONE_DAY_IN_MS, cap: 2 }] },
};
const impressions = [0, 1, 2, 3, ONE_DAY_IN_MS + 1];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isTrue(result);
});
it("should return false if daily impressions > the daily cap and there is no lifetime cap", () => {
clock.tick(ONE_DAY_IN_MS + 10);
const item = {
id: "msg1",
frequency: { custom: [{ period: ONE_DAY_IN_MS, cap: 2 }] },
};
const impressions = [
0,
1,
2,
3,
ONE_DAY_IN_MS + 1,
ONE_DAY_IN_MS + 2,
ONE_DAY_IN_MS + 3,
];
const result = Router._isBelowItemFrequencyCap(item, impressions);
assert.isFalse(result);
});
});
});
describe("#getLongestPeriod", () => {
it("should return the period if there is only one definition", () => {
const message = {
id: "foo",
frequency: { custom: [{ period: 200, cap: 2 }] },
};
assert.equal(Router.getLongestPeriod(message), 200);
});
it("should return the longest period if there are more than one definitions", () => {
const message = {
id: "foo",
frequency: {
custom: [
{ period: 1000, cap: 3 },
{ period: ONE_DAY_IN_MS, cap: 5 },
{ period: 100, cap: 2 },
],
},
};
assert.equal(Router.getLongestPeriod(message), ONE_DAY_IN_MS);
});
it("should return null if there are is no .frequency", () => {
const message = { id: "foo" };
assert.isNull(Router.getLongestPeriod(message));
});
it("should return null if there are is no .frequency.custom", () => {
const message = { id: "foo", frequency: { lifetime: 10 } };
assert.isNull(Router.getLongestPeriod(message));
});
});
describe("cleanup on init", () => {
it("should clear messageImpressions for messages which do not exist in state.messages", async () => {
const messages = [{ id: "foo", frequency: { lifetime: 10 } }];
messageImpressions = { foo: [0], bar: [0, 1] };
// Impressions for "bar" should be removed since that id does not exist in messages
const result = { foo: [0] };
await createRouterAndInit([
{ id: "onboarding", type: "local", messages, enabled: true },
]);
assert.calledWith(Router._storage.set, "messageImpressions", result);
assert.deepEqual(Router.state.messageImpressions, result);
});
it("should clear messageImpressions older than the period if no lifetime impression cap is included", async () => {
const CURRENT_TIME = ONE_DAY_IN_MS * 2;
clock.tick(CURRENT_TIME);
const messages = [
{
id: "foo",
frequency: { custom: [{ period: ONE_DAY_IN_MS, cap: 5 }] },
},
];
messageImpressions = { foo: [0, 1, CURRENT_TIME - 10] };
// Only 0 and 1 are more than 24 hours before CURRENT_TIME
const result = { foo: [CURRENT_TIME - 10] };
await createRouterAndInit([
{ id: "onboarding", type: "local", messages, enabled: true },
]);
assert.calledWith(Router._storage.set, "messageImpressions", result);
assert.deepEqual(Router.state.messageImpressions, result);
});
it("should clear messageImpressions older than the longest period if no lifetime impression cap is included", async () => {
const CURRENT_TIME = ONE_DAY_IN_MS * 2;
clock.tick(CURRENT_TIME);
const messages = [
{
id: "foo",
frequency: {
custom: [
{ period: ONE_DAY_IN_MS, cap: 5 },
{ period: 100, cap: 2 },
],
},
},
];
messageImpressions = { foo: [0, 1, CURRENT_TIME - 10] };
// Only 0 and 1 are more than 24 hours before CURRENT_TIME
const result = { foo: [CURRENT_TIME - 10] };
await createRouterAndInit([
{ id: "onboarding", type: "local", messages, enabled: true },
]);
assert.calledWith(Router._storage.set, "messageImpressions", result);
assert.deepEqual(Router.state.messageImpressions, result);
});
it("should clear messageImpressions if they are not properly formatted", async () => {
const messages = [{ id: "foo", frequency: { lifetime: 10 } }];
// this is impromperly formatted since messageImpressions are supposed to be an array
messageImpressions = { foo: 0 };
const result = {};
await createRouterAndInit([
{ id: "onboarding", type: "local", messages, enabled: true },
]);
assert.calledWith(Router._storage.set, "messageImpressions", result);
assert.deepEqual(Router.state.messageImpressions, result);
});
it("should not clear messageImpressions for messages which do exist in state.messages", async () => {
const messages = [
{ id: "foo", frequency: { lifetime: 10 } },
{ id: "bar", frequency: { lifetime: 10 } },
];
messageImpressions = { foo: [0], bar: [] };
await createRouterAndInit([
{ id: "onboarding", type: "local", messages, enabled: true },
]);
assert.notCalled(Router._storage.set);
assert.deepEqual(Router.state.messageImpressions, messageImpressions);
});
});
});
describe("#_onLocaleChanged", () => {
it("should call _maybeUpdateL10nAttachment in the handler", async () => {
sandbox.spy(Router, "_maybeUpdateL10nAttachment");
await Router._onLocaleChanged();
assert.calledOnce(Router._maybeUpdateL10nAttachment);
});
});
describe("#_maybeUpdateL10nAttachment", () => {
it("should update the l10n attachment if the locale was changed", async () => {
const getter = sandbox.stub();
getter.onFirstCall().returns("en-US");
getter.onSecondCall().returns("fr");
sandbox.stub(global.Services.locale, "appLocaleAsBCP47").get(getter);
const provider = {
id: "cfr",
enabled: true,
type: "remote-settings",
collection: "cfr",
};
await createRouterAndInit([provider]);
sandbox.spy(Router, "setState");
Router.loadMessagesFromAllProviders.resetHistory();
await Router._maybeUpdateL10nAttachment();
assert.calledWith(Router.setState, {
localeInUse: "fr",
providers: [
{
id: "cfr",
enabled: true,
type: "remote-settings",
collection: "cfr",
lastUpdated: undefined,
errors: [],
},
],
});
assert.calledOnce(Router.loadMessagesFromAllProviders);
});
it("should not update the l10n attachment if the provider doesn't need l10n attachment", async () => {
const getter = sandbox.stub();
getter.onFirstCall().returns("en-US");
getter.onSecondCall().returns("fr");
sandbox.stub(global.Services.locale, "appLocaleAsBCP47").get(getter);
const provider = {
id: "localProvider",
enabled: true,
type: "local",
};
await createRouterAndInit([provider]);
Router.loadMessagesFromAllProviders.resetHistory();
sandbox.spy(Router, "setState");
await Router._maybeUpdateL10nAttachment();
assert.notCalled(Router.setState);
assert.notCalled(Router.loadMessagesFromAllProviders);
});
});
describe("#observe", () => {
it("should reload l10n for CFRPageActions when the `USE_REMOTE_L10N_PREF` pref is changed", () => {
sandbox.spy(CFRPageActions, "reloadL10n");
Router.observe("", "", USE_REMOTE_L10N_PREF);
assert.calledOnce(CFRPageActions.reloadL10n);
});
it("should not react to other pref changes", () => {
sandbox.spy(CFRPageActions, "reloadL10n");
Router.observe("", "", "foo");
assert.notCalled(CFRPageActions.reloadL10n);
});
});
describe("#loadAllMessageGroups", () => {
it("should disable the group if the pref is false", async () => {
sandbox.stub(ASRouterPreferences, "getUserPreference").returns(false);
sandbox.stub(MessageLoaderUtils, "_getRemoteSettingsMessages").resolves([
{
id: "provider-group",
enabled: true,
type: "remote",
userPreferences: ["cfrAddons"],
},
]);
await Router.setState({
providers: [
{
id: "message-groups",
enabled: true,
collection: "collection",
type: "remote-settings",
},
],
});
await Router.loadAllMessageGroups();
const group = Router.state.groups.find(g => g.id === "provider-group");
assert.ok(group);
assert.propertyVal(group, "enabled", false);
});
it("should enable the group if at least one pref is true", async () => {
sandbox
.stub(ASRouterPreferences, "getUserPreference")
.withArgs("cfrAddons")
.returns(false)
.withArgs("cfrFeatures")
.returns(true);
sandbox.stub(MessageLoaderUtils, "_getRemoteSettingsMessages").resolves([
{
id: "provider-group",
enabled: true,
type: "remote",
userPreferences: ["cfrAddons", "cfrFeatures"],
},
]);
await Router.setState({
providers: [
{
id: "message-groups",
enabled: true,
collection: "collection",
type: "remote-settings",
},
],
});
await Router.loadAllMessageGroups();
const group = Router.state.groups.find(g => g.id === "provider-group");
assert.ok(group);
assert.propertyVal(group, "enabled", true);
});
it("should be keep the group disabled if disabled is true", async () => {
sandbox.stub(ASRouterPreferences, "getUserPreference").returns(true);
sandbox.stub(MessageLoaderUtils, "_getRemoteSettingsMessages").resolves([
{
id: "provider-group",
enabled: false,
type: "remote",
userPreferences: ["cfrAddons"],
},
]);
await Router.setState({
providers: [
{
id: "message-groups",
enabled: true,
collection: "collection",
type: "remote-settings",
},
],
});
await Router.loadAllMessageGroups();
const group = Router.state.groups.find(g => g.id === "provider-group");
assert.ok(group);
assert.propertyVal(group, "enabled", false);
});
it("should keep local groups unchanged if provider doesn't require an update", async () => {
sandbox.stub(MessageLoaderUtils, "shouldProviderUpdate").returns(false);
sandbox.stub(MessageLoaderUtils, "_loadDataForProvider");
await Router.setState({
groups: [
{
id: "cfr",
enabled: true,
collection: "collection",
type: "remote-settings",
},
],
});
await Router.loadAllMessageGroups();
const group = Router.state.groups.find(g => g.id === "cfr");
assert.ok(group);
assert.propertyVal(group, "enabled", true);
// Because it should not have updated
assert.notCalled(MessageLoaderUtils._loadDataForProvider);
});
it("should update local groups on pref change (no RS update)", async () => {
sandbox.stub(MessageLoaderUtils, "shouldProviderUpdate").returns(false);
sandbox.stub(ASRouterPreferences, "getUserPreference").returns(false);
await Router.setState({
groups: [
{
id: "cfr",
enabled: true,
collection: "collection",
type: "remote-settings",
userPreferences: ["cfrAddons"],
},
],
});
await Router.loadAllMessageGroups();
const group = Router.state.groups.find(g => g.id === "cfr");
assert.ok(group);
// Pref changed, updated the group state
assert.propertyVal(group, "enabled", false);
});
});
describe("unblockAll", () => {
it("Clears the message block list and returns the state value", async () => {
await Router.setState({ messageBlockList: ["one", "two", "three"] });
assert.equal(Router.state.messageBlockList.length, 3);
const state = await Router.unblockAll();
assert.equal(Router.state.messageBlockList.length, 0);
assert.equal(state.messageBlockList.length, 0);
});
});
describe("#loadMessagesForProvider", () => {
it("should fetch messages from the ExperimentAPI", async () => {
const args = {
type: "remote-experiments",
featureIds: ["spotlight"],
};
await MessageLoaderUtils.loadMessagesForProvider(args);
assert.calledOnce(global.NimbusFeatures.spotlight.getAllVariables);
assert.calledOnce(global.ExperimentAPI.getExperimentMetaData);
assert.calledWithExactly(global.ExperimentAPI.getExperimentMetaData, {
featureId: "spotlight",
});
});
it("should handle the case of no experiments in the ExperimentAPI", async () => {
const args = {
type: "remote-experiments",
featureIds: ["infobar"],
};
global.ExperimentAPI.getExperiment.returns(null);
const result = await MessageLoaderUtils.loadMessagesForProvider(args);
assert.lengthOf(result.messages, 0);
});
it("should normally load ExperimentAPI messages", async () => {
const args = {
type: "remote-experiments",
featureIds: ["infobar"],
};
const enrollment = {
branch: {
slug: "branch01",
infobar: {
featureId: "infobar",
value: { id: "id01", trigger: { id: "openURL" } },
},
},
};
global.NimbusFeatures.infobar.getAllVariables.returns(
enrollment.branch.infobar.value
);
global.ExperimentAPI.getExperimentMetaData.returns({
branch: { slug: enrollment.branch.slug },
});
global.ExperimentAPI.getAllBranches.returns([
enrollment.branch,
{
slug: "control",
infobar: {
featureId: "infobar",
value: null,
},
},
]);
const result = await MessageLoaderUtils.loadMessagesForProvider(args);
assert.lengthOf(result.messages, 1);
});
it("should skip disabled features and not load the messages", async () => {
const args = {
type: "remote-experiments",
featureIds: ["cfr"],
};
global.NimbusFeatures.cfr.getAllVariables.returns(null);
const result = await MessageLoaderUtils.loadMessagesForProvider(args);
assert.lengthOf(result.messages, 0);
});
it("should fetch branches with trigger", async () => {
const args = {
type: "remote-experiments",
featureIds: ["cfr"],
};
const enrollment = {
slug: "exp01",
branch: {
slug: "branch01",
cfr: {
featureId: "cfr",
value: { id: "id01", trigger: { id: "openURL" } },
},
},
};
global.NimbusFeatures.cfr.getAllVariables.returns(
enrollment.branch.cfr.value
);
global.ExperimentAPI.getExperimentMetaData.returns({
slug: enrollment.slug,
active: true,
branch: { slug: enrollment.branch.slug },
});
global.ExperimentAPI.getAllBranches.resolves([
enrollment.branch,
{
slug: "branch02",
cfr: {
featureId: "cfr",
value: { id: "id02", trigger: { id: "openURL" } },
},
},
{
// This branch should not be loaded as it doesn't have the trigger
slug: "branch03",
cfr: {
featureId: "cfr",
value: { id: "id03" },
},
},
]);
const result = await MessageLoaderUtils.loadMessagesForProvider(args);
assert.equal(result.messages.length, 2);
assert.equal(result.messages[0].id, "id01");
assert.equal(result.messages[1].id, "id02");
assert.equal(result.messages[1].experimentSlug, "exp01");
assert.equal(result.messages[1].branchSlug, "branch02");
assert.deepEqual(result.messages[1].forReachEvent, {
sent: false,
group: "cfr",
});
});
it("should fetch branches with trigger even if enrolled branch is disabled", async () => {
const args = {
type: "remote-experiments",
featureIds: ["cfr"],
};
const enrollment = {
slug: "exp01",
branch: {
slug: "branch01",
cfr: {
featureId: "cfr",
value: {},
},
},
};
// Nedds to match the `featureIds` value to return an enrollment
// for that feature
global.NimbusFeatures.cfr.getAllVariables.returns(
enrollment.branch.cfr.value
);
global.ExperimentAPI.getExperimentMetaData.returns({
slug: enrollment.slug,
active: true,
branch: { slug: enrollment.branch.slug },
});
global.ExperimentAPI.getAllBranches.resolves([
enrollment.branch,
{
slug: "branch02",
cfr: {
featureId: "cfr",
value: { id: "id02", trigger: { id: "openURL" } },
},
},
{
// This branch should not be loaded as it doesn't have the trigger
slug: "branch03",
cfr: {
featureId: "cfr",
value: { id: "id03" },
},
},
]);
const result = await MessageLoaderUtils.loadMessagesForProvider(args);
assert.equal(result.messages.length, 1);
assert.equal(result.messages[0].id, "id02");
assert.equal(result.messages[0].experimentSlug, "exp01");
assert.equal(result.messages[0].branchSlug, "branch02");
assert.deepEqual(result.messages[0].forReachEvent, {
sent: false,
group: "cfr",
});
});
});
describe("#_remoteSettingsLoader", () => {
let provider;
let spy;
beforeEach(() => {
provider = {
id: "cfr",
collection: "cfr",
};
sandbox
.stub(MessageLoaderUtils, "_getRemoteSettingsMessages")
.resolves([{ id: "message_1" }]);
spy = sandbox.spy();
global.Downloader.prototype.downloadToDisk = spy;
});
it("should be called with the expected dir path", async () => {
const dlSpy = sandbox.spy(global, "Downloader");
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "en-US");
await MessageLoaderUtils._remoteSettingsLoader(provider, {});
assert.calledWith(
dlSpy,
"main",
"ms-language-packs",
"browser",
"newtab"
);
});
it("should allow fetch for known locales", async () => {
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "en-US");
await MessageLoaderUtils._remoteSettingsLoader(provider, {});
assert.calledOnce(spy);
});
it("should fallback to 'en-US' for locale 'und' ", async () => {
sandbox.stub(global.Services.locale, "appLocaleAsBCP47").get(() => "und");
const getRecordSpy = sandbox.spy(
global.KintoHttpClient.prototype,
"getRecord"
);
await MessageLoaderUtils._remoteSettingsLoader(provider, {});
assert.ok(getRecordSpy.args[0][0].includes("en-US"));
assert.calledOnce(spy);
});
it("should fallback to 'ja-JP-mac' for locale 'ja-JP-macos'", async () => {
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "ja-JP-macos");
const getRecordSpy = sandbox.spy(
global.KintoHttpClient.prototype,
"getRecord"
);
await MessageLoaderUtils._remoteSettingsLoader(provider, {});
assert.ok(getRecordSpy.args[0][0].includes("ja-JP-mac"));
assert.calledOnce(spy);
});
it("should not allow fetch for unsupported locales", async () => {
sandbox
.stub(global.Services.locale, "appLocaleAsBCP47")
.get(() => "unkown");
await MessageLoaderUtils._remoteSettingsLoader(provider, {});
assert.notCalled(spy);
});
});
describe("#resetMessageState", () => {
it("should reset all message impressions", async () => {
await Router.setState({
messages: [{ id: "1" }, { id: "2" }],
});
await Router.setState({
messageImpressions: { 1: [0, 1, 2], 2: [0, 1, 2] },
}); // Add impressions for test messages
let impressions = Object.values(Router.state.messageImpressions);
assert.equal(impressions.filter(i => i.length).length, 2); // Both messages have impressions
Router.resetMessageState();
impressions = Object.values(Router.state.messageImpressions);
assert.isEmpty(impressions.filter(i => i.length)); // Both messages now have zero impressions
assert.calledWithExactly(Router._storage.set, "messageImpressions", {
1: [],
2: [],
});
});
});
describe("#resetGroupsState", () => {
it("should reset all group impressions", async () => {
await Router.setState({
groups: [{ id: "1" }, { id: "2" }],
});
await Router.setState({
groupImpressions: { 1: [0, 1, 2], 2: [0, 1, 2] },
}); // Add impressions for test groups
let impressions = Object.values(Router.state.groupImpressions);
assert.equal(impressions.filter(i => i.length).length, 2); // Both groups have impressions
Router.resetGroupsState();
impressions = Object.values(Router.state.groupImpressions);
assert.isEmpty(impressions.filter(i => i.length)); // Both groups now have zero impressions
assert.calledWithExactly(Router._storage.set, "groupImpressions", {
1: [],
2: [],
});
});
});
describe("#resetScreenImpressions", () => {
it("should reset all screen impressions", async () => {
await Router.setState({ screenImpressions: { 1: 1, 2: 2 } });
let impressions = Object.values(Router.state.screenImpressions);
assert.equal(impressions.filter(i => i).length, 2); // Both screens have impressions
Router.resetScreenImpressions();
impressions = Object.values(Router.state.screenImpressions);
assert.isEmpty(impressions.filter(i => i)); // Both screens now have zero impressions
assert.calledWithExactly(Router._storage.set, "screenImpressions", {});
});
});
describe("#editState", () => {
it("should update message impressions", async () => {
sandbox.stub(ASRouterPreferences, "devtoolsEnabled").get(() => true);
await Router.setState({ messages: [{ id: "1" }, { id: "2" }] });
await Router.setState({
messageImpressions: { 1: [0, 1, 2], 2: [0, 1, 2] },
});
let impressions = Object.values(Router.state.messageImpressions);
assert.equal(impressions.filter(i => i.length).length, 2); // Both messages have impressions
Router.editState("messageImpressions", {
1: [],
2: [],
3: [0, 1, 2],
});
// The original messages now have zero impressions
assert.isEmpty(Router.state.messageImpressions["1"]);
assert.isEmpty(Router.state.messageImpressions["2"]);
// A new impression array was added for the new message
assert.equal(Router.state.messageImpressions["3"].length, 3);
assert.calledWithExactly(Router._storage.set, "messageImpressions", {
1: [],
2: [],
3: [0, 1, 2],
});
});
});
});