mirror of https://github.com/vector-im/riot-web
				
				
				
			
		
			
				
	
	
		
			376 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			376 lines
		
	
	
		
			14 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 { render, RenderResult } from "@testing-library/react";
 | |
| // eslint-disable-next-line deprecate/import
 | |
| import { mount, ReactWrapper } from "enzyme";
 | |
| import { MessageEvent } from 'matrix-events-sdk';
 | |
| import { ReceiptType } from "matrix-js-sdk/src/@types/read_receipts";
 | |
| import {
 | |
|     EventTimelineSet,
 | |
|     EventType,
 | |
|     MatrixClient,
 | |
|     MatrixEvent,
 | |
|     PendingEventOrdering,
 | |
|     Room,
 | |
| } from 'matrix-js-sdk/src/matrix';
 | |
| import { EventTimeline } from "matrix-js-sdk/src/models/event-timeline";
 | |
| import {
 | |
|     FeatureSupport,
 | |
|     Thread,
 | |
|     THREAD_RELATION_TYPE,
 | |
|     ThreadEvent,
 | |
|     ThreadFilterType,
 | |
| } from "matrix-js-sdk/src/models/thread";
 | |
| import React from 'react';
 | |
| 
 | |
| import TimelinePanel from '../../../src/components/structures/TimelinePanel';
 | |
| import MatrixClientContext from "../../../src/contexts/MatrixClientContext";
 | |
| import { MatrixClientPeg } from '../../../src/MatrixClientPeg';
 | |
| import SettingsStore from "../../../src/settings/SettingsStore";
 | |
| import { mkRoom, stubClient } from "../../test-utils";
 | |
| 
 | |
| const newReceipt = (eventId: string, userId: string, readTs: number, fullyReadTs: number): MatrixEvent => {
 | |
|     const receiptContent = {
 | |
|         [eventId]: {
 | |
|             [ReceiptType.Read]: { [userId]: { ts: readTs } },
 | |
|             [ReceiptType.ReadPrivate]: { [userId]: { ts: readTs } },
 | |
|             [ReceiptType.FullyRead]: { [userId]: { ts: fullyReadTs } },
 | |
|         },
 | |
|     };
 | |
|     return new MatrixEvent({ content: receiptContent, type: EventType.Receipt });
 | |
| };
 | |
| 
 | |
| const getProps = (room: Room, events: MatrixEvent[]): TimelinePanel["props"] => {
 | |
|     const timelineSet = { room: room as Room } as EventTimelineSet;
 | |
|     const timeline = new EventTimeline(timelineSet);
 | |
|     events.forEach((event) => timeline.addEvent(event, true));
 | |
|     timelineSet.getLiveTimeline = () => timeline;
 | |
|     timelineSet.getTimelineForEvent = () => timeline;
 | |
|     timelineSet.getPendingEvents = () => events;
 | |
|     timelineSet.room!.getEventReadUpTo = () => events[1].getId() ?? null;
 | |
| 
 | |
|     return {
 | |
|         timelineSet,
 | |
|         manageReadReceipts: true,
 | |
|         sendReadReceiptOnLoad: true,
 | |
|     };
 | |
| };
 | |
| 
 | |
| const renderPanel = (room: Room, events: MatrixEvent[]): RenderResult => {
 | |
|     const props = getProps(room, events);
 | |
|     return render(<TimelinePanel {...props} />);
 | |
| };
 | |
| 
 | |
| const mockEvents = (room: Room, count = 2): MatrixEvent[] => {
 | |
|     const events: MatrixEvent[] = [];
 | |
|     for (let index = 0; index < count; index++) {
 | |
|         events.push(new MatrixEvent({
 | |
|             room_id: room.roomId,
 | |
|             event_id: `event_${index}`,
 | |
|             type: EventType.RoomMessage,
 | |
|             user_id: "userId",
 | |
|             content: MessageEvent.from(`Event${index}`).serialize().content,
 | |
|         }));
 | |
|     }
 | |
| 
 | |
|     return events;
 | |
| };
 | |
| 
 | |
| describe('TimelinePanel', () => {
 | |
|     beforeEach(() => {
 | |
|         stubClient();
 | |
|     });
 | |
| 
 | |
|     describe('read receipts and markers', () => {
 | |
|         it('should forget the read marker when asked to', () => {
 | |
|             const cli = MatrixClientPeg.get();
 | |
|             const readMarkersSent: string[] = [];
 | |
| 
 | |
|             // Track calls to setRoomReadMarkers
 | |
|             cli.setRoomReadMarkers = (_roomId, rmEventId, _a, _b) => {
 | |
|                 readMarkersSent.push(rmEventId);
 | |
|                 return Promise.resolve({});
 | |
|             };
 | |
| 
 | |
|             const ev0 = new MatrixEvent({
 | |
|                 event_id: "ev0",
 | |
|                 sender: "@u2:m.org",
 | |
|                 origin_server_ts: 111,
 | |
|                 ...MessageEvent.from("hello 1").serialize(),
 | |
|             });
 | |
|             const ev1 = new MatrixEvent({
 | |
|                 event_id: "ev1",
 | |
|                 sender: "@u2:m.org",
 | |
|                 origin_server_ts: 222,
 | |
|                 ...MessageEvent.from("hello 2").serialize(),
 | |
|             });
 | |
| 
 | |
|             const roomId = "#room:example.com";
 | |
|             const userId = cli.credentials.userId!;
 | |
|             const room = new Room(
 | |
|                 roomId,
 | |
|                 cli,
 | |
|                 userId,
 | |
|                 { pendingEventOrdering: PendingEventOrdering.Detached },
 | |
|             );
 | |
| 
 | |
|             // Create a TimelinePanel with ev0 already present
 | |
|             const timelineSet = new EventTimelineSet(room, {});
 | |
|             timelineSet.addLiveEvent(ev0);
 | |
|             const component: ReactWrapper<TimelinePanel> = mount(<TimelinePanel
 | |
|                 timelineSet={timelineSet}
 | |
|                 manageReadMarkers={true}
 | |
|                 manageReadReceipts={true}
 | |
|                 eventId={ev0.getId()}
 | |
|             />);
 | |
|             const timelinePanel = component.instance() as TimelinePanel;
 | |
| 
 | |
|             // An event arrived, and we read it
 | |
|             timelineSet.addLiveEvent(ev1);
 | |
|             room.addEphemeralEvents([newReceipt("ev1", userId, 222, 220)]);
 | |
| 
 | |
|             // Sanity: We have not sent any read marker yet
 | |
|             expect(readMarkersSent).toEqual([]);
 | |
| 
 | |
|             // This is what we are testing: forget the read marker - this should
 | |
|             // update our read marker to match the latest receipt we sent
 | |
|             timelinePanel.forgetReadMarker();
 | |
| 
 | |
|             // We sent off a read marker for the new event
 | |
|             expect(readMarkersSent).toEqual(["ev1"]);
 | |
|         });
 | |
| 
 | |
|         it("sends public read receipt when enabled", () => {
 | |
|             const client = MatrixClientPeg.get();
 | |
|             const room = mkRoom(client, "roomId");
 | |
|             const events = mockEvents(room);
 | |
| 
 | |
|             const getValueCopy = SettingsStore.getValue;
 | |
|             SettingsStore.getValue = jest.fn().mockImplementation((name: string) => {
 | |
|                 if (name === "sendReadReceipts") return true;
 | |
|                 return getValueCopy(name);
 | |
|             });
 | |
| 
 | |
|             renderPanel(room, events);
 | |
|             expect(client.setRoomReadMarkers).toHaveBeenCalledWith(room.roomId, "", events[0], events[0]);
 | |
|         });
 | |
| 
 | |
|         it("does not send public read receipt when enabled", () => {
 | |
|             const client = MatrixClientPeg.get();
 | |
|             const room = mkRoom(client, "roomId");
 | |
|             const events = mockEvents(room);
 | |
| 
 | |
|             const getValueCopy = SettingsStore.getValue;
 | |
|             SettingsStore.getValue = jest.fn().mockImplementation((name: string) => {
 | |
|                 if (name === "sendReadReceipts") return false;
 | |
|                 return getValueCopy(name);
 | |
|             });
 | |
| 
 | |
|             renderPanel(room, events);
 | |
|             expect(client.setRoomReadMarkers).toHaveBeenCalledWith(room.roomId, "", undefined, events[0]);
 | |
|         });
 | |
|     });
 | |
| 
 | |
|     it("should scroll event into view when props.eventId changes", () => {
 | |
|         const client = MatrixClientPeg.get();
 | |
|         const room = mkRoom(client, "roomId");
 | |
|         const events = mockEvents(room);
 | |
| 
 | |
|         const props = {
 | |
|             ...getProps(room, events),
 | |
|             onEventScrolledIntoView: jest.fn(),
 | |
|         };
 | |
| 
 | |
|         const { rerender } = render(<TimelinePanel {...props} />);
 | |
|         expect(props.onEventScrolledIntoView).toHaveBeenCalledWith(undefined);
 | |
|         props.eventId = events[1].getId();
 | |
|         rerender(<TimelinePanel {...props} />);
 | |
|         expect(props.onEventScrolledIntoView).toHaveBeenCalledWith(events[1].getId());
 | |
|     });
 | |
| 
 | |
|     describe("when a thread updates", () => {
 | |
|         let client: MatrixClient;
 | |
|         let room: Room;
 | |
|         let allThreads: EventTimelineSet;
 | |
|         let root: MatrixEvent;
 | |
|         let reply1: MatrixEvent;
 | |
|         let reply2: MatrixEvent;
 | |
| 
 | |
|         beforeEach(() => {
 | |
|             client = MatrixClientPeg.get();
 | |
| 
 | |
|             Thread.hasServerSideSupport = FeatureSupport.Stable;
 | |
|             client.supportsExperimentalThreads = () => true;
 | |
|             const getValueCopy = SettingsStore.getValue;
 | |
|             SettingsStore.getValue = jest.fn().mockImplementation((name: string) => {
 | |
|                 if (name === "feature_thread") return true;
 | |
|                 return getValueCopy(name);
 | |
|             });
 | |
| 
 | |
|             room = new Room("roomId", client, "userId");
 | |
|             allThreads = new EventTimelineSet(room, {
 | |
|                 pendingEvents: false,
 | |
|             }, undefined, undefined, ThreadFilterType.All);
 | |
|             const timeline = new EventTimeline(allThreads);
 | |
|             allThreads.getLiveTimeline = () => timeline;
 | |
|             allThreads.getTimelineForEvent = () => timeline;
 | |
| 
 | |
|             reply1 = new MatrixEvent({
 | |
|                 room_id: room.roomId,
 | |
|                 event_id: 'event_reply_1',
 | |
|                 type: EventType.RoomMessage,
 | |
|                 user_id: "userId",
 | |
|                 content: MessageEvent.from(`ReplyEvent1`).serialize().content,
 | |
|             });
 | |
| 
 | |
|             reply2 = new MatrixEvent({
 | |
|                 room_id: room.roomId,
 | |
|                 event_id: 'event_reply_2',
 | |
|                 type: EventType.RoomMessage,
 | |
|                 user_id: "userId",
 | |
|                 content: MessageEvent.from(`ReplyEvent2`).serialize().content,
 | |
|             });
 | |
| 
 | |
|             root = new MatrixEvent({
 | |
|                 room_id: room.roomId,
 | |
|                 event_id: 'event_root_1',
 | |
|                 type: EventType.RoomMessage,
 | |
|                 user_id: "userId",
 | |
|                 content: MessageEvent.from(`RootEvent`).serialize().content,
 | |
|             });
 | |
| 
 | |
|             const eventMap: { [key: string]: MatrixEvent } = {
 | |
|                 [root.getId()!]: root,
 | |
|                 [reply1.getId()!]: reply1,
 | |
|                 [reply2.getId()!]: reply2,
 | |
|             };
 | |
| 
 | |
|             room.findEventById = (eventId: string) => eventMap[eventId];
 | |
|             client.fetchRoomEvent = async (roomId: string, eventId: string) =>
 | |
|                 roomId === room.roomId ? eventMap[eventId]?.event : {};
 | |
|         });
 | |
| 
 | |
|         it('updates thread previews', async () => {
 | |
|             root.setUnsigned({
 | |
|                 "m.relations": {
 | |
|                     [THREAD_RELATION_TYPE.name]: {
 | |
|                         "latest_event": reply1.event,
 | |
|                         "count": 1,
 | |
|                         "current_user_participated": true,
 | |
|                     },
 | |
|                 },
 | |
|             });
 | |
| 
 | |
|             const thread = room.createThread(root.getId()!, root, [], true);
 | |
|             // So that we do not have to mock the thread loading
 | |
|             thread.initialEventsFetched = true;
 | |
|             // @ts-ignore
 | |
|             thread.fetchEditsWhereNeeded = () => Promise.resolve();
 | |
|             await thread.addEvent(reply1, true);
 | |
|             await allThreads.getLiveTimeline().addEvent(thread.rootEvent!, true);
 | |
|             const replyToEvent = jest.spyOn(thread, "replyToEvent", "get");
 | |
| 
 | |
|             const dom = render(
 | |
|                 <MatrixClientContext.Provider value={client}>
 | |
|                     <TimelinePanel
 | |
|                         timelineSet={allThreads}
 | |
|                         manageReadReceipts
 | |
|                         sendReadReceiptOnLoad
 | |
|                     />
 | |
|                 </MatrixClientContext.Provider>,
 | |
|             );
 | |
|             await dom.findByText("RootEvent");
 | |
|             await dom.findByText("ReplyEvent1");
 | |
|             expect(replyToEvent).toHaveBeenCalled();
 | |
| 
 | |
|             root.setUnsigned({
 | |
|                 "m.relations": {
 | |
|                     [THREAD_RELATION_TYPE.name]: {
 | |
|                         "latest_event": reply2.event,
 | |
|                         "count": 2,
 | |
|                         "current_user_participated": true,
 | |
|                     },
 | |
|                 },
 | |
|             });
 | |
| 
 | |
|             replyToEvent.mockClear();
 | |
|             await thread.addEvent(reply2, false, true);
 | |
|             await dom.findByText("RootEvent");
 | |
|             await dom.findByText("ReplyEvent2");
 | |
|             expect(replyToEvent).toHaveBeenCalled();
 | |
|         });
 | |
| 
 | |
|         it('ignores thread updates for unknown threads', async () => {
 | |
|             root.setUnsigned({
 | |
|                 "m.relations": {
 | |
|                     [THREAD_RELATION_TYPE.name]: {
 | |
|                         "latest_event": reply1.event,
 | |
|                         "count": 1,
 | |
|                         "current_user_participated": true,
 | |
|                     },
 | |
|                 },
 | |
|             });
 | |
| 
 | |
|             const realThread = room.createThread(root.getId()!, root, [], true);
 | |
|             // So that we do not have to mock the thread loading
 | |
|             realThread.initialEventsFetched = true;
 | |
|             // @ts-ignore
 | |
|             realThread.fetchEditsWhereNeeded = () => Promise.resolve();
 | |
|             await realThread.addEvent(reply1, true);
 | |
|             await allThreads.getLiveTimeline().addEvent(realThread.rootEvent!, true);
 | |
|             const replyToEvent = jest.spyOn(realThread, "replyToEvent", "get");
 | |
| 
 | |
|             // @ts-ignore
 | |
|             const fakeThread1: Thread = {
 | |
|                 id: undefined!,
 | |
|                 get roomId(): string {
 | |
|                     return room.roomId;
 | |
|                 },
 | |
|             };
 | |
| 
 | |
|             const fakeRoom = new Room("thisroomdoesnotexist", client, "userId");
 | |
|             // @ts-ignore
 | |
|             const fakeThread2: Thread = {
 | |
|                 id: root.getId()!,
 | |
|                 get roomId(): string {
 | |
|                     return fakeRoom.roomId;
 | |
|                 },
 | |
|             };
 | |
| 
 | |
|             const dom = render(
 | |
|                 <MatrixClientContext.Provider value={client}>
 | |
|                     <TimelinePanel
 | |
|                         timelineSet={allThreads}
 | |
|                         manageReadReceipts
 | |
|                         sendReadReceiptOnLoad
 | |
|                     />
 | |
|                 </MatrixClientContext.Provider>,
 | |
|             );
 | |
|             await dom.findByText("RootEvent");
 | |
|             await dom.findByText("ReplyEvent1");
 | |
|             expect(replyToEvent).toHaveBeenCalled();
 | |
| 
 | |
|             replyToEvent.mockClear();
 | |
|             room.emit(ThreadEvent.Update, fakeThread1);
 | |
|             room.emit(ThreadEvent.Update, fakeThread2);
 | |
|             await dom.findByText("ReplyEvent1");
 | |
|             expect(replyToEvent).not.toHaveBeenCalled();
 | |
|             replyToEvent.mockClear();
 | |
|         });
 | |
|     });
 | |
| });
 |