2022-09-19 06:42:29 +00:00
|
|
|
/*
|
2024-09-09 13:57:16 +00:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2022-09-19 06:42:29 +00:00
|
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
|
2024-09-09 13:57:16 +00:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
Please see LICENSE files in the repository root for full details.
|
2022-09-19 06:42:29 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import React from "react";
|
2024-10-14 16:11:58 +00:00
|
|
|
import { fireEvent, getByRole, render, RenderResult, screen, waitFor } from "jest-matrix-react";
|
2024-03-18 14:40:52 +00:00
|
|
|
import { MatrixClient, EventType, MatrixEvent, Room, RoomMember, ISendEventResponse } from "matrix-js-sdk/src/matrix";
|
|
|
|
import { KnownMembership } from "matrix-js-sdk/src/types";
|
2022-12-01 09:49:46 +00:00
|
|
|
import { mocked } from "jest-mock";
|
2024-03-08 11:58:36 +00:00
|
|
|
import { defer } from "matrix-js-sdk/src/utils";
|
2024-03-19 13:45:23 +00:00
|
|
|
import userEvent from "@testing-library/user-event";
|
2022-09-19 06:42:29 +00:00
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
import RolesRoomSettingsTab from "../../../../../../../src/components/views/settings/tabs/room/RolesRoomSettingsTab";
|
|
|
|
import { mkStubRoom, withClientContextRenderOptions, stubClient } from "../../../../../../test-utils";
|
|
|
|
import { MatrixClientPeg } from "../../../../../../../src/MatrixClientPeg";
|
|
|
|
import { VoiceBroadcastInfoEventType } from "../../../../../../../src/voice-broadcast";
|
|
|
|
import SettingsStore from "../../../../../../../src/settings/SettingsStore";
|
|
|
|
import { ElementCall } from "../../../../../../../src/models/Call";
|
2022-09-19 06:42:29 +00:00
|
|
|
|
|
|
|
describe("RolesRoomSettingsTab", () => {
|
2023-04-27 01:20:02 +00:00
|
|
|
const userId = "@alice:server.org";
|
2022-09-19 06:42:29 +00:00
|
|
|
const roomId = "!room:example.com";
|
|
|
|
let cli: MatrixClient;
|
2022-12-01 09:49:46 +00:00
|
|
|
let room: Room;
|
2022-09-19 06:42:29 +00:00
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
const renderTab = async (propRoom: Room = room): Promise<RenderResult> => {
|
|
|
|
const renderResult = render(<RolesRoomSettingsTab room={propRoom} />, withClientContextRenderOptions(cli));
|
|
|
|
// Wait for the tab to be ready
|
|
|
|
await waitFor(() => expect(screen.getByText("Permissions")).toBeInTheDocument());
|
|
|
|
return renderResult;
|
2022-10-07 18:10:17 +00:00
|
|
|
};
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
const getVoiceBroadcastsSelect = async (): Promise<Element> => {
|
|
|
|
return (await renderTab()).container.querySelector("select[label='Voice broadcasts']")!;
|
2022-09-19 06:42:29 +00:00
|
|
|
};
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
const getVoiceBroadcastsSelectedOption = async (): Promise<Element> => {
|
|
|
|
return (await renderTab()).container.querySelector("select[label='Voice broadcasts'] option:checked")!;
|
2022-09-19 06:42:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
stubClient();
|
2023-06-05 17:12:23 +00:00
|
|
|
cli = MatrixClientPeg.safeGet();
|
2022-12-01 09:49:46 +00:00
|
|
|
room = mkStubRoom(roomId, "test room", cli);
|
|
|
|
});
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
it("should allow an Admin to demote themselves but not others", async () => {
|
2022-12-01 09:49:46 +00:00
|
|
|
mocked(cli.getRoom).mockReturnValue(room);
|
|
|
|
// @ts-ignore - mocked doesn't support overloads properly
|
|
|
|
mocked(room.currentState.getStateEvents).mockImplementation((type, key) => {
|
|
|
|
if (key === undefined) return [] as MatrixEvent[];
|
|
|
|
if (type === "m.room.power_levels") {
|
|
|
|
return new MatrixEvent({
|
|
|
|
sender: "@sender:server",
|
|
|
|
room_id: roomId,
|
|
|
|
type: "m.room.power_levels",
|
|
|
|
state_key: "",
|
|
|
|
content: {
|
|
|
|
users: {
|
2023-02-13 17:01:43 +00:00
|
|
|
[cli.getUserId()!]: 100,
|
2022-12-01 09:49:46 +00:00
|
|
|
"@admin:server": 100,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
mocked(room.currentState.mayClientSendStateEvent).mockReturnValue(true);
|
2024-11-20 14:27:09 +00:00
|
|
|
const { container } = await renderTab();
|
2022-12-01 09:49:46 +00:00
|
|
|
|
|
|
|
expect(container.querySelector(`[placeholder="${cli.getUserId()}"]`)).not.toBeDisabled();
|
|
|
|
expect(container.querySelector(`[placeholder="@admin:server"]`)).toBeDisabled();
|
2022-09-19 06:42:29 +00:00
|
|
|
});
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
it("should initially show »Moderator« permission for »Voice broadcasts«", async () => {
|
|
|
|
expect((await getVoiceBroadcastsSelectedOption()).textContent).toBe("Moderator");
|
2022-09-19 06:42:29 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("when setting »Default« permission for »Voice broadcasts«", () => {
|
2024-11-20 14:27:09 +00:00
|
|
|
beforeEach(async () => {
|
|
|
|
fireEvent.change(await getVoiceBroadcastsSelect(), {
|
2022-09-19 06:42:29 +00:00
|
|
|
target: { value: 0 },
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should update the power levels", () => {
|
|
|
|
expect(cli.sendStateEvent).toHaveBeenCalledWith(roomId, EventType.RoomPowerLevels, {
|
|
|
|
events: {
|
|
|
|
[VoiceBroadcastInfoEventType]: 0,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2022-10-07 18:10:17 +00:00
|
|
|
|
|
|
|
describe("Element Call", () => {
|
|
|
|
const setGroupCallsEnabled = (val: boolean): void => {
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((name: string) => {
|
|
|
|
if (name === "feature_group_calls") return val;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-12-01 09:49:46 +00:00
|
|
|
const getStartCallSelect = (tab: RenderResult): HTMLElement => {
|
2023-02-13 17:01:43 +00:00
|
|
|
return tab.container.querySelector("select[label='Start Element Call calls']")!;
|
2022-10-07 18:10:17 +00:00
|
|
|
};
|
|
|
|
|
2022-12-01 09:49:46 +00:00
|
|
|
const getStartCallSelectedOption = (tab: RenderResult): HTMLElement => {
|
2023-02-13 17:01:43 +00:00
|
|
|
return tab.container.querySelector("select[label='Start Element Call calls'] option:checked")!;
|
2022-10-07 18:10:17 +00:00
|
|
|
};
|
|
|
|
|
2022-12-01 09:49:46 +00:00
|
|
|
const getJoinCallSelect = (tab: RenderResult): HTMLElement => {
|
2023-02-13 17:01:43 +00:00
|
|
|
return tab.container.querySelector("select[label='Join Element Call calls']")!;
|
2022-10-07 18:10:17 +00:00
|
|
|
};
|
|
|
|
|
2022-12-01 09:49:46 +00:00
|
|
|
const getJoinCallSelectedOption = (tab: RenderResult): HTMLElement => {
|
2023-02-13 17:01:43 +00:00
|
|
|
return tab.container.querySelector("select[label='Join Element Call calls'] option:checked")!;
|
2022-10-07 18:10:17 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
describe("Element Call enabled", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
setGroupCallsEnabled(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("Join Element calls", () => {
|
2024-11-20 14:27:09 +00:00
|
|
|
it("defaults to moderator for joining calls", async () => {
|
|
|
|
expect(getJoinCallSelectedOption(await renderTab())?.textContent).toBe("Moderator");
|
2022-10-07 18:10:17 +00:00
|
|
|
});
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
it("can change joining calls power level", async () => {
|
|
|
|
const tab = await renderTab();
|
2022-10-07 18:10:17 +00:00
|
|
|
|
|
|
|
fireEvent.change(getJoinCallSelect(tab), {
|
|
|
|
target: { value: 0 },
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(getJoinCallSelectedOption(tab)?.textContent).toBe("Default");
|
|
|
|
expect(cli.sendStateEvent).toHaveBeenCalledWith(roomId, EventType.RoomPowerLevels, {
|
|
|
|
events: {
|
|
|
|
[ElementCall.MEMBER_EVENT_TYPE.name]: 0,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("Start Element calls", () => {
|
2024-11-20 14:27:09 +00:00
|
|
|
it("defaults to moderator for starting calls", async () => {
|
|
|
|
expect(getStartCallSelectedOption(await renderTab())?.textContent).toBe("Moderator");
|
2022-10-07 18:10:17 +00:00
|
|
|
});
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
it("can change starting calls power level", async () => {
|
|
|
|
const tab = await renderTab();
|
2022-10-07 18:10:17 +00:00
|
|
|
|
|
|
|
fireEvent.change(getStartCallSelect(tab), {
|
|
|
|
target: { value: 0 },
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(getStartCallSelectedOption(tab)?.textContent).toBe("Default");
|
|
|
|
expect(cli.sendStateEvent).toHaveBeenCalledWith(roomId, EventType.RoomPowerLevels, {
|
|
|
|
events: {
|
|
|
|
[ElementCall.CALL_EVENT_TYPE.name]: 0,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
it("hides when group calls disabled", async () => {
|
2022-10-07 18:10:17 +00:00
|
|
|
setGroupCallsEnabled(false);
|
|
|
|
|
2024-11-20 14:27:09 +00:00
|
|
|
const tab = await renderTab();
|
2022-10-07 18:10:17 +00:00
|
|
|
|
|
|
|
expect(getStartCallSelect(tab)).toBeFalsy();
|
|
|
|
expect(getStartCallSelectedOption(tab)).toBeFalsy();
|
|
|
|
|
|
|
|
expect(getJoinCallSelect(tab)).toBeFalsy();
|
|
|
|
expect(getJoinCallSelectedOption(tab)).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
2023-04-27 01:20:02 +00:00
|
|
|
|
|
|
|
describe("Banned users", () => {
|
|
|
|
it("should not render banned section when no banned users", () => {
|
|
|
|
const room = new Room(roomId, cli, userId);
|
|
|
|
renderTab(room);
|
|
|
|
|
|
|
|
expect(screen.queryByText("Banned users")).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders banned users", () => {
|
|
|
|
const bannedMember = new RoomMember(roomId, "@bob:server.org");
|
|
|
|
bannedMember.setMembershipEvent(
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomMember,
|
|
|
|
content: {
|
2024-03-12 14:52:54 +00:00
|
|
|
membership: KnownMembership.Ban,
|
2023-04-27 01:20:02 +00:00
|
|
|
reason: "just testing",
|
|
|
|
},
|
|
|
|
sender: userId,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
const room = new Room(roomId, cli, userId);
|
|
|
|
jest.spyOn(room, "getMembersWithMembership").mockReturnValue([bannedMember]);
|
|
|
|
renderTab(room);
|
|
|
|
|
|
|
|
expect(screen.getByText("Banned users").parentElement).toMatchSnapshot();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("uses banners display name when available", () => {
|
|
|
|
const bannedMember = new RoomMember(roomId, "@bob:server.org");
|
|
|
|
const senderMember = new RoomMember(roomId, "@alice:server.org");
|
|
|
|
senderMember.name = "Alice";
|
|
|
|
bannedMember.setMembershipEvent(
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomMember,
|
|
|
|
content: {
|
2024-03-12 14:52:54 +00:00
|
|
|
membership: KnownMembership.Ban,
|
2023-04-27 01:20:02 +00:00
|
|
|
reason: "just testing",
|
|
|
|
},
|
|
|
|
sender: userId,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
const room = new Room(roomId, cli, userId);
|
|
|
|
jest.spyOn(room, "getMembersWithMembership").mockReturnValue([bannedMember]);
|
|
|
|
jest.spyOn(room, "getMember").mockReturnValue(senderMember);
|
|
|
|
renderTab(room);
|
|
|
|
|
|
|
|
expect(screen.getByTitle("Banned by Alice")).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
});
|
2024-03-08 11:58:36 +00:00
|
|
|
|
|
|
|
it("should roll back power level change on error", async () => {
|
|
|
|
const deferred = defer<ISendEventResponse>();
|
|
|
|
mocked(cli.sendStateEvent).mockReturnValue(deferred.promise);
|
|
|
|
mocked(cli.getRoom).mockReturnValue(room);
|
|
|
|
// @ts-ignore - mocked doesn't support overloads properly
|
|
|
|
mocked(room.currentState.getStateEvents).mockImplementation((type, key) => {
|
|
|
|
if (key === undefined) return [] as MatrixEvent[];
|
|
|
|
if (type === "m.room.power_levels") {
|
|
|
|
return new MatrixEvent({
|
|
|
|
sender: "@sender:server",
|
|
|
|
room_id: roomId,
|
|
|
|
type: "m.room.power_levels",
|
|
|
|
state_key: "",
|
|
|
|
content: {
|
|
|
|
users: {
|
|
|
|
[cli.getUserId()!]: 100,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
});
|
|
|
|
mocked(room.currentState.mayClientSendStateEvent).mockReturnValue(true);
|
2024-11-20 14:27:09 +00:00
|
|
|
const { container } = await renderTab();
|
2024-03-08 11:58:36 +00:00
|
|
|
|
|
|
|
const selector = container.querySelector(`[placeholder="${cli.getUserId()}"]`)!;
|
|
|
|
fireEvent.change(selector, { target: { value: "50" } });
|
|
|
|
expect(selector).toHaveValue("50");
|
|
|
|
|
2024-03-19 13:45:23 +00:00
|
|
|
// Get the apply button of the privileged user section and click on it
|
|
|
|
const privilegedUsersSection = screen.getByRole("group", { name: "Privileged Users" });
|
|
|
|
const applyButton = getByRole(privilegedUsersSection, "button", { name: "Apply" });
|
|
|
|
await userEvent.click(applyButton);
|
|
|
|
|
2024-03-08 11:58:36 +00:00
|
|
|
deferred.reject("Error");
|
|
|
|
await waitFor(() => expect(selector).toHaveValue("100"));
|
|
|
|
});
|
2022-09-19 06:42:29 +00:00
|
|
|
});
|