2022-10-17 12:31:03 +00:00
|
|
|
/*
|
|
|
|
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 { mocked } from "jest-mock";
|
|
|
|
import {
|
|
|
|
EventTimelineSet,
|
|
|
|
EventType,
|
|
|
|
MatrixClient,
|
|
|
|
MatrixEvent,
|
|
|
|
MatrixEventEvent,
|
|
|
|
RelationType,
|
|
|
|
Room,
|
|
|
|
} from "matrix-js-sdk/src/matrix";
|
|
|
|
import { Relations } from "matrix-js-sdk/src/models/relations";
|
|
|
|
import { RelationsContainer } from "matrix-js-sdk/src/models/relations-container";
|
|
|
|
|
|
|
|
import { RelationsHelper, RelationsHelperEvent } from "../../src/events/RelationsHelper";
|
2022-11-07 14:19:49 +00:00
|
|
|
import { mkEvent, mkRelationsContainer, mkStubRoom, stubClient } from "../test-utils";
|
2022-10-17 12:31:03 +00:00
|
|
|
|
|
|
|
describe("RelationsHelper", () => {
|
|
|
|
const roomId = "!room:example.com";
|
2022-11-07 14:19:49 +00:00
|
|
|
let userId: string;
|
2022-10-17 12:31:03 +00:00
|
|
|
let event: MatrixEvent;
|
|
|
|
let relatedEvent1: MatrixEvent;
|
|
|
|
let relatedEvent2: MatrixEvent;
|
2022-11-07 14:19:49 +00:00
|
|
|
let relatedEvent3: MatrixEvent;
|
2022-10-17 12:31:03 +00:00
|
|
|
let room: Room;
|
|
|
|
let client: MatrixClient;
|
|
|
|
let relationsHelper: RelationsHelper;
|
|
|
|
let onAdd: (event: MatrixEvent) => void;
|
|
|
|
let timelineSet: EventTimelineSet;
|
|
|
|
let relationsContainer: RelationsContainer;
|
|
|
|
let relations: Relations;
|
|
|
|
let relationsOnAdd: (event: MatrixEvent) => void;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
client = stubClient();
|
2022-11-07 14:19:49 +00:00
|
|
|
userId = client.getUserId() || "";
|
|
|
|
mocked(client.relations).mockClear();
|
2022-10-17 12:31:03 +00:00
|
|
|
room = mkStubRoom(roomId, "test room", client);
|
2022-11-07 14:19:49 +00:00
|
|
|
mocked(client.getRoom).mockImplementation((getRoomId?: string) => {
|
2022-10-17 12:31:03 +00:00
|
|
|
if (getRoomId === roomId) {
|
|
|
|
return room;
|
|
|
|
}
|
2022-11-07 14:19:49 +00:00
|
|
|
|
|
|
|
return null;
|
2022-10-17 12:31:03 +00:00
|
|
|
});
|
|
|
|
event = mkEvent({
|
|
|
|
event: true,
|
|
|
|
type: EventType.RoomMessage,
|
|
|
|
room: roomId,
|
2022-11-07 14:19:49 +00:00
|
|
|
user: userId,
|
2022-10-17 12:31:03 +00:00
|
|
|
content: {},
|
|
|
|
});
|
|
|
|
relatedEvent1 = mkEvent({
|
|
|
|
event: true,
|
|
|
|
type: EventType.RoomMessage,
|
|
|
|
room: roomId,
|
2022-11-07 14:19:49 +00:00
|
|
|
user: userId,
|
|
|
|
content: { relatedEvent: 1 },
|
2022-10-17 12:31:03 +00:00
|
|
|
});
|
|
|
|
relatedEvent2 = mkEvent({
|
|
|
|
event: true,
|
|
|
|
type: EventType.RoomMessage,
|
|
|
|
room: roomId,
|
2022-11-07 14:19:49 +00:00
|
|
|
user: userId,
|
|
|
|
content: { relatedEvent: 2 },
|
|
|
|
});
|
|
|
|
relatedEvent3 = mkEvent({
|
|
|
|
event: true,
|
|
|
|
type: EventType.RoomMessage,
|
|
|
|
room: roomId,
|
|
|
|
user: userId,
|
|
|
|
content: { relatedEvent: 3 },
|
2022-10-17 12:31:03 +00:00
|
|
|
});
|
|
|
|
onAdd = jest.fn();
|
2022-11-07 14:19:49 +00:00
|
|
|
relationsContainer = mkRelationsContainer();
|
2022-10-17 12:31:03 +00:00
|
|
|
// TODO Michael W: create test utils, remove casts
|
|
|
|
relations = {
|
|
|
|
getRelations: jest.fn(),
|
2022-12-12 11:24:14 +00:00
|
|
|
on: jest.fn().mockImplementation((type, l) => (relationsOnAdd = l)),
|
2022-11-07 14:19:49 +00:00
|
|
|
off: jest.fn(),
|
2022-10-17 12:31:03 +00:00
|
|
|
} as unknown as Relations;
|
|
|
|
timelineSet = {
|
|
|
|
relations: relationsContainer,
|
|
|
|
} as unknown as EventTimelineSet;
|
|
|
|
});
|
|
|
|
|
2022-11-07 14:19:49 +00:00
|
|
|
afterEach(() => {
|
|
|
|
relationsHelper?.destroy();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when there is an event without ID", () => {
|
|
|
|
it("should raise an error", () => {
|
|
|
|
jest.spyOn(event, "getId").mockReturnValue(undefined);
|
|
|
|
|
|
|
|
expect(() => {
|
|
|
|
new RelationsHelper(event, RelationType.Reference, EventType.RoomMessage, client);
|
|
|
|
}).toThrowError("unable to create RelationsHelper: missing event ID");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when there is an event without room ID", () => {
|
|
|
|
it("should raise an error", () => {
|
|
|
|
jest.spyOn(event, "getRoomId").mockReturnValue(undefined);
|
|
|
|
|
|
|
|
expect(() => {
|
|
|
|
new RelationsHelper(event, RelationType.Reference, EventType.RoomMessage, client);
|
|
|
|
}).toThrowError("unable to create RelationsHelper: missing room ID");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-10-17 12:31:03 +00:00
|
|
|
describe("when there is an event without relations", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
relationsHelper = new RelationsHelper(event, RelationType.Reference, EventType.RoomMessage, client);
|
|
|
|
relationsHelper.on(RelationsHelperEvent.Add, onAdd);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("emitCurrent", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
relationsHelper.emitCurrent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should not emit any event", () => {
|
|
|
|
expect(onAdd).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("and relations are created and a new event appears", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
mocked(room.getUnfilteredTimelineSet).mockReturnValue(timelineSet);
|
|
|
|
mocked(relationsContainer.getChildEventsForEvent).mockReturnValue(relations);
|
|
|
|
mocked(relations.getRelations).mockReturnValue([relatedEvent1]);
|
|
|
|
event.emit(MatrixEventEvent.RelationsCreated, RelationType.Reference, EventType.RoomMessage);
|
|
|
|
relationsOnAdd(relatedEvent2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should emit the new event", () => {
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent2);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-11-07 14:19:49 +00:00
|
|
|
describe("when there is an event with two pages server side relations", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
mocked(client.relations)
|
|
|
|
.mockResolvedValueOnce({
|
|
|
|
events: [relatedEvent1, relatedEvent2],
|
|
|
|
nextBatch: "next",
|
|
|
|
})
|
|
|
|
.mockResolvedValueOnce({
|
|
|
|
events: [relatedEvent3],
|
|
|
|
nextBatch: null,
|
|
|
|
});
|
|
|
|
relationsHelper = new RelationsHelper(event, RelationType.Reference, EventType.RoomMessage, client);
|
|
|
|
relationsHelper.on(RelationsHelperEvent.Add, onAdd);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("emitFetchCurrent", () => {
|
|
|
|
beforeEach(async () => {
|
|
|
|
await relationsHelper.emitFetchCurrent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should emit the server side events", () => {
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent1);
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent2);
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent3);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2022-10-17 12:31:03 +00:00
|
|
|
describe("when there is an event with relations", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
mocked(room.getUnfilteredTimelineSet).mockReturnValue(timelineSet);
|
|
|
|
mocked(relationsContainer.getChildEventsForEvent).mockReturnValue(relations);
|
|
|
|
mocked(relations.getRelations).mockReturnValue([relatedEvent1]);
|
|
|
|
relationsHelper = new RelationsHelper(event, RelationType.Reference, EventType.RoomMessage, client);
|
|
|
|
relationsHelper.on(RelationsHelperEvent.Add, onAdd);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("emitCurrent", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
relationsHelper.emitCurrent();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should emit the related event", () => {
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("and a new event appears", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
relationsOnAdd(relatedEvent2);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("should emit the new event", () => {
|
|
|
|
expect(onAdd).toHaveBeenCalledWith(relatedEvent2);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|