2024-04-08 10:43:59 +02:00
|
|
|
/*
|
2024-09-09 15:57:16 +02:00
|
|
|
* Copyright 2024 New Vector Ltd.
|
2024-04-08 10:43:59 +02:00
|
|
|
* Copyright 2024 The Matrix.org Foundation C.I.C.
|
|
|
|
*
|
2024-09-09 15:57:16 +02:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
* Please see LICENSE files in the repository root for full details.
|
2024-04-08 10:43:59 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
import { mocked } from "jest-mock";
|
|
|
|
|
|
|
|
import SettingsStore, { CallbackFn } from "../../src/settings/SettingsStore";
|
|
|
|
import { Feature, ReleaseAnnouncementStore } from "../../src/stores/ReleaseAnnouncementStore";
|
|
|
|
import { SettingLevel } from "../../src/settings/SettingLevel";
|
|
|
|
|
|
|
|
jest.mock("../../src/settings/SettingsStore");
|
|
|
|
|
|
|
|
describe("ReleaseAnnouncementStore", () => {
|
|
|
|
let releaseAnnouncementStore: ReleaseAnnouncementStore;
|
|
|
|
// Local settings
|
|
|
|
// Instead of using the real SettingsStore, we use a local settings object
|
|
|
|
// to avoid side effects between tests
|
|
|
|
let settings: Record<string, any> = {};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
// Default settings
|
|
|
|
settings = {
|
|
|
|
feature_release_announcement: true,
|
|
|
|
releaseAnnouncementData: {},
|
|
|
|
};
|
|
|
|
const watchCallbacks: Array<CallbackFn> = [];
|
|
|
|
|
|
|
|
mocked(SettingsStore.getValue).mockImplementation((setting: string) => {
|
|
|
|
return settings[setting];
|
|
|
|
});
|
|
|
|
mocked(SettingsStore.setValue).mockImplementation(
|
|
|
|
(settingName: string, roomId: string | null, level: SettingLevel, value: any): Promise<void> => {
|
|
|
|
settings[settingName] = value;
|
|
|
|
// we don't care about the parameters, just call the callbacks
|
|
|
|
// @ts-ignore
|
|
|
|
watchCallbacks.forEach((cb) => cb());
|
|
|
|
return Promise.resolve();
|
|
|
|
},
|
|
|
|
);
|
|
|
|
mocked(SettingsStore.isLevelSupported).mockReturnValue(true);
|
|
|
|
mocked(SettingsStore.canSetValue).mockReturnValue(true);
|
|
|
|
mocked(SettingsStore.watchSetting).mockImplementation((settingName: string, roomId: null, callback: any) => {
|
|
|
|
watchCallbacks.push(callback);
|
|
|
|
return "watcherId";
|
|
|
|
});
|
|
|
|
|
|
|
|
releaseAnnouncementStore = new ReleaseAnnouncementStore();
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Disables the release announcement feature.
|
|
|
|
*/
|
|
|
|
function disableReleaseAnnouncement() {
|
|
|
|
settings["feature_release_announcement"] = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Listens to the next release announcement change event.
|
|
|
|
*/
|
|
|
|
function listenReleaseAnnouncementChanged() {
|
|
|
|
return new Promise<Feature | null>((resolve) =>
|
|
|
|
releaseAnnouncementStore.once("releaseAnnouncementChanged", resolve),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
it("should be a singleton", () => {
|
|
|
|
expect(ReleaseAnnouncementStore.instance).toBeDefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return null when the release announcement is disabled", async () => {
|
|
|
|
disableReleaseAnnouncement();
|
|
|
|
|
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBeNull();
|
|
|
|
|
|
|
|
// Wait for the next release announcement change event
|
|
|
|
const promise = listenReleaseAnnouncementChanged();
|
|
|
|
// Call the next release announcement
|
|
|
|
// because the release announcement is disabled, the next release announcement should be null
|
|
|
|
await releaseAnnouncementStore.nextReleaseAnnouncement();
|
|
|
|
expect(await promise).toBeNull();
|
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBeNull();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should return the next feature when the next release announcement is called", async () => {
|
|
|
|
// Sanity check
|
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBe("threadsActivityCentre");
|
|
|
|
|
2024-09-18 11:26:17 +02:00
|
|
|
let promise = listenReleaseAnnouncementChanged();
|
|
|
|
await releaseAnnouncementStore.nextReleaseAnnouncement();
|
|
|
|
|
|
|
|
expect(await promise).toBe("pinningMessageList");
|
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBe("pinningMessageList");
|
|
|
|
|
|
|
|
promise = listenReleaseAnnouncementChanged();
|
2024-07-23 13:46:24 +02:00
|
|
|
await releaseAnnouncementStore.nextReleaseAnnouncement();
|
|
|
|
|
2024-04-08 10:43:59 +02:00
|
|
|
expect(await promise).toBeNull();
|
2024-09-11 11:17:44 +02:00
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBeNull();
|
2024-04-08 10:43:59 +02:00
|
|
|
|
|
|
|
const secondStore = new ReleaseAnnouncementStore();
|
2024-07-23 13:46:24 +02:00
|
|
|
// All the release announcements have been viewed, so it should be updated in the store account
|
2024-04-08 10:43:59 +02:00
|
|
|
// The release announcement viewing states should be share among all instances (devices in the same account)
|
|
|
|
expect(secondStore.getReleaseAnnouncement()).toBeNull();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should listen to release announcement data changes in the store", async () => {
|
|
|
|
const secondStore = new ReleaseAnnouncementStore();
|
|
|
|
expect(secondStore.getReleaseAnnouncement()).toBe("threadsActivityCentre");
|
|
|
|
|
|
|
|
const promise = listenReleaseAnnouncementChanged();
|
|
|
|
await secondStore.nextReleaseAnnouncement();
|
|
|
|
|
2024-09-18 11:26:17 +02:00
|
|
|
expect(await promise).toBe("pinningMessageList");
|
|
|
|
expect(releaseAnnouncementStore.getReleaseAnnouncement()).toBe("pinningMessageList");
|
2024-04-08 10:43:59 +02:00
|
|
|
});
|
|
|
|
});
|