mirror of https://github.com/vector-im/riot-web
				
				
				
			
		
			
				
	
	
		
			428 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			428 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			TypeScript
		
	
	
| /*
 | |
| Copyright 2022 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 from "react";
 | |
| import { act } from "react-dom/test-utils";
 | |
| import { Room, Beacon, BeaconEvent, getBeaconInfoIdentifier, MatrixEvent } from "matrix-js-sdk/src/matrix";
 | |
| import { logger } from "matrix-js-sdk/src/logger";
 | |
| import { fireEvent, getByTestId, render, screen, waitFor } from "@testing-library/react";
 | |
| 
 | |
| import RoomLiveShareWarning from "../../../../src/components/views/beacon/RoomLiveShareWarning";
 | |
| import { OwnBeaconStore, OwnBeaconStoreEvent } from "../../../../src/stores/OwnBeaconStore";
 | |
| import {
 | |
|     advanceDateAndTime,
 | |
|     flushPromisesWithFakeTimers,
 | |
|     getMockClientWithEventEmitter,
 | |
|     makeBeaconInfoEvent,
 | |
|     mockGeolocation,
 | |
|     resetAsyncStoreWithClient,
 | |
|     setupAsyncStoreWithClient,
 | |
| } from "../../../test-utils";
 | |
| import defaultDispatcher from "../../../../src/dispatcher/dispatcher";
 | |
| import { Action } from "../../../../src/dispatcher/actions";
 | |
| 
 | |
| jest.useFakeTimers();
 | |
| describe("<RoomLiveShareWarning />", () => {
 | |
|     const aliceId = "@alice:server.org";
 | |
|     const room1Id = "$room1:server.org";
 | |
|     const room2Id = "$room2:server.org";
 | |
|     const room3Id = "$room3:server.org";
 | |
|     const mockClient = getMockClientWithEventEmitter({
 | |
|         getVisibleRooms: jest.fn().mockReturnValue([]),
 | |
|         getUserId: jest.fn().mockReturnValue(aliceId),
 | |
|         unstable_setLiveBeacon: jest.fn().mockResolvedValue({ event_id: "1" }),
 | |
|         sendEvent: jest.fn(),
 | |
|     });
 | |
| 
 | |
|     // 14.03.2022 16:15
 | |
|     const now = 1647270879403;
 | |
|     const MINUTE_MS = 60000;
 | |
|     const HOUR_MS = 3600000;
 | |
|     // mock the date so events are stable for snapshots etc
 | |
|     jest.spyOn(global.Date, "now").mockReturnValue(now);
 | |
|     const room1Beacon1 = makeBeaconInfoEvent(
 | |
|         aliceId,
 | |
|         room1Id,
 | |
|         {
 | |
|             isLive: true,
 | |
|             timeout: HOUR_MS,
 | |
|         },
 | |
|         "$0",
 | |
|     );
 | |
|     const room2Beacon1 = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true, timeout: HOUR_MS }, "$1");
 | |
|     const room2Beacon2 = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true, timeout: HOUR_MS * 12 }, "$2");
 | |
|     const room3Beacon1 = makeBeaconInfoEvent(aliceId, room3Id, { isLive: true, timeout: HOUR_MS }, "$3");
 | |
| 
 | |
|     // make fresh rooms every time
 | |
|     // as we update room state
 | |
|     const makeRoomsWithStateEvents = (stateEvents: MatrixEvent[] = []): [Room, Room] => {
 | |
|         const room1 = new Room(room1Id, mockClient, aliceId);
 | |
|         const room2 = new Room(room2Id, mockClient, aliceId);
 | |
| 
 | |
|         room1.currentState.setStateEvents(stateEvents);
 | |
|         room2.currentState.setStateEvents(stateEvents);
 | |
|         mockClient.getVisibleRooms.mockReturnValue([room1, room2]);
 | |
| 
 | |
|         return [room1, room2];
 | |
|     };
 | |
| 
 | |
|     const makeOwnBeaconStore = async () => {
 | |
|         const store = OwnBeaconStore.instance;
 | |
| 
 | |
|         await setupAsyncStoreWithClient(store, mockClient);
 | |
|         return store;
 | |
|     };
 | |
| 
 | |
|     const defaultProps = {
 | |
|         roomId: room1Id,
 | |
|     };
 | |
|     const getComponent = (props = {}) => {
 | |
|         return render(<RoomLiveShareWarning {...defaultProps} {...props} />);
 | |
|     };
 | |
| 
 | |
|     const localStorageSpy = jest.spyOn(localStorage.__proto__, "getItem").mockReturnValue(undefined);
 | |
| 
 | |
|     beforeEach(() => {
 | |
|         mockGeolocation();
 | |
|         jest.spyOn(global.Date, "now").mockReturnValue(now);
 | |
|         mockClient.unstable_setLiveBeacon.mockReset().mockResolvedValue({ event_id: "1" });
 | |
| 
 | |
|         // assume all beacons were created on this device
 | |
|         localStorageSpy.mockReturnValue(
 | |
|             JSON.stringify([room1Beacon1.getId(), room2Beacon1.getId(), room2Beacon2.getId(), room3Beacon1.getId()]),
 | |
|         );
 | |
|     });
 | |
| 
 | |
|     afterEach(async () => {
 | |
|         jest.spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError").mockRestore();
 | |
|         await resetAsyncStoreWithClient(OwnBeaconStore.instance);
 | |
|     });
 | |
| 
 | |
|     afterAll(() => {
 | |
|         jest.spyOn(global.Date, "now").mockRestore();
 | |
|         localStorageSpy.mockRestore();
 | |
|         jest.spyOn(defaultDispatcher, "dispatch").mockRestore();
 | |
|     });
 | |
| 
 | |
|     it("renders nothing when user has no live beacons at all", async () => {
 | |
|         await makeOwnBeaconStore();
 | |
|         const { asFragment } = getComponent();
 | |
|         expect(asFragment()).toMatchInlineSnapshot(`<DocumentFragment />`);
 | |
|     });
 | |
| 
 | |
|     it("renders nothing when user has no live beacons in room", async () => {
 | |
|         await act(async () => {
 | |
|             await makeRoomsWithStateEvents([room2Beacon1]);
 | |
|             await makeOwnBeaconStore();
 | |
|         });
 | |
|         const { asFragment } = getComponent({ roomId: room1Id });
 | |
|         expect(asFragment()).toMatchInlineSnapshot(`<DocumentFragment />`);
 | |
|     });
 | |
| 
 | |
|     it("does not render when geolocation is not working", async () => {
 | |
|         jest.spyOn(logger, "error").mockImplementation(() => {});
 | |
|         // @ts-ignore
 | |
|         navigator.geolocation = undefined;
 | |
|         await act(async () => {
 | |
|             await makeRoomsWithStateEvents([room1Beacon1, room2Beacon1, room2Beacon2]);
 | |
|             await makeOwnBeaconStore();
 | |
|         });
 | |
|         const { asFragment } = getComponent({ roomId: room1Id });
 | |
| 
 | |
|         expect(asFragment()).toMatchInlineSnapshot(`<DocumentFragment />`);
 | |
|     });
 | |
|     describe("when user has live beacons and geolocation is available", () => {
 | |
|         beforeEach(async () => {
 | |
|             await act(async () => {
 | |
|                 await makeRoomsWithStateEvents([room1Beacon1, room2Beacon1, room2Beacon2]);
 | |
|                 await makeOwnBeaconStore();
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         it("renders correctly with one live beacon in room", () => {
 | |
|             const { asFragment } = getComponent({ roomId: room1Id });
 | |
|             // beacons have generated ids that break snapshots
 | |
|             // assert on html
 | |
|             expect(asFragment()).toMatchSnapshot();
 | |
|         });
 | |
| 
 | |
|         it("renders correctly with two live beacons in room", () => {
 | |
|             const { asFragment, container } = getComponent({ roomId: room2Id });
 | |
|             // beacons have generated ids that break snapshots
 | |
|             // assert on html
 | |
|             expect(asFragment()).toMatchSnapshot();
 | |
|             // later expiry displayed
 | |
|             expect(container).toHaveTextContent("12h left");
 | |
|         });
 | |
| 
 | |
|         it("removes itself when user stops having live beacons", async () => {
 | |
|             const { container } = getComponent({ roomId: room1Id });
 | |
|             // started out rendered
 | |
|             expect(container.firstChild).toBeTruthy();
 | |
| 
 | |
|             // time travel until room1Beacon1 is expired
 | |
|             act(() => {
 | |
|                 advanceDateAndTime(HOUR_MS + 1);
 | |
|             });
 | |
|             act(() => {
 | |
|                 mockClient.emit(BeaconEvent.LivenessChange, false, new Beacon(room1Beacon1));
 | |
|             });
 | |
| 
 | |
|             await waitFor(() => expect(container.firstChild).toBeFalsy());
 | |
|         });
 | |
| 
 | |
|         it("removes itself when user stops monitoring live position", async () => {
 | |
|             const { container } = getComponent({ roomId: room1Id });
 | |
|             // started out rendered
 | |
|             expect(container.firstChild).toBeTruthy();
 | |
| 
 | |
|             act(() => {
 | |
|                 // cheat to clear this
 | |
|                 // @ts-ignore
 | |
|                 OwnBeaconStore.instance.clearPositionWatch = undefined;
 | |
|                 OwnBeaconStore.instance.emit(OwnBeaconStoreEvent.MonitoringLivePosition);
 | |
|             });
 | |
| 
 | |
|             await waitFor(() => expect(container.firstChild).toBeFalsy());
 | |
|         });
 | |
| 
 | |
|         it("renders when user adds a live beacon", async () => {
 | |
|             const { container } = getComponent({ roomId: room3Id });
 | |
|             // started out not rendered
 | |
|             expect(container.firstChild).toBeFalsy();
 | |
|             act(() => {
 | |
|                 mockClient.emit(BeaconEvent.New, room3Beacon1, new Beacon(room3Beacon1));
 | |
|             });
 | |
| 
 | |
|             await waitFor(() => expect(container.firstChild).toBeTruthy());
 | |
|         });
 | |
| 
 | |
|         it("updates beacon time left periodically", () => {
 | |
|             const { container } = getComponent({ roomId: room1Id });
 | |
|             expect(container).toHaveTextContent("1h left");
 | |
| 
 | |
|             act(() => {
 | |
|                 advanceDateAndTime(MINUTE_MS * 25);
 | |
|             });
 | |
| 
 | |
|             expect(container).toHaveTextContent("35m left");
 | |
|         });
 | |
| 
 | |
|         it("updates beacon time left when beacon updates", () => {
 | |
|             const { container } = getComponent({ roomId: room1Id });
 | |
|             expect(container).toHaveTextContent("1h left");
 | |
| 
 | |
|             act(() => {
 | |
|                 const beacon = OwnBeaconStore.instance.getBeaconById(getBeaconInfoIdentifier(room1Beacon1));
 | |
|                 const room1Beacon1Update = makeBeaconInfoEvent(
 | |
|                     aliceId,
 | |
|                     room1Id,
 | |
|                     {
 | |
|                         isLive: true,
 | |
|                         timeout: 3 * HOUR_MS,
 | |
|                     },
 | |
|                     "$0",
 | |
|                 );
 | |
|                 beacon?.update(room1Beacon1Update);
 | |
|             });
 | |
| 
 | |
|             // update to expiry of new beacon
 | |
|             expect(container).toHaveTextContent("3h left");
 | |
|         });
 | |
| 
 | |
|         it("clears expiry time interval on unmount", () => {
 | |
|             const clearIntervalSpy = jest.spyOn(global, "clearInterval");
 | |
|             const { container, unmount } = getComponent({ roomId: room1Id });
 | |
|             expect(container).toHaveTextContent("1h left");
 | |
| 
 | |
|             unmount();
 | |
| 
 | |
|             expect(clearIntervalSpy).toHaveBeenCalled();
 | |
|         });
 | |
| 
 | |
|         it("navigates to beacon tile on click", () => {
 | |
|             const dispatcherSpy = jest.spyOn(defaultDispatcher, "dispatch");
 | |
|             const { container } = getComponent({ roomId: room1Id });
 | |
| 
 | |
|             act(() => {
 | |
|                 fireEvent.click(container.firstChild! as Node);
 | |
|             });
 | |
| 
 | |
|             expect(dispatcherSpy).toHaveBeenCalledWith({
 | |
|                 action: Action.ViewRoom,
 | |
|                 event_id: room1Beacon1.getId(),
 | |
|                 room_id: room1Id,
 | |
|                 highlighted: true,
 | |
|                 scroll_into_view: true,
 | |
|                 metricsTrigger: undefined,
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         describe("stopping beacons", () => {
 | |
|             it("stops beacon on stop sharing click", async () => {
 | |
|                 const { container } = getComponent({ roomId: room2Id });
 | |
| 
 | |
|                 const btn = getByTestId(container, "room-live-share-primary-button");
 | |
| 
 | |
|                 fireEvent.click(btn);
 | |
| 
 | |
|                 expect(mockClient.unstable_setLiveBeacon).toHaveBeenCalled();
 | |
| 
 | |
|                 await waitFor(() => expect(screen.queryByTestId("spinner")).toBeInTheDocument());
 | |
| 
 | |
|                 expect(btn.hasAttribute("disabled")).toBe(true);
 | |
|             });
 | |
| 
 | |
|             it("displays error when stop sharing fails", async () => {
 | |
|                 const { container, asFragment } = getComponent({ roomId: room1Id });
 | |
|                 const btn = getByTestId(container, "room-live-share-primary-button");
 | |
| 
 | |
|                 // fail first time
 | |
|                 mockClient.unstable_setLiveBeacon
 | |
|                     .mockRejectedValueOnce(new Error("oups"))
 | |
|                     .mockResolvedValue({ event_id: "1" });
 | |
| 
 | |
|                 await act(async () => {
 | |
|                     fireEvent.click(btn);
 | |
|                     await flushPromisesWithFakeTimers();
 | |
|                 });
 | |
| 
 | |
|                 expect(asFragment()).toMatchSnapshot();
 | |
| 
 | |
|                 act(() => {
 | |
|                     fireEvent.click(btn);
 | |
|                 });
 | |
| 
 | |
|                 expect(mockClient.unstable_setLiveBeacon).toHaveBeenCalledTimes(2);
 | |
|             });
 | |
| 
 | |
|             it("displays again with correct state after stopping a beacon", () => {
 | |
|                 // make sure the loading state is reset correctly after removing a beacon
 | |
|                 const { container } = getComponent({ roomId: room1Id });
 | |
|                 const btn = getByTestId(container, "room-live-share-primary-button");
 | |
| 
 | |
|                 // stop the beacon
 | |
|                 act(() => {
 | |
|                     fireEvent.click(btn);
 | |
|                 });
 | |
|                 // time travel until room1Beacon1 is expired
 | |
|                 act(() => {
 | |
|                     advanceDateAndTime(HOUR_MS + 1);
 | |
|                 });
 | |
|                 act(() => {
 | |
|                     mockClient.emit(BeaconEvent.LivenessChange, false, new Beacon(room1Beacon1));
 | |
|                 });
 | |
| 
 | |
|                 const newLiveBeacon = makeBeaconInfoEvent(aliceId, room1Id, { isLive: true });
 | |
|                 act(() => {
 | |
|                     mockClient.emit(BeaconEvent.New, newLiveBeacon, new Beacon(newLiveBeacon));
 | |
|                 });
 | |
| 
 | |
|                 // button not disabled and expiry time shown
 | |
|                 expect(btn.hasAttribute("disabled")).toBe(true);
 | |
|             });
 | |
|         });
 | |
| 
 | |
|         describe("with location publish errors", () => {
 | |
|             it("displays location publish error when mounted with location publish errors", async () => {
 | |
|                 const locationPublishErrorSpy = jest
 | |
|                     .spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError")
 | |
|                     .mockReturnValue(true);
 | |
|                 const { asFragment } = getComponent({ roomId: room2Id });
 | |
| 
 | |
|                 expect(asFragment()).toMatchSnapshot();
 | |
|                 expect(locationPublishErrorSpy).toHaveBeenCalledWith(getBeaconInfoIdentifier(room2Beacon1), 0, [
 | |
|                     getBeaconInfoIdentifier(room2Beacon1),
 | |
|                 ]);
 | |
|             });
 | |
| 
 | |
|             it(
 | |
|                 "displays location publish error when locationPublishError event is emitted" +
 | |
|                     " and beacons have errors",
 | |
|                 async () => {
 | |
|                     const locationPublishErrorSpy = jest
 | |
|                         .spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError")
 | |
|                         .mockReturnValue(false);
 | |
|                     const { container } = getComponent({ roomId: room2Id });
 | |
| 
 | |
|                     // update mock and emit event
 | |
|                     act(() => {
 | |
|                         locationPublishErrorSpy.mockReturnValue(true);
 | |
|                         OwnBeaconStore.instance.emit(
 | |
|                             OwnBeaconStoreEvent.LocationPublishError,
 | |
|                             getBeaconInfoIdentifier(room2Beacon1),
 | |
|                         );
 | |
|                     });
 | |
| 
 | |
|                     // renders wire error ui
 | |
|                     expect(container).toHaveTextContent(
 | |
|                         "An error occurred whilst sharing your live location, please try again",
 | |
|                     );
 | |
| 
 | |
|                     expect(screen.queryByTestId("room-live-share-wire-error-close-button")).toBeInTheDocument();
 | |
|                 },
 | |
|             );
 | |
| 
 | |
|             it("stops displaying wire error when errors are cleared", async () => {
 | |
|                 const locationPublishErrorSpy = jest
 | |
|                     .spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError")
 | |
|                     .mockReturnValue(true);
 | |
|                 const { container } = getComponent({ roomId: room2Id });
 | |
| 
 | |
|                 // update mock and emit event
 | |
|                 act(() => {
 | |
|                     locationPublishErrorSpy.mockReturnValue(false);
 | |
|                     OwnBeaconStore.instance.emit(
 | |
|                         OwnBeaconStoreEvent.LocationPublishError,
 | |
|                         getBeaconInfoIdentifier(room2Beacon1),
 | |
|                     );
 | |
|                 });
 | |
| 
 | |
|                 // renders error-free ui
 | |
|                 expect(container).toHaveTextContent("You are sharing your live location");
 | |
|                 expect(screen.queryByTestId("room-live-share-wire-error-close-button")).not.toBeInTheDocument();
 | |
|             });
 | |
| 
 | |
|             it("clicking retry button resets location publish errors", async () => {
 | |
|                 jest.spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError").mockReturnValue(true);
 | |
|                 const resetErrorSpy = jest.spyOn(OwnBeaconStore.instance, "resetLocationPublishError");
 | |
| 
 | |
|                 const { container } = getComponent({ roomId: room2Id });
 | |
|                 const btn = getByTestId(container, "room-live-share-primary-button");
 | |
| 
 | |
|                 act(() => {
 | |
|                     fireEvent.click(btn);
 | |
|                 });
 | |
| 
 | |
|                 expect(resetErrorSpy).toHaveBeenCalledWith(getBeaconInfoIdentifier(room2Beacon1));
 | |
|             });
 | |
| 
 | |
|             it("clicking close button stops beacons", async () => {
 | |
|                 jest.spyOn(OwnBeaconStore.instance, "beaconHasLocationPublishError").mockReturnValue(true);
 | |
|                 const stopBeaconSpy = jest.spyOn(OwnBeaconStore.instance, "stopBeacon");
 | |
| 
 | |
|                 const { container } = getComponent({ roomId: room2Id });
 | |
|                 const btn = getByTestId(container, "room-live-share-wire-error-close-button");
 | |
|                 act(() => {
 | |
|                     fireEvent.click(btn);
 | |
|                 });
 | |
| 
 | |
|                 expect(stopBeaconSpy).toHaveBeenCalledWith(getBeaconInfoIdentifier(room2Beacon1));
 | |
|             });
 | |
|         });
 | |
|     });
 | |
| });
 |