2023-03-08 16:13:32 +01:00
|
|
|
/*
|
2024-09-09 15:57:16 +02:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2023-03-08 16:13:32 +01:00
|
|
|
Copyright 2023 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.
|
2023-03-08 16:13:32 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
import { mocked } from "jest-mock";
|
|
|
|
import { MatrixClient, Room } from "matrix-js-sdk/src/matrix";
|
|
|
|
|
2024-10-15 15:57:26 +02:00
|
|
|
import { createTestClient, flushPromises, setupAsyncStoreWithClient } from "../../test-utils";
|
|
|
|
import SettingsStore from "../../../src/settings/SettingsStore";
|
|
|
|
import { BreadcrumbsStore } from "../../../src/stores/BreadcrumbsStore";
|
|
|
|
import { Action } from "../../../src/dispatcher/actions";
|
|
|
|
import { defaultDispatcher } from "../../../src/dispatcher/dispatcher";
|
2023-03-08 16:13:32 +01:00
|
|
|
|
|
|
|
describe("BreadcrumbsStore", () => {
|
|
|
|
let store: BreadcrumbsStore;
|
|
|
|
const client: MatrixClient = createTestClient();
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.resetAllMocks();
|
|
|
|
store = BreadcrumbsStore.instance;
|
|
|
|
setupAsyncStoreWithClient(store, client);
|
|
|
|
jest.spyOn(SettingsStore, "setValue").mockImplementation(() => Promise.resolve());
|
2023-06-19 10:18:46 +02:00
|
|
|
jest.spyOn(SettingsStore, "getValue").mockReturnValue(false);
|
2023-03-08 16:13:32 +01:00
|
|
|
});
|
|
|
|
|
2023-06-19 10:18:46 +02:00
|
|
|
it("does not meet room requirements if there are not enough rooms", () => {
|
|
|
|
// We don't have enough rooms, so we don't meet requirements
|
|
|
|
mocked(client.getVisibleRooms).mockReturnValue(fakeRooms(2));
|
|
|
|
expect(store.meetsRoomRequirement).toBe(false);
|
|
|
|
});
|
2023-03-08 16:13:32 +01:00
|
|
|
|
2023-06-19 10:18:46 +02:00
|
|
|
it("meets room requirements if there are enough rooms", () => {
|
|
|
|
// We do have enough rooms to show breadcrumbs
|
|
|
|
mocked(client.getVisibleRooms).mockReturnValue(fakeRooms(25));
|
|
|
|
expect(store.meetsRoomRequirement).toBe(true);
|
2023-03-08 16:13:32 +01:00
|
|
|
});
|
|
|
|
|
2023-06-19 10:18:46 +02:00
|
|
|
describe("And the feature_dynamic_room_predecessors is enabled", () => {
|
2023-03-08 16:13:32 +01:00
|
|
|
beforeEach(() => {
|
2023-06-19 10:18:46 +02:00
|
|
|
// Turn on feature_dynamic_room_predecessors setting
|
2023-03-08 16:13:32 +01:00
|
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation(
|
2023-06-19 10:18:46 +02:00
|
|
|
(settingName) => settingName === "feature_dynamic_room_predecessors",
|
2023-03-08 16:13:32 +01:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
2023-06-19 10:18:46 +02:00
|
|
|
it("passes through the dynamic room precessors flag", () => {
|
2023-03-08 16:13:32 +01:00
|
|
|
mocked(client.getVisibleRooms).mockReturnValue(fakeRooms(25));
|
2024-10-16 17:43:07 +02:00
|
|
|
expect(store.meetsRoomRequirement).toBeTruthy();
|
2023-06-19 10:18:46 +02:00
|
|
|
expect(client.getVisibleRooms).toHaveBeenCalledWith(true);
|
|
|
|
});
|
|
|
|
});
|
2023-03-08 16:13:32 +01:00
|
|
|
|
2023-06-19 10:18:46 +02:00
|
|
|
describe("And the feature_dynamic_room_predecessors is not enabled", () => {
|
|
|
|
it("passes through the dynamic room precessors flag", () => {
|
|
|
|
mocked(client.getVisibleRooms).mockReturnValue(fakeRooms(25));
|
2024-10-16 17:43:07 +02:00
|
|
|
expect(store.meetsRoomRequirement).toBeTruthy();
|
2023-06-19 10:18:46 +02:00
|
|
|
expect(client.getVisibleRooms).toHaveBeenCalledWith(false);
|
2023-03-08 16:13:32 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("If the feature_dynamic_room_predecessors is not enabled", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockReturnValue(false);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Appends a room when you join", async () => {
|
|
|
|
// Sanity: no rooms initially
|
|
|
|
expect(store.rooms).toEqual([]);
|
|
|
|
|
|
|
|
// Given a room
|
|
|
|
const room = fakeRoom();
|
|
|
|
mocked(client.getRoom).mockReturnValue(room);
|
|
|
|
mocked(client.getRoomUpgradeHistory).mockReturnValue([]);
|
|
|
|
|
|
|
|
// When we hear that we have joined it
|
|
|
|
await dispatchJoinRoom(room.roomId);
|
|
|
|
|
|
|
|
// It is stored in the store's room list
|
|
|
|
expect(store.rooms.map((r) => r.roomId)).toEqual([room.roomId]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Replaces the old room when a newer one joins", async () => {
|
|
|
|
// Given an old room and a new room
|
|
|
|
const oldRoom = fakeRoom();
|
|
|
|
const newRoom = fakeRoom();
|
|
|
|
mocked(client.getRoom).mockImplementation((roomId) => {
|
|
|
|
if (roomId === oldRoom.roomId) return oldRoom;
|
|
|
|
return newRoom;
|
|
|
|
});
|
|
|
|
// Where the new one is a predecessor of the old one
|
|
|
|
mocked(client.getRoomUpgradeHistory).mockReturnValue([oldRoom, newRoom]);
|
|
|
|
|
|
|
|
// When we hear that we joined the old room, then the new one
|
|
|
|
await dispatchJoinRoom(oldRoom.roomId);
|
|
|
|
await dispatchJoinRoom(newRoom.roomId);
|
|
|
|
|
|
|
|
// The store only has the new one
|
|
|
|
expect(store.rooms.map((r) => r.roomId)).toEqual([newRoom.roomId]);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Passes through the dynamic predecessor setting", async () => {
|
|
|
|
// Given a room
|
|
|
|
const room = fakeRoom();
|
|
|
|
mocked(client.getRoom).mockReturnValue(room);
|
|
|
|
mocked(client.getRoomUpgradeHistory).mockReturnValue([]);
|
|
|
|
|
|
|
|
// When we signal that we have joined
|
|
|
|
await dispatchJoinRoom(room.roomId);
|
|
|
|
|
|
|
|
// We pass the value of the dynamic predecessor setting through
|
|
|
|
expect(client.getRoomUpgradeHistory).toHaveBeenCalledWith(room.roomId, false, false);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("If the feature_dynamic_room_predecessors is enabled", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
// Turn on feature_dynamic_room_predecessors setting
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation(
|
|
|
|
(settingName) => settingName === "feature_dynamic_room_predecessors",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("Passes through the dynamic predecessor setting", async () => {
|
|
|
|
// Given a room
|
|
|
|
const room = fakeRoom();
|
|
|
|
mocked(client.getRoom).mockReturnValue(room);
|
|
|
|
mocked(client.getRoomUpgradeHistory).mockReturnValue([]);
|
|
|
|
|
|
|
|
// When we signal that we have joined
|
|
|
|
await dispatchJoinRoom(room.roomId);
|
|
|
|
|
|
|
|
// We pass the value of the dynamic predecessor setting through
|
|
|
|
expect(client.getRoomUpgradeHistory).toHaveBeenCalledWith(room.roomId, false, true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a JoinRoom event via the dispatcher, and wait for it to process.
|
|
|
|
*/
|
|
|
|
async function dispatchJoinRoom(roomId: string) {
|
|
|
|
defaultDispatcher.dispatch(
|
|
|
|
{
|
|
|
|
action: Action.JoinRoom,
|
|
|
|
roomId,
|
|
|
|
metricsTrigger: null,
|
|
|
|
},
|
|
|
|
true, // synchronous dispatch
|
|
|
|
);
|
|
|
|
|
|
|
|
// Wait for event dispatch to happen
|
2023-03-08 17:10:23 +01:00
|
|
|
await flushPromises();
|
2023-03-08 16:13:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create as many fake rooms in an array as you ask for.
|
|
|
|
*/
|
2023-03-10 15:55:06 +01:00
|
|
|
function fakeRooms(howMany: number): Room[] {
|
|
|
|
const ret: Room[] = [];
|
2023-03-08 16:13:32 +01:00
|
|
|
for (let i = 0; i < howMany; i++) {
|
|
|
|
ret.push(fakeRoom());
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
let roomIdx = 0;
|
|
|
|
|
|
|
|
function fakeRoom(): Room {
|
|
|
|
roomIdx++;
|
|
|
|
return new Room(`room${roomIdx}`, client, "@user:example.com");
|
|
|
|
}
|
|
|
|
});
|