2022-05-06 10:25:18 +02:00
|
|
|
/*
|
2024-09-09 15:57:16 +02:00
|
|
|
Copyright 2024 New Vector Ltd.
|
|
|
|
Copyright 2020-2022 The Matrix.org Foundation C.I.C.
|
2022-05-06 10:25:18 +02:00
|
|
|
|
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.
|
2022-05-06 10:25:18 +02:00
|
|
|
*/
|
|
|
|
|
2023-02-13 12:39:16 +01:00
|
|
|
import React, { ComponentProps } from "react";
|
2024-03-22 13:28:13 +01:00
|
|
|
import { SecretStorage, MatrixClient } from "matrix-js-sdk/src/matrix";
|
2024-10-14 18:11:58 +02:00
|
|
|
import { act, fireEvent, render, screen } from "jest-matrix-react";
|
2023-02-03 09:39:25 +01:00
|
|
|
import userEvent from "@testing-library/user-event";
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2024-10-15 15:57:26 +02:00
|
|
|
import { mockPlatformPeg, stubClient } from "../../../../test-utils";
|
|
|
|
import AccessSecretStorageDialog from "../../../../../src/components/views/dialogs/security/AccessSecretStorageDialog";
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
const securityKey = "EsTc WKmb ivvk jLS7 Y1NH 5CcQ mP1E JJwj B3Fd pFWm t4Dp dbyu";
|
|
|
|
|
2022-05-06 10:25:18 +02:00
|
|
|
describe("AccessSecretStorageDialog", () => {
|
2024-10-14 17:08:42 +02:00
|
|
|
let mockClient: MatrixClient;
|
2023-02-03 09:39:25 +01:00
|
|
|
|
2023-02-13 12:39:16 +01:00
|
|
|
const defaultProps: ComponentProps<typeof AccessSecretStorageDialog> = {
|
2023-04-28 10:45:36 +02:00
|
|
|
keyInfo: {} as any,
|
2022-05-06 10:25:18 +02:00
|
|
|
onFinished: jest.fn(),
|
|
|
|
checkPrivateKey: jest.fn(),
|
|
|
|
};
|
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
const renderComponent = (props = {}): void => {
|
|
|
|
render(<AccessSecretStorageDialog {...defaultProps} {...props} />);
|
|
|
|
};
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
const enterSecurityKey = (placeholder = "Security Key"): void => {
|
2022-05-06 10:25:18 +02:00
|
|
|
act(() => {
|
2023-02-03 09:39:25 +01:00
|
|
|
fireEvent.change(screen.getByPlaceholderText(placeholder), {
|
|
|
|
target: {
|
|
|
|
value: securityKey,
|
|
|
|
},
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
2023-02-03 09:39:25 +01:00
|
|
|
// wait for debounce
|
|
|
|
jest.advanceTimersByTime(250);
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
2023-02-03 09:39:25 +01:00
|
|
|
};
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
const submitDialog = async (): Promise<void> => {
|
|
|
|
await userEvent.click(screen.getByText("Continue"), { delay: null });
|
|
|
|
};
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
beforeAll(() => {
|
|
|
|
jest.useFakeTimers();
|
|
|
|
mockPlatformPeg();
|
|
|
|
});
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
afterAll(() => {
|
|
|
|
jest.useRealTimers();
|
|
|
|
jest.restoreAllMocks();
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
beforeEach(() => {
|
2024-10-14 17:08:42 +02:00
|
|
|
mockClient = stubClient();
|
2023-02-03 09:39:25 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it("Closes the dialog when the form is submitted with a valid key", async () => {
|
2024-10-14 17:08:42 +02:00
|
|
|
jest.spyOn(mockClient.secretStorage, "checkKey").mockResolvedValue(true);
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
const onFinished = jest.fn();
|
|
|
|
const checkPrivateKey = jest.fn().mockResolvedValue(true);
|
|
|
|
renderComponent({ onFinished, checkPrivateKey });
|
|
|
|
|
|
|
|
// check that the input field is focused
|
|
|
|
expect(screen.getByPlaceholderText("Security Key")).toHaveFocus();
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
await enterSecurityKey();
|
|
|
|
await submitDialog();
|
|
|
|
|
|
|
|
expect(screen.getByText("Looks good!")).toBeInTheDocument();
|
|
|
|
expect(checkPrivateKey).toHaveBeenCalledWith({ recoveryKey: securityKey });
|
|
|
|
expect(onFinished).toHaveBeenCalledWith({ recoveryKey: securityKey });
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
it("Notifies the user if they input an invalid Security Key", async () => {
|
2023-02-03 09:39:25 +01:00
|
|
|
const onFinished = jest.fn();
|
|
|
|
const checkPrivateKey = jest.fn().mockResolvedValue(true);
|
|
|
|
renderComponent({ onFinished, checkPrivateKey });
|
|
|
|
|
2024-10-14 17:08:42 +02:00
|
|
|
jest.spyOn(mockClient.secretStorage, "checkKey").mockImplementation(() => {
|
2024-09-19 17:39:20 +02:00
|
|
|
throw new Error("invalid key");
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
await enterSecurityKey();
|
|
|
|
await submitDialog();
|
|
|
|
|
|
|
|
expect(screen.getByText("Continue")).toBeDisabled();
|
|
|
|
expect(screen.getByText("Invalid Security Key")).toBeInTheDocument();
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
it("Notifies the user if they input an invalid passphrase", async function () {
|
2022-05-06 10:25:18 +02:00
|
|
|
const keyInfo = {
|
2022-12-12 12:24:14 +01:00
|
|
|
name: "test",
|
|
|
|
algorithm: "test",
|
|
|
|
iv: "test",
|
|
|
|
mac: "1:2:3:4",
|
2022-05-06 10:25:18 +02:00
|
|
|
passphrase: {
|
|
|
|
// this type is weird in js-sdk
|
|
|
|
// cast 'm.pbkdf2' to itself
|
2024-03-22 13:28:13 +01:00
|
|
|
algorithm: "m.pbkdf2" as SecretStorage.PassphraseInfo["algorithm"],
|
2022-05-06 10:25:18 +02:00
|
|
|
iterations: 2,
|
2022-12-12 12:24:14 +01:00
|
|
|
salt: "nonempty",
|
2022-05-06 10:25:18 +02:00
|
|
|
},
|
|
|
|
};
|
|
|
|
const checkPrivateKey = jest.fn().mockResolvedValue(false);
|
2023-02-03 09:39:25 +01:00
|
|
|
renderComponent({ checkPrivateKey, keyInfo });
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2023-02-03 09:39:25 +01:00
|
|
|
await enterSecurityKey("Security Phrase");
|
|
|
|
expect(screen.getByPlaceholderText("Security Phrase")).toHaveValue(securityKey);
|
|
|
|
await submitDialog();
|
2022-05-06 10:25:18 +02:00
|
|
|
|
2024-08-06 19:22:02 +02:00
|
|
|
await expect(
|
|
|
|
screen.findByText(
|
2023-02-03 09:39:25 +01:00
|
|
|
"👎 Unable to access secret storage. Please verify that you entered the correct Security Phrase.",
|
|
|
|
),
|
2024-08-06 19:22:02 +02:00
|
|
|
).resolves.toBeInTheDocument();
|
2023-03-08 12:32:50 +01:00
|
|
|
|
|
|
|
expect(screen.getByPlaceholderText("Security Phrase")).toHaveFocus();
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|
2024-11-19 04:17:24 +01:00
|
|
|
|
|
|
|
it("Can reset secret storage", async () => {
|
|
|
|
jest.spyOn(mockClient.secretStorage, "checkKey").mockResolvedValue(true);
|
|
|
|
|
|
|
|
const onFinished = jest.fn();
|
|
|
|
const checkPrivateKey = jest.fn().mockResolvedValue(true);
|
|
|
|
renderComponent({ onFinished, checkPrivateKey });
|
|
|
|
|
|
|
|
await userEvent.click(screen.getByText("Reset all"), { delay: null });
|
|
|
|
|
|
|
|
// It will prompt the user to confirm resetting
|
|
|
|
expect(screen.getByText("Reset everything")).toBeInTheDocument();
|
|
|
|
await userEvent.click(screen.getByText("Reset"), { delay: null });
|
|
|
|
|
|
|
|
// Then it will prompt the user to create a key/passphrase
|
|
|
|
await screen.findByText("Set up Secure Backup");
|
|
|
|
document.execCommand = jest.fn().mockReturnValue(true);
|
|
|
|
jest.spyOn(mockClient.getCrypto()!, "createRecoveryKeyFromPassphrase").mockResolvedValue({
|
|
|
|
privateKey: new Uint8Array(),
|
|
|
|
encodedPrivateKey: securityKey,
|
|
|
|
});
|
|
|
|
screen.getByRole("button", { name: "Continue" }).click();
|
|
|
|
|
|
|
|
await screen.findByText(/Save your Security Key/);
|
|
|
|
screen.getByRole("button", { name: "Copy" }).click();
|
|
|
|
await screen.findByText("Copied!");
|
|
|
|
screen.getByRole("button", { name: "Continue" }).click();
|
|
|
|
|
|
|
|
await screen.findByText("Secure Backup successful");
|
|
|
|
});
|
2022-05-06 10:25:18 +02:00
|
|
|
});
|