2023-04-27 01:20:02 +00:00
|
|
|
/*
|
2024-09-09 13:57:16 +00:00
|
|
|
Copyright 2024 New Vector Ltd.
|
2023-04-27 01:20:02 +00:00
|
|
|
Copyright 2023 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.
|
2023-04-27 01:20:02 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
import React from "react";
|
2024-10-14 16:11:58 +00:00
|
|
|
import { fireEvent, render, screen, waitFor, within } from "jest-matrix-react";
|
2023-04-27 01:20:02 +00:00
|
|
|
import { EventType, GuestAccess, HistoryVisibility, JoinRule, MatrixEvent, Room } from "matrix-js-sdk/src/matrix";
|
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
2024-10-24 08:34:07 +00:00
|
|
|
import { mocked } from "jest-mock";
|
2023-04-27 01:20:02 +00:00
|
|
|
|
2024-10-15 13:57:26 +00:00
|
|
|
import SecurityRoomSettingsTab from "../../../../../../../src/components/views/settings/tabs/room/SecurityRoomSettingsTab";
|
|
|
|
import MatrixClientContext from "../../../../../../../src/contexts/MatrixClientContext";
|
|
|
|
import SettingsStore from "../../../../../../../src/settings/SettingsStore";
|
2024-10-24 08:34:07 +00:00
|
|
|
import { clearAllModals, flushPromises, stubClient } from "../../../../../../test-utils";
|
2024-10-15 13:57:26 +00:00
|
|
|
import { filterBoolean } from "../../../../../../../src/utils/arrays";
|
2023-04-27 01:20:02 +00:00
|
|
|
|
|
|
|
describe("<SecurityRoomSettingsTab />", () => {
|
|
|
|
const userId = "@alice:server.org";
|
2024-10-24 08:34:07 +00:00
|
|
|
const client = mocked(stubClient());
|
2023-04-27 01:20:02 +00:00
|
|
|
const roomId = "!room:server.org";
|
|
|
|
|
|
|
|
const getComponent = (room: Room, closeSettingsFn = jest.fn()) =>
|
|
|
|
render(<SecurityRoomSettingsTab room={room} closeSettingsFn={closeSettingsFn} />, {
|
|
|
|
wrapper: ({ children }) => (
|
|
|
|
<MatrixClientContext.Provider value={client}>{children}</MatrixClientContext.Provider>
|
|
|
|
),
|
|
|
|
});
|
|
|
|
|
|
|
|
const setRoomStateEvents = (
|
|
|
|
room: Room,
|
|
|
|
joinRule?: JoinRule,
|
|
|
|
guestAccess?: GuestAccess,
|
|
|
|
history?: HistoryVisibility,
|
|
|
|
): void => {
|
|
|
|
const events = filterBoolean<MatrixEvent>([
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomCreate,
|
|
|
|
content: { version: "test" },
|
|
|
|
sender: userId,
|
|
|
|
state_key: "",
|
|
|
|
room_id: room.roomId,
|
|
|
|
}),
|
|
|
|
guestAccess &&
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomGuestAccess,
|
|
|
|
content: { guest_access: guestAccess },
|
|
|
|
sender: userId,
|
|
|
|
state_key: "",
|
|
|
|
room_id: room.roomId,
|
|
|
|
}),
|
|
|
|
history &&
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomHistoryVisibility,
|
|
|
|
content: { history_visibility: history },
|
|
|
|
sender: userId,
|
|
|
|
state_key: "",
|
|
|
|
room_id: room.roomId,
|
|
|
|
}),
|
|
|
|
joinRule &&
|
|
|
|
new MatrixEvent({
|
|
|
|
type: EventType.RoomJoinRules,
|
|
|
|
content: { join_rule: joinRule },
|
|
|
|
sender: userId,
|
|
|
|
state_key: "",
|
|
|
|
room_id: room.roomId,
|
|
|
|
}),
|
|
|
|
]);
|
|
|
|
|
|
|
|
room.currentState.setStateEvents(events);
|
|
|
|
};
|
|
|
|
|
2023-05-02 07:46:32 +00:00
|
|
|
beforeEach(async () => {
|
2023-04-27 01:20:02 +00:00
|
|
|
client.sendStateEvent.mockReset().mockResolvedValue({ event_id: "test" });
|
|
|
|
client.isRoomEncrypted.mockReturnValue(false);
|
2023-06-21 23:17:30 +00:00
|
|
|
client.getClientWellKnown.mockReturnValue(undefined);
|
2023-04-27 01:20:02 +00:00
|
|
|
jest.spyOn(SettingsStore, "getValue").mockRestore();
|
2023-05-02 07:46:32 +00:00
|
|
|
|
|
|
|
await clearAllModals();
|
2023-04-27 01:20:02 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
describe("join rule", () => {
|
|
|
|
it("warns when trying to make an encrypted room public", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-04-27 01:20:02 +00:00
|
|
|
setRoomStateEvents(room, JoinRule.Invite);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
await waitFor(() => expect(screen.getByLabelText("Encrypted")).toBeChecked());
|
2023-04-27 01:20:02 +00:00
|
|
|
fireEvent.click(screen.getByLabelText("Public"));
|
|
|
|
|
|
|
|
const modal = await screen.findByRole("dialog");
|
|
|
|
|
|
|
|
expect(modal).toMatchSnapshot();
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByText("Cancel"));
|
|
|
|
|
|
|
|
// join rule not updated
|
|
|
|
expect(screen.getByLabelText("Private (invite only)").hasAttribute("checked")).toBeTruthy();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("updates join rule", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Invite);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Public"));
|
|
|
|
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
room.roomId,
|
|
|
|
EventType.RoomJoinRules,
|
|
|
|
{
|
|
|
|
join_rule: JoinRule.Public,
|
|
|
|
},
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("handles error when updating join rule fails", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
client.sendStateEvent.mockRejectedValue("oups");
|
|
|
|
setRoomStateEvents(room, JoinRule.Invite);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Public"));
|
|
|
|
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
const dialog = await screen.findByRole("dialog");
|
|
|
|
|
|
|
|
expect(dialog).toMatchSnapshot();
|
|
|
|
|
|
|
|
fireEvent.click(within(dialog).getByText("OK"));
|
|
|
|
});
|
|
|
|
|
|
|
|
it("displays advanced section toggle when join rule is public", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Public);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByText("Show advanced")).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not display advanced section toggle when join rule is not public", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Invite);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.queryByText("Show advanced")).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("guest access", () => {
|
|
|
|
it("uses forbidden by default when room has no guest access event", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Public);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByText("Show advanced"));
|
|
|
|
|
|
|
|
expect(screen.getByLabelText("Enable guest access").getAttribute("aria-checked")).toBe("false");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("updates guest access on toggle", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Public);
|
|
|
|
getComponent(room);
|
|
|
|
fireEvent.click(screen.getByText("Show advanced"));
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Enable guest access"));
|
|
|
|
|
|
|
|
// toggle set immediately
|
|
|
|
expect(screen.getByLabelText("Enable guest access").getAttribute("aria-checked")).toBe("true");
|
|
|
|
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
room.roomId,
|
|
|
|
EventType.RoomGuestAccess,
|
|
|
|
{ guest_access: GuestAccess.CanJoin },
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("logs error and resets state when updating guest access fails", async () => {
|
|
|
|
client.sendStateEvent.mockRejectedValue("oups");
|
|
|
|
jest.spyOn(logger, "error").mockImplementation(() => {});
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room, JoinRule.Public, GuestAccess.CanJoin);
|
|
|
|
getComponent(room);
|
|
|
|
fireEvent.click(screen.getByText("Show advanced"));
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Enable guest access"));
|
|
|
|
|
|
|
|
// toggle set immediately
|
|
|
|
expect(screen.getByLabelText("Enable guest access").getAttribute("aria-checked")).toBe("false");
|
|
|
|
|
|
|
|
await flushPromises();
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalled();
|
|
|
|
expect(logger.error).toHaveBeenCalledWith("oups");
|
|
|
|
|
|
|
|
// toggle reset to old value
|
|
|
|
expect(screen.getByLabelText("Enable guest access").getAttribute("aria-checked")).toBe("true");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("history visibility", () => {
|
|
|
|
it("does not render section when RoomHistorySettings feature is disabled", () => {
|
|
|
|
jest.spyOn(SettingsStore, "getValue").mockReturnValue(false);
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.queryByText("Who can read history")).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("uses shared as default history visibility when no state event found", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByText("Who can read history?").parentElement).toMatchSnapshot();
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Shared)).toBeChecked();
|
|
|
|
});
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
it("does not render world readable option when room is encrypted", async () => {
|
2023-04-27 01:20:02 +00:00
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-04-27 01:20:02 +00:00
|
|
|
setRoomStateEvents(room);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
await waitFor(() =>
|
|
|
|
expect(screen.queryByDisplayValue(HistoryVisibility.WorldReadable)).not.toBeInTheDocument(),
|
|
|
|
);
|
2023-04-27 01:20:02 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("renders world readable option when room is encrypted and history is already set to world readable", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-04-27 01:20:02 +00:00
|
|
|
setRoomStateEvents(room, undefined, undefined, HistoryVisibility.WorldReadable);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.WorldReadable)).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("updates history visibility", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByDisplayValue(HistoryVisibility.Invited));
|
|
|
|
|
|
|
|
// toggle updated immediately
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Invited)).toBeChecked();
|
|
|
|
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
room.roomId,
|
|
|
|
EventType.RoomHistoryVisibility,
|
|
|
|
{
|
|
|
|
history_visibility: HistoryVisibility.Invited,
|
|
|
|
},
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("handles error when updating history visibility", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
client.sendStateEvent.mockRejectedValue("oups");
|
|
|
|
jest.spyOn(logger, "error").mockImplementation(() => {});
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByDisplayValue(HistoryVisibility.Invited));
|
|
|
|
|
|
|
|
// toggle updated immediately
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Invited)).toBeChecked();
|
|
|
|
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// reset to before updated value
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Shared)).toBeChecked();
|
|
|
|
expect(logger.error).toHaveBeenCalledWith("oups");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("encryption", () => {
|
2024-10-24 08:34:07 +00:00
|
|
|
it("displays encryption as enabled", async () => {
|
2023-04-27 01:20:02 +00:00
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-04-27 01:20:02 +00:00
|
|
|
setRoomStateEvents(room);
|
|
|
|
getComponent(room);
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
await waitFor(() => expect(screen.getByLabelText("Encrypted")).toBeChecked());
|
2023-04-27 01:20:02 +00:00
|
|
|
// can't disable encryption once enabled
|
|
|
|
expect(screen.getByLabelText("Encrypted").getAttribute("aria-disabled")).toEqual("true");
|
|
|
|
});
|
|
|
|
|
|
|
|
it("asks users to confirm when setting room to encrypted", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room);
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByLabelText("Encrypted")).not.toBeChecked();
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Encrypted"));
|
|
|
|
|
|
|
|
const dialog = await screen.findByRole("dialog");
|
|
|
|
|
|
|
|
fireEvent.click(within(dialog).getByText("Cancel"));
|
|
|
|
|
|
|
|
expect(client.sendStateEvent).not.toHaveBeenCalled();
|
|
|
|
expect(screen.getByLabelText("Encrypted")).not.toBeChecked();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("enables encryption after confirmation", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room);
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByLabelText("Encrypted")).not.toBeChecked();
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByLabelText("Encrypted"));
|
|
|
|
|
|
|
|
const dialog = await screen.findByRole("dialog");
|
|
|
|
|
2023-05-02 07:46:32 +00:00
|
|
|
expect(within(dialog).getByText("Enable encryption?")).toBeInTheDocument();
|
2023-04-27 01:20:02 +00:00
|
|
|
fireEvent.click(within(dialog).getByText("OK"));
|
|
|
|
|
2023-05-04 10:40:49 +00:00
|
|
|
await waitFor(() =>
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalledWith(room.roomId, EventType.RoomEncryption, {
|
|
|
|
algorithm: "m.megolm.v1.aes-sha2",
|
|
|
|
}),
|
|
|
|
);
|
2023-04-27 01:20:02 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("renders world readable option when room is encrypted and history is already set to world readable", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-04-27 01:20:02 +00:00
|
|
|
setRoomStateEvents(room, undefined, undefined, HistoryVisibility.WorldReadable);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.WorldReadable)).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("updates history visibility", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByDisplayValue(HistoryVisibility.Invited));
|
|
|
|
|
|
|
|
// toggle updated immediately
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Invited)).toBeChecked();
|
|
|
|
|
|
|
|
expect(client.sendStateEvent).toHaveBeenCalledWith(
|
|
|
|
room.roomId,
|
|
|
|
EventType.RoomHistoryVisibility,
|
|
|
|
{
|
|
|
|
history_visibility: HistoryVisibility.Invited,
|
|
|
|
},
|
|
|
|
"",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("handles error when updating history visibility", async () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
client.sendStateEvent.mockRejectedValue("oups");
|
|
|
|
jest.spyOn(logger, "error").mockImplementation(() => {});
|
|
|
|
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
fireEvent.click(screen.getByDisplayValue(HistoryVisibility.Invited));
|
|
|
|
|
|
|
|
// toggle updated immediately
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Invited)).toBeChecked();
|
|
|
|
|
|
|
|
await flushPromises();
|
|
|
|
|
|
|
|
// reset to before updated value
|
|
|
|
expect(screen.getByDisplayValue(HistoryVisibility.Shared)).toBeChecked();
|
|
|
|
expect(logger.error).toHaveBeenCalledWith("oups");
|
|
|
|
});
|
2023-06-21 23:17:30 +00:00
|
|
|
|
|
|
|
describe("when encryption is force disabled by e2ee well-known config", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
client.getClientWellKnown.mockReturnValue({
|
|
|
|
"io.element.e2ee": {
|
|
|
|
force_disable: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
it("displays encrypted rooms as encrypted", async () => {
|
2023-06-21 23:17:30 +00:00
|
|
|
// rooms that are already encrypted still show encrypted
|
|
|
|
const room = new Room(roomId, client, userId);
|
2024-10-24 08:34:07 +00:00
|
|
|
jest.spyOn(client.getCrypto()!, "isEncryptionEnabledInRoom").mockResolvedValue(true);
|
2023-06-21 23:17:30 +00:00
|
|
|
setRoomStateEvents(room);
|
|
|
|
getComponent(room);
|
|
|
|
|
2024-10-24 08:34:07 +00:00
|
|
|
await waitFor(() => expect(screen.getByLabelText("Encrypted")).toBeChecked());
|
2023-06-21 23:17:30 +00:00
|
|
|
expect(screen.getByLabelText("Encrypted").getAttribute("aria-disabled")).toEqual("true");
|
|
|
|
expect(screen.getByText("Once enabled, encryption cannot be disabled.")).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("displays unencrypted rooms with toggle disabled", () => {
|
|
|
|
const room = new Room(roomId, client, userId);
|
|
|
|
setRoomStateEvents(room);
|
|
|
|
getComponent(room);
|
|
|
|
|
|
|
|
expect(screen.getByLabelText("Encrypted")).not.toBeChecked();
|
|
|
|
expect(screen.getByLabelText("Encrypted").getAttribute("aria-disabled")).toEqual("true");
|
|
|
|
expect(screen.queryByText("Once enabled, encryption cannot be disabled.")).not.toBeInTheDocument();
|
|
|
|
expect(screen.getByText("Your server requires encryption to be disabled.")).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
});
|
2023-04-27 01:20:02 +00:00
|
|
|
});
|
|
|
|
});
|