mirror of https://github.com/vector-im/riot-web
				
				
				
			
		
			
				
	
	
		
			244 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			244 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			TypeScript
		
	
	
/*
 | 
						|
Copyright 2021 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 './skinned-sdk';
 | 
						|
 | 
						|
import { CallEvent, CallState, CallType } from 'matrix-js-sdk/src/webrtc/call';
 | 
						|
import EventEmitter from 'events';
 | 
						|
 | 
						|
import CallHandler, { CallHandlerEvent } from '../src/CallHandler';
 | 
						|
import { stubClient, mkStubRoom } from './test-utils';
 | 
						|
import { MatrixClientPeg } from '../src/MatrixClientPeg';
 | 
						|
import dis from '../src/dispatcher/dispatcher';
 | 
						|
import DMRoomMap from '../src/utils/DMRoomMap';
 | 
						|
import SdkConfig from '../src/SdkConfig';
 | 
						|
import { ActionPayload } from '../src/dispatcher/payloads';
 | 
						|
import { Action } from "../src/dispatcher/actions";
 | 
						|
 | 
						|
const REAL_ROOM_ID = '$room1:example.org';
 | 
						|
const MAPPED_ROOM_ID = '$room2:example.org';
 | 
						|
const MAPPED_ROOM_ID_2 = '$room3:example.org';
 | 
						|
 | 
						|
function mkStubDM(roomId, userId) {
 | 
						|
    const room = mkStubRoom(roomId);
 | 
						|
    room.getJoinedMembers = jest.fn().mockReturnValue([
 | 
						|
        {
 | 
						|
            userId: '@me:example.org',
 | 
						|
            name: 'Member',
 | 
						|
            rawDisplayName: 'Member',
 | 
						|
            roomId: roomId,
 | 
						|
            membership: 'join',
 | 
						|
            getAvatarUrl: () => 'mxc://avatar.url/image.png',
 | 
						|
            getMxcAvatarUrl: () => 'mxc://avatar.url/image.png',
 | 
						|
        },
 | 
						|
        {
 | 
						|
            userId: userId,
 | 
						|
            name: 'Member',
 | 
						|
            rawDisplayName: 'Member',
 | 
						|
            roomId: roomId,
 | 
						|
            membership: 'join',
 | 
						|
            getAvatarUrl: () => 'mxc://avatar.url/image.png',
 | 
						|
            getMxcAvatarUrl: () => 'mxc://avatar.url/image.png',
 | 
						|
        },
 | 
						|
    ]);
 | 
						|
    room.currentState.getMembers = room.getJoinedMembers;
 | 
						|
 | 
						|
    return room;
 | 
						|
}
 | 
						|
 | 
						|
class FakeCall extends EventEmitter {
 | 
						|
    roomId: string;
 | 
						|
    callId = "fake call id";
 | 
						|
 | 
						|
    constructor(roomId) {
 | 
						|
        super();
 | 
						|
 | 
						|
        this.roomId = roomId;
 | 
						|
    }
 | 
						|
 | 
						|
    setRemoteOnHold() {}
 | 
						|
    setRemoteAudioElement() {}
 | 
						|
 | 
						|
    placeVoiceCall() {
 | 
						|
        this.emit(CallEvent.State, CallState.Connected, null);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function untilDispatch(waitForAction: string): Promise<ActionPayload> {
 | 
						|
    let dispatchHandle;
 | 
						|
    return new Promise<ActionPayload>(resolve => {
 | 
						|
        dispatchHandle = dis.register(payload => {
 | 
						|
            if (payload.action === waitForAction) {
 | 
						|
                dis.unregister(dispatchHandle);
 | 
						|
                resolve(payload);
 | 
						|
            }
 | 
						|
        });
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
function untilCallHandlerEvent(callHandler: CallHandler, event: CallHandlerEvent): Promise<void> {
 | 
						|
    return new Promise<void>((resolve) => {
 | 
						|
        callHandler.addListener(event, () => {
 | 
						|
            resolve();
 | 
						|
        });
 | 
						|
    });
 | 
						|
}
 | 
						|
 | 
						|
describe('CallHandler', () => {
 | 
						|
    let dmRoomMap;
 | 
						|
    let callHandler;
 | 
						|
    let audioElement;
 | 
						|
    let fakeCall;
 | 
						|
 | 
						|
    beforeEach(() => {
 | 
						|
        stubClient();
 | 
						|
        MatrixClientPeg.get().createCall = roomId => {
 | 
						|
            if (fakeCall && fakeCall.roomId !== roomId) {
 | 
						|
                throw new Error("Only one call is supported!");
 | 
						|
            }
 | 
						|
            fakeCall = new FakeCall(roomId);
 | 
						|
            return fakeCall;
 | 
						|
        };
 | 
						|
 | 
						|
        callHandler = new CallHandler();
 | 
						|
        callHandler.start();
 | 
						|
 | 
						|
        const realRoom = mkStubDM(REAL_ROOM_ID, '@user1:example.org');
 | 
						|
        const mappedRoom = mkStubDM(MAPPED_ROOM_ID, '@user2:example.org');
 | 
						|
        const mappedRoom2 = mkStubDM(MAPPED_ROOM_ID_2, '@user3:example.org');
 | 
						|
 | 
						|
        MatrixClientPeg.get().getRoom = roomId => {
 | 
						|
            switch (roomId) {
 | 
						|
                case REAL_ROOM_ID:
 | 
						|
                    return realRoom;
 | 
						|
                case MAPPED_ROOM_ID:
 | 
						|
                    return mappedRoom;
 | 
						|
                case MAPPED_ROOM_ID_2:
 | 
						|
                    return mappedRoom2;
 | 
						|
            }
 | 
						|
        };
 | 
						|
 | 
						|
        dmRoomMap = {
 | 
						|
            getUserIdForRoomId: roomId => {
 | 
						|
                if (roomId === REAL_ROOM_ID) {
 | 
						|
                    return '@user1:example.org';
 | 
						|
                } else if (roomId === MAPPED_ROOM_ID) {
 | 
						|
                    return '@user2:example.org';
 | 
						|
                } else if (roomId === MAPPED_ROOM_ID_2) {
 | 
						|
                    return '@user3:example.org';
 | 
						|
                } else {
 | 
						|
                    return null;
 | 
						|
                }
 | 
						|
            },
 | 
						|
            getDMRoomsForUserId: userId => {
 | 
						|
                if (userId === '@user2:example.org') {
 | 
						|
                    return [MAPPED_ROOM_ID];
 | 
						|
                } else if (userId === '@user3:example.org') {
 | 
						|
                    return [MAPPED_ROOM_ID_2];
 | 
						|
                } else {
 | 
						|
                    return [];
 | 
						|
                }
 | 
						|
            },
 | 
						|
        };
 | 
						|
        DMRoomMap.setShared(dmRoomMap);
 | 
						|
 | 
						|
        audioElement = document.createElement('audio');
 | 
						|
        audioElement.id = "remoteAudio";
 | 
						|
        document.body.appendChild(audioElement);
 | 
						|
    });
 | 
						|
 | 
						|
    afterEach(() => {
 | 
						|
        callHandler.stop();
 | 
						|
        DMRoomMap.setShared(null);
 | 
						|
        // @ts-ignore
 | 
						|
        window.mxCallHandler = null;
 | 
						|
        fakeCall = null;
 | 
						|
        MatrixClientPeg.unset();
 | 
						|
 | 
						|
        document.body.removeChild(audioElement);
 | 
						|
        SdkConfig.unset();
 | 
						|
    });
 | 
						|
 | 
						|
    it('should look up the correct user and start a call in the room when a phone number is dialled', async () => {
 | 
						|
        MatrixClientPeg.get().getThirdpartyUser = jest.fn().mockResolvedValue([{
 | 
						|
            userid: '@user2:example.org',
 | 
						|
            protocol: "im.vector.protocol.sip_native",
 | 
						|
            fields: {
 | 
						|
                is_native: true,
 | 
						|
                lookup_success: true,
 | 
						|
            },
 | 
						|
        }]);
 | 
						|
 | 
						|
        await callHandler.dialNumber('01818118181');
 | 
						|
 | 
						|
        const viewRoomPayload = await untilDispatch(Action.ViewRoom);
 | 
						|
        expect(viewRoomPayload.room_id).toEqual(MAPPED_ROOM_ID);
 | 
						|
 | 
						|
        // Check that a call was started
 | 
						|
        expect(fakeCall.roomId).toEqual(MAPPED_ROOM_ID);
 | 
						|
    });
 | 
						|
 | 
						|
    it('should move calls between rooms when remote asserted identity changes', async () => {
 | 
						|
        callHandler.placeCall(REAL_ROOM_ID, CallType.Voice);
 | 
						|
 | 
						|
        await untilCallHandlerEvent(callHandler, CallHandlerEvent.CallState);
 | 
						|
 | 
						|
        // should start off in the actual room ID it's in at the protocol level
 | 
						|
        expect(callHandler.getCallForRoom(REAL_ROOM_ID)).toBe(fakeCall);
 | 
						|
 | 
						|
        let callRoomChangeEventCount = 0;
 | 
						|
        const roomChangePromise = new Promise<void>(resolve => {
 | 
						|
            callHandler.addListener(CallHandlerEvent.CallChangeRoom, () => {
 | 
						|
                ++callRoomChangeEventCount;
 | 
						|
                resolve();
 | 
						|
            });
 | 
						|
        });
 | 
						|
 | 
						|
        // Now emit an asserted identity for user2: this should be ignored
 | 
						|
        // because we haven't set the config option to obey asserted identity
 | 
						|
        fakeCall.getRemoteAssertedIdentity = jest.fn().mockReturnValue({
 | 
						|
            id: "@user2:example.org",
 | 
						|
        });
 | 
						|
        fakeCall.emit(CallEvent.AssertedIdentityChanged);
 | 
						|
 | 
						|
        // Now set the config option
 | 
						|
        SdkConfig.put({
 | 
						|
            voip: {
 | 
						|
                obeyAssertedIdentity: true,
 | 
						|
            },
 | 
						|
        });
 | 
						|
 | 
						|
        // ...and send another asserted identity event for a different user
 | 
						|
        fakeCall.getRemoteAssertedIdentity = jest.fn().mockReturnValue({
 | 
						|
            id: "@user3:example.org",
 | 
						|
        });
 | 
						|
        fakeCall.emit(CallEvent.AssertedIdentityChanged);
 | 
						|
 | 
						|
        await roomChangePromise;
 | 
						|
        callHandler.removeAllListeners();
 | 
						|
 | 
						|
        // If everything's gone well, we should have seen only one room change
 | 
						|
        // event and the call should now be in user 3's room.
 | 
						|
        // If it's not obeying any, the call will still be in REAL_ROOM_ID.
 | 
						|
        // If it incorrectly obeyed both asserted identity changes, either it will
 | 
						|
        // have just processed one and the call will be in the wrong room, or we'll
 | 
						|
        // have seen two room change dispatches.
 | 
						|
        expect(callRoomChangeEventCount).toEqual(1);
 | 
						|
        expect(callHandler.getCallForRoom(REAL_ROOM_ID)).toBeNull();
 | 
						|
        expect(callHandler.getCallForRoom(MAPPED_ROOM_ID_2)).toBe(fakeCall);
 | 
						|
    });
 | 
						|
});
 |