You've already forked matrix-js-sdk
mirror of
https://github.com/matrix-org/matrix-js-sdk.git
synced 2025-08-10 21:23:02 +03:00
* WIP doodles on MembershipManager test cases * . * initial membership manager test setup. * Updates from discussion * revert renaming comments * remove unused import * fix leave delayed event resend test. It was missing a flush. * comment out and remove unused variables * es lint * use jsdom instead of node test environment * remove unused variables * remove unused export * temp * review * fixup tests * more review * remove wait for expect dependency * flatten tests and add comments * add more leave test cases * use defer * remove @jest/environment dependency * Cleanup awaits and Make mock types more correct. Make every mock return a Promise if the real implementation does return a pormise. * remove flush promise dependency * add linting to matrixrtc tests * Add fix async lints and use matrix rtc logger for test environment. * prettier * change to MatrixRTCSession logger * make accessing the full room deprecated * remove deprecated usage of full room * Clean up the deprecation --------- Co-authored-by: Hugh Nimmo-Smith <hughns@matrix.org>
194 lines
8.2 KiB
TypeScript
194 lines
8.2 KiB
TypeScript
/*
|
|
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 { type Mock } from "jest-mock";
|
|
|
|
import {
|
|
ClientEvent,
|
|
EventTimeline,
|
|
EventType,
|
|
type IRoomTimelineData,
|
|
MatrixClient,
|
|
type MatrixEvent,
|
|
RoomEvent,
|
|
} from "../../../src";
|
|
import { RoomStateEvent } from "../../../src/models/room-state";
|
|
import { MatrixRTCSessionManagerEvents } from "../../../src/matrixrtc/MatrixRTCSessionManager";
|
|
import { makeMockRoom, makeMockRoomState, membershipTemplate } from "./mocks";
|
|
|
|
describe("MatrixRTCSessionManager", () => {
|
|
let client: MatrixClient;
|
|
|
|
beforeEach(() => {
|
|
client = new MatrixClient({ baseUrl: "base_url" });
|
|
client.matrixRTC.start();
|
|
});
|
|
|
|
afterEach(() => {
|
|
client.stopClient();
|
|
client.matrixRTC.stop();
|
|
});
|
|
|
|
it("Fires event when session starts", () => {
|
|
const onStarted = jest.fn();
|
|
client.matrixRTC.on(MatrixRTCSessionManagerEvents.SessionStarted, onStarted);
|
|
|
|
try {
|
|
const room1 = makeMockRoom([membershipTemplate]);
|
|
jest.spyOn(client, "getRooms").mockReturnValue([room1]);
|
|
|
|
client.emit(ClientEvent.Room, room1);
|
|
expect(onStarted).toHaveBeenCalledWith(room1.roomId, client.matrixRTC.getActiveRoomSession(room1));
|
|
} finally {
|
|
client.matrixRTC.off(MatrixRTCSessionManagerEvents.SessionStarted, onStarted);
|
|
}
|
|
});
|
|
|
|
it("Fires event when session ends", () => {
|
|
const onEnded = jest.fn();
|
|
client.matrixRTC.on(MatrixRTCSessionManagerEvents.SessionEnded, onEnded);
|
|
const room1 = makeMockRoom(membershipTemplate);
|
|
jest.spyOn(client, "getRooms").mockReturnValue([room1]);
|
|
jest.spyOn(client, "getRoom").mockReturnValue(room1);
|
|
|
|
client.emit(ClientEvent.Room, room1);
|
|
|
|
(room1.getLiveTimeline as Mock).mockReturnValue({
|
|
getState: jest.fn().mockReturnValue(makeMockRoomState([{}], room1.roomId)),
|
|
});
|
|
|
|
const roomState = room1.getLiveTimeline().getState(EventTimeline.FORWARDS)!;
|
|
const membEvent = roomState.getStateEvents("")[0];
|
|
|
|
client.emit(RoomStateEvent.Events, membEvent, roomState, null);
|
|
|
|
expect(onEnded).toHaveBeenCalledWith(room1.roomId, client.matrixRTC.getActiveRoomSession(room1));
|
|
});
|
|
|
|
it("Calls onCallEncryption on encryption keys event", async () => {
|
|
const room1 = makeMockRoom([membershipTemplate]);
|
|
jest.spyOn(client, "getRooms").mockReturnValue([room1]);
|
|
jest.spyOn(client, "getRoom").mockReturnValue(room1);
|
|
|
|
client.emit(ClientEvent.Room, room1);
|
|
const onCallEncryptionMock = jest.fn();
|
|
client.matrixRTC.getRoomSession(room1).onCallEncryption = onCallEncryptionMock;
|
|
client.decryptEventIfNeeded = () => Promise.resolve();
|
|
const timelineEvent = {
|
|
getType: jest.fn().mockReturnValue(EventType.CallEncryptionKeysPrefix),
|
|
getContent: jest.fn().mockReturnValue({}),
|
|
getSender: jest.fn().mockReturnValue("@mock:user.example"),
|
|
getRoomId: jest.fn().mockReturnValue("!room:id"),
|
|
isDecryptionFailure: jest.fn().mockReturnValue(false),
|
|
sender: {
|
|
userId: "@mock:user.example",
|
|
},
|
|
} as unknown as MatrixEvent;
|
|
client.emit(RoomEvent.Timeline, timelineEvent, undefined, undefined, false, {} as IRoomTimelineData);
|
|
await new Promise(process.nextTick);
|
|
expect(onCallEncryptionMock).toHaveBeenCalled();
|
|
});
|
|
|
|
describe("event decryption", () => {
|
|
it("Retries decryption and processes success", async () => {
|
|
try {
|
|
jest.useFakeTimers();
|
|
const room1 = makeMockRoom([membershipTemplate]);
|
|
jest.spyOn(client, "getRooms").mockReturnValue([room1]);
|
|
jest.spyOn(client, "getRoom").mockReturnValue(room1);
|
|
|
|
client.emit(ClientEvent.Room, room1);
|
|
const onCallEncryptionMock = jest.fn();
|
|
client.matrixRTC.getRoomSession(room1).onCallEncryption = onCallEncryptionMock;
|
|
let isDecryptionFailure = true;
|
|
client.decryptEventIfNeeded = jest
|
|
.fn()
|
|
.mockReturnValueOnce(Promise.resolve())
|
|
.mockImplementation(() => {
|
|
isDecryptionFailure = false;
|
|
return Promise.resolve();
|
|
});
|
|
const timelineEvent = {
|
|
getType: jest.fn().mockReturnValue(EventType.CallEncryptionKeysPrefix),
|
|
getContent: jest.fn().mockReturnValue({}),
|
|
getSender: jest.fn().mockReturnValue("@mock:user.example"),
|
|
getRoomId: jest.fn().mockReturnValue("!room:id"),
|
|
isDecryptionFailure: jest.fn().mockImplementation(() => isDecryptionFailure),
|
|
getId: jest.fn().mockReturnValue("event_id"),
|
|
sender: {
|
|
userId: "@mock:user.example",
|
|
},
|
|
} as unknown as MatrixEvent;
|
|
client.emit(RoomEvent.Timeline, timelineEvent, undefined, undefined, false, {} as IRoomTimelineData);
|
|
|
|
expect(client.decryptEventIfNeeded).toHaveBeenCalledTimes(1);
|
|
expect(onCallEncryptionMock).toHaveBeenCalledTimes(0);
|
|
|
|
// should retry after one second:
|
|
await jest.advanceTimersByTimeAsync(1500);
|
|
|
|
expect(client.decryptEventIfNeeded).toHaveBeenCalledTimes(2);
|
|
expect(onCallEncryptionMock).toHaveBeenCalledTimes(1);
|
|
} finally {
|
|
jest.useRealTimers();
|
|
}
|
|
});
|
|
|
|
it("Retries decryption and processes failure", async () => {
|
|
try {
|
|
jest.useFakeTimers();
|
|
const room1 = makeMockRoom([membershipTemplate]);
|
|
jest.spyOn(client, "getRooms").mockReturnValue([room1]);
|
|
jest.spyOn(client, "getRoom").mockReturnValue(room1);
|
|
|
|
client.emit(ClientEvent.Room, room1);
|
|
const onCallEncryptionMock = jest.fn();
|
|
client.matrixRTC.getRoomSession(room1).onCallEncryption = onCallEncryptionMock;
|
|
client.decryptEventIfNeeded = jest.fn().mockReturnValue(Promise.resolve());
|
|
const timelineEvent = {
|
|
getType: jest.fn().mockReturnValue(EventType.CallEncryptionKeysPrefix),
|
|
getContent: jest.fn().mockReturnValue({}),
|
|
getSender: jest.fn().mockReturnValue("@mock:user.example"),
|
|
getRoomId: jest.fn().mockReturnValue("!room:id"),
|
|
isDecryptionFailure: jest.fn().mockReturnValue(true), // always fail
|
|
getId: jest.fn().mockReturnValue("event_id"),
|
|
sender: {
|
|
userId: "@mock:user.example",
|
|
},
|
|
} as unknown as MatrixEvent;
|
|
client.emit(RoomEvent.Timeline, timelineEvent, undefined, undefined, false, {} as IRoomTimelineData);
|
|
|
|
expect(client.decryptEventIfNeeded).toHaveBeenCalledTimes(1);
|
|
expect(onCallEncryptionMock).toHaveBeenCalledTimes(0);
|
|
|
|
// should retry after one second:
|
|
await jest.advanceTimersByTimeAsync(1500);
|
|
|
|
expect(client.decryptEventIfNeeded).toHaveBeenCalledTimes(2);
|
|
expect(onCallEncryptionMock).toHaveBeenCalledTimes(0);
|
|
|
|
// doesn't retry again:
|
|
await jest.advanceTimersByTimeAsync(1500);
|
|
|
|
expect(client.decryptEventIfNeeded).toHaveBeenCalledTimes(2);
|
|
expect(onCallEncryptionMock).toHaveBeenCalledTimes(0);
|
|
} finally {
|
|
jest.useRealTimers();
|
|
}
|
|
});
|
|
});
|
|
});
|