/* Copyright 2023 The Matrix.org Foundation C.I.C. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import React, { ComponentProps } from "react"; import { mocked, Mocked } from "jest-mock"; import { act, render, RenderResult } from "@testing-library/react"; import { VerificationRequest, VerificationRequestEvent, } from "matrix-js-sdk/src/crypto/verification/request/VerificationRequest"; import { TypedEventEmitter, IMyDevice, MatrixClient, Device } from "matrix-js-sdk/src/matrix"; import VerificationRequestToast from "../../../../src/components/views/toasts/VerificationRequestToast"; import { flushPromises, getMockClientWithEventEmitter, mockClientMethodsCrypto, mockClientMethodsUser, } from "../../../test-utils"; import ToastStore from "../../../../src/stores/ToastStore"; function renderComponent( props: Partial> & { request: VerificationRequest }, ): RenderResult { const propsWithDefaults = { toastKey: "test", ...props, }; return render(); } describe("VerificationRequestToast", () => { let client: Mocked; beforeEach(() => { client = getMockClientWithEventEmitter({ ...mockClientMethodsUser(), ...mockClientMethodsCrypto(), getDevice: jest.fn(), }); }); it("should render a self-verification", async () => { const otherDeviceId = "other_device"; const otherIDevice: IMyDevice = { device_id: otherDeviceId, last_seen_ip: "1.1.1.1" }; client.getDevice.mockResolvedValue(otherIDevice); const otherDeviceInfo = new Device({ algorithms: [], keys: new Map(), userId: "", deviceId: otherDeviceId, displayName: "my other device", }); const deviceMap = new Map([[client.getSafeUserId(), new Map([[otherDeviceId, otherDeviceInfo]])]]); mocked(client.getCrypto()!.getUserDeviceInfo).mockResolvedValue(deviceMap); const request = makeMockVerificationRequest({ isSelfVerification: true, otherDeviceId, }); const result = renderComponent({ request }); await act(async () => { await flushPromises(); }); expect(result.container).toMatchSnapshot(); }); it("should render a cross-user verification", async () => { const otherUserId = "@other:user"; const request = makeMockVerificationRequest({ isSelfVerification: false, otherUserId, }); const result = renderComponent({ request }); await act(async () => { await flushPromises(); }); expect(result.container).toMatchSnapshot(); }); it("dismisses itself once the request can no longer be accepted", async () => { const otherUserId = "@other:user"; const request = makeMockVerificationRequest({ isSelfVerification: false, otherUserId, }); renderComponent({ request, toastKey: "testKey" }); await act(async () => { await flushPromises(); }); const dismiss = jest.spyOn(ToastStore.sharedInstance(), "dismissToast"); Object.defineProperty(request, "accepting", { value: true }); request.emit(VerificationRequestEvent.Change); expect(dismiss).toHaveBeenCalledWith("testKey"); }); }); function makeMockVerificationRequest(props: Partial = {}): Mocked { const request = new TypedEventEmitter(); Object.assign(request, { ...props, }); return request as unknown as Mocked; }