/* Copyright 2019, 2023 New Vector Ltd 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 from "react"; import { fireEvent, render, screen, waitForElementToBeRemoved } from "@testing-library/react"; import { mocked, MockedObject } from "jest-mock"; import { createClient, MatrixClient } from "matrix-js-sdk/src/matrix"; import fetchMock from "fetch-mock-jest"; import { DELEGATED_OIDC_COMPATIBILITY, IdentityProviderBrand } from "matrix-js-sdk/src/@types/auth"; import SdkConfig from "../../../../src/SdkConfig"; import { mkServerConfig, mockPlatformPeg, unmockPlatformPeg } from "../../../test-utils"; import Login from "../../../../src/components/structures/auth/Login"; import BasePlatform from "../../../../src/BasePlatform"; jest.mock("matrix-js-sdk/src/matrix"); jest.useRealTimers(); describe("Login", function () { let platform: MockedObject; const mockClient = mocked({ login: jest.fn().mockResolvedValue({}), loginFlows: jest.fn(), } as unknown as MatrixClient); beforeEach(function () { SdkConfig.put({ brand: "test-brand", disable_custom_urls: true, }); mockClient.login.mockClear().mockResolvedValue({}); mockClient.loginFlows.mockClear().mockResolvedValue({ flows: [{ type: "m.login.password" }] }); mocked(createClient).mockImplementation((opts) => { mockClient.idBaseUrl = opts.idBaseUrl; mockClient.baseUrl = opts.baseUrl; return mockClient; }); fetchMock.resetBehavior(); fetchMock.get("https://matrix.org/_matrix/client/versions", { unstable_features: {}, versions: [], }); platform = mockPlatformPeg({ startSingleSignOn: jest.fn(), }); }); afterEach(function () { fetchMock.restore(); SdkConfig.reset(); // we touch the config, so clean up unmockPlatformPeg(); }); function getRawComponent(hsUrl = "https://matrix.org", isUrl = "https://vector.im") { return ( {}} onRegisterClick={() => {}} onServerConfigChange={() => {}} /> ); } function getComponent(hsUrl?: string, isUrl?: string) { return render(getRawComponent(hsUrl, isUrl)); } it("should show form with change server link", async () => { SdkConfig.put({ brand: "test-brand", disable_custom_urls: false, }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); expect(container.querySelector("form")).toBeTruthy(); expect(container.querySelector(".mx_ServerPicker_change")).toBeTruthy(); }); it("should show form without change server link when custom URLs disabled", async () => { const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); expect(container.querySelector("form")).toBeTruthy(); expect(container.querySelectorAll(".mx_ServerPicker_change")).toHaveLength(0); }); it("should show SSO button if that flow is available", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [{ type: "m.login.sso" }] }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); const ssoButton = container.querySelector(".mx_SSOButton"); expect(ssoButton).toBeTruthy(); }); it("should show both SSO button and username+password if both are available", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [{ type: "m.login.password" }, { type: "m.login.sso" }] }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); expect(container.querySelector("form")).toBeTruthy(); const ssoButton = container.querySelector(".mx_SSOButton"); expect(ssoButton).toBeTruthy(); }); it("should show multiple SSO buttons if multiple identity_providers are available", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "m.login.sso", identity_providers: [ { id: "a", name: "Provider 1", }, { id: "b", name: "Provider 2", }, { id: "c", name: "Provider 3", }, ], }, ], }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); const ssoButtons = container.querySelectorAll(".mx_SSOButton"); expect(ssoButtons.length).toBe(3); }); it("should show single SSO button if identity_providers is null", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "m.login.sso", }, ], }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); const ssoButtons = container.querySelectorAll(".mx_SSOButton"); expect(ssoButtons.length).toBe(1); }); it("should handle serverConfig updates correctly", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "m.login.sso", }, ], }); const { container, rerender } = render(getRawComponent()); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); fireEvent.click(container.querySelector(".mx_SSOButton")!); expect(platform.startSingleSignOn.mock.calls[0][0].baseUrl).toBe("https://matrix.org"); fetchMock.get("https://server2/_matrix/client/versions", { unstable_features: {}, versions: [], }); rerender(getRawComponent("https://server2")); fireEvent.click(container.querySelector(".mx_SSOButton")!); expect(platform.startSingleSignOn.mock.calls[1][0].baseUrl).toBe("https://server2"); }); it("should show single Continue button if OIDC MSC3824 compatibility is given by server", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "m.login.sso", [DELEGATED_OIDC_COMPATIBILITY.name]: true, }, { type: "m.login.password", }, ], }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); const ssoButtons = container.querySelectorAll(".mx_SSOButton"); expect(ssoButtons.length).toBe(1); expect(ssoButtons[0].textContent).toBe("Continue"); // no password form visible expect(container.querySelector("form")).toBeFalsy(); }); it("should show branded SSO buttons", async () => { const idpsWithIcons = Object.values(IdentityProviderBrand).map((brand) => ({ id: brand, brand, name: `Provider ${brand}`, })); mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "m.login.sso", identity_providers: [ ...idpsWithIcons, { id: "foo", name: "Provider foo", }, ], }, ], }); const { container } = getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); for (const idp of idpsWithIcons) { const ssoButton = container.querySelector(`.mx_SSOButton.mx_SSOButton_brand_${idp.brand}`); expect(ssoButton).toBeTruthy(); expect(ssoButton?.querySelector(`img[alt="${idp.brand}"]`)).toBeTruthy(); } const ssoButtons = container.querySelectorAll(".mx_SSOButton"); expect(ssoButtons.length).toBe(idpsWithIcons.length + 1); }); it("should display an error when homeserver doesn't offer any supported login flows", async () => { mockClient.loginFlows.mockResolvedValue({ flows: [ { type: "just something weird", }, ], }); getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); expect( screen.getByText("This homeserver doesn't offer any login flows which are supported by this client."), ).toBeInTheDocument(); }); it("should display a connection error when getting login flows fails", async () => { mockClient.loginFlows.mockRejectedValue("oups"); getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); expect( screen.getByText("There was a problem communicating with the homeserver, please try again later."), ).toBeInTheDocument(); }); it("should display an error when homeserver fails liveliness check", async () => { fetchMock.resetBehavior(); fetchMock.get("https://matrix.org/_matrix/client/versions", { status: 400, }); getComponent(); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); // error displayed expect(screen.getByText("Your test-brand is misconfigured")).toBeInTheDocument(); }); it("should reset liveliness error when server config changes", async () => { fetchMock.resetBehavior(); // matrix.org is not alive fetchMock.get("https://matrix.org/_matrix/client/versions", { status: 400, }); // but server2 is fetchMock.get("https://server2/_matrix/client/versions", { unstable_features: {}, versions: [], }); const { rerender } = render(getRawComponent()); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); // error displayed expect(screen.getByText("Your test-brand is misconfigured")).toBeInTheDocument(); rerender(getRawComponent("https://server2")); await waitForElementToBeRemoved(() => screen.queryAllByLabelText("Loading…")); // error cleared expect(screen.queryByText("Your test-brand is misconfigured")).not.toBeInTheDocument(); }); });