You've already forked element-web
mirror of
https://github.com/element-hq/element-web.git
synced 2025-08-08 03:42:14 +03:00
* remove incorrect check for cross-signing SETUP_ENCRYPTION tries to set up everything (4S, cross-signing and key backup), rather than just setting up encryption, as its name would imply. crossSigningReady == false happens when the user's device isn't verified, so it should trigger VERIFY_THIS_SESSION rather than SETUP_ENCRYPTION * reorder conditions in allSystemsReady to match the order in the if statements * explicitly handle secrets missing from 4S rather than falling back to the SETUP_ENCRYPTION catch-all. Also, remove SETUP_ENCRYPTION since it is no longer used. * convert button handlers to switch statements for consistency (almost) all the other functions that use make decisions based on Kind use switch statements * update i18n (remove obsolete string)
180 lines
7.2 KiB
TypeScript
180 lines
7.2 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import React from "react";
|
|
import { render, screen } from "jest-matrix-react";
|
|
import userEvent from "@testing-library/user-event";
|
|
|
|
import * as SecurityManager from "../../../src/SecurityManager";
|
|
import ToastContainer from "../../../src/components/structures/ToastContainer";
|
|
import { Kind, showToast } from "../../../src/toasts/SetupEncryptionToast";
|
|
import dis from "../../../src/dispatcher/dispatcher";
|
|
import DeviceListener from "../../../src/DeviceListener";
|
|
import Modal from "../../../src/Modal";
|
|
import ConfirmKeyStorageOffDialog from "../../../src/components/views/dialogs/ConfirmKeyStorageOffDialog";
|
|
|
|
jest.mock("../../../src/dispatcher/dispatcher", () => ({
|
|
dispatch: jest.fn(),
|
|
register: jest.fn(),
|
|
unregister: jest.fn(),
|
|
}));
|
|
|
|
describe("SetupEncryptionToast", () => {
|
|
beforeEach(() => {
|
|
jest.resetAllMocks();
|
|
render(<ToastContainer />);
|
|
});
|
|
|
|
describe("Set up recovery", () => {
|
|
it("should render the toast", async () => {
|
|
showToast(Kind.SET_UP_RECOVERY);
|
|
|
|
expect(await screen.findByRole("heading", { name: "Set up recovery" })).toBeInTheDocument();
|
|
});
|
|
|
|
it("should dismiss the toast when 'Dismiss' button clicked, and remember it", async () => {
|
|
jest.spyOn(DeviceListener.sharedInstance(), "recordRecoveryDisabled");
|
|
jest.spyOn(DeviceListener.sharedInstance(), "dismissEncryptionSetup");
|
|
|
|
showToast(Kind.SET_UP_RECOVERY);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByRole("button", { name: "Dismiss" }));
|
|
|
|
expect(DeviceListener.sharedInstance().recordRecoveryDisabled).toHaveBeenCalled();
|
|
expect(DeviceListener.sharedInstance().dismissEncryptionSetup).toHaveBeenCalled();
|
|
});
|
|
});
|
|
|
|
describe("Key storage out of sync (retrieve secrets)", () => {
|
|
it("should render the toast", async () => {
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC);
|
|
|
|
await expect(screen.findByText("Your key storage is out of sync.")).resolves.toBeInTheDocument();
|
|
});
|
|
|
|
it("should open settings to the reset flow when 'forgot recovery key' clicked", async () => {
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByText("Forgot recovery key?"));
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
action: "view_user_settings",
|
|
initialTabId: "USER_ENCRYPTION_TAB",
|
|
props: { initialEncryptionState: "reset_identity_forgot" },
|
|
});
|
|
});
|
|
|
|
it("should open settings to the reset flow when recovering fails", async () => {
|
|
jest.spyOn(SecurityManager, "accessSecretStorage").mockImplementation(async () => {
|
|
throw new Error("Something went wrong while recovering!");
|
|
});
|
|
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByText("Enter recovery key"));
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
action: "view_user_settings",
|
|
initialTabId: "USER_ENCRYPTION_TAB",
|
|
props: { initialEncryptionState: "reset_identity_sync_failed" },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("Key storage out of sync (store secrets)", () => {
|
|
it("should render the toast", async () => {
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC_STORE);
|
|
|
|
await expect(screen.findByText("Your key storage is out of sync.")).resolves.toBeInTheDocument();
|
|
});
|
|
|
|
it("should open settings to the reset flow when 'forgot recovery key' clicked", async () => {
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC_STORE);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByText("Forgot recovery key?"));
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
action: "view_user_settings",
|
|
initialTabId: "USER_ENCRYPTION_TAB",
|
|
props: { initialEncryptionState: "change_recovery_key" },
|
|
});
|
|
});
|
|
|
|
it("should open settings to the reset flow when recovering fails", async () => {
|
|
jest.spyOn(SecurityManager, "accessSecretStorage").mockImplementation(async () => {
|
|
throw new Error("Something went wrong while recovering!");
|
|
});
|
|
|
|
showToast(Kind.KEY_STORAGE_OUT_OF_SYNC_STORE);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByText("Enter recovery key"));
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
action: "view_user_settings",
|
|
initialTabId: "USER_ENCRYPTION_TAB",
|
|
props: { initialEncryptionState: "change_recovery_key" },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe("Turn on key storage", () => {
|
|
it("should render the toast", async () => {
|
|
showToast(Kind.TURN_ON_KEY_STORAGE);
|
|
|
|
await expect(screen.findByText("Turn on key storage")).resolves.toBeInTheDocument();
|
|
await expect(screen.findByRole("button", { name: "Dismiss" })).resolves.toBeInTheDocument();
|
|
await expect(screen.findByRole("button", { name: "Continue" })).resolves.toBeInTheDocument();
|
|
});
|
|
|
|
it("should open settings to the Encryption tab when 'Continue' clicked", async () => {
|
|
jest.spyOn(DeviceListener.sharedInstance(), "recordKeyBackupDisabled");
|
|
|
|
showToast(Kind.TURN_ON_KEY_STORAGE);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByRole("button", { name: "Continue" }));
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
action: "view_user_settings",
|
|
initialTabId: "USER_ENCRYPTION_TAB",
|
|
});
|
|
|
|
expect(DeviceListener.sharedInstance().recordKeyBackupDisabled).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("should open the confirm key storage off dialog when 'Dismiss' clicked", async () => {
|
|
jest.spyOn(DeviceListener.sharedInstance(), "recordKeyBackupDisabled");
|
|
|
|
// Given that as soon as the dialog opens, it closes and says "yes they clicked dismiss"
|
|
jest.spyOn(Modal, "createDialog").mockImplementation(() => {
|
|
return { finished: Promise.resolve([true]) } as any;
|
|
});
|
|
|
|
// When we show the toast, and click Dismiss
|
|
showToast(Kind.TURN_ON_KEY_STORAGE);
|
|
|
|
const user = userEvent.setup();
|
|
await user.click(await screen.findByRole("button", { name: "Dismiss" }));
|
|
|
|
// Then the dialog was opened
|
|
expect(Modal.createDialog).toHaveBeenCalledWith(
|
|
ConfirmKeyStorageOffDialog,
|
|
undefined,
|
|
"mx_ConfirmKeyStorageOffDialog",
|
|
);
|
|
|
|
// And the backup was disabled when the dialog's onFinished was called
|
|
expect(DeviceListener.sharedInstance().recordKeyBackupDisabled).toHaveBeenCalledTimes(1);
|
|
});
|
|
});
|
|
});
|