Merge pull request #5886 from matrix-org/dbkr/asserted_identity
Support MSC3086 asserted identity
This commit is contained in:
commit
c95c1aeffd
10 changed files with 315 additions and 19 deletions
|
@ -59,7 +59,6 @@ import {MatrixClientPeg} from './MatrixClientPeg';
|
||||||
import PlatformPeg from './PlatformPeg';
|
import PlatformPeg from './PlatformPeg';
|
||||||
import Modal from './Modal';
|
import Modal from './Modal';
|
||||||
import { _t } from './languageHandler';
|
import { _t } from './languageHandler';
|
||||||
import { createNewMatrixCall } from 'matrix-js-sdk/src/webrtc/call';
|
|
||||||
import dis from './dispatcher/dispatcher';
|
import dis from './dispatcher/dispatcher';
|
||||||
import WidgetUtils from './utils/WidgetUtils';
|
import WidgetUtils from './utils/WidgetUtils';
|
||||||
import WidgetEchoStore from './stores/WidgetEchoStore';
|
import WidgetEchoStore from './stores/WidgetEchoStore';
|
||||||
|
@ -86,6 +85,8 @@ import { Action } from './dispatcher/actions';
|
||||||
import VoipUserMapper from './VoipUserMapper';
|
import VoipUserMapper from './VoipUserMapper';
|
||||||
import { addManagedHybridWidget, isManagedHybridWidgetEnabled } from './widgets/ManagedHybrid';
|
import { addManagedHybridWidget, isManagedHybridWidgetEnabled } from './widgets/ManagedHybrid';
|
||||||
import { randomUppercaseString, randomLowercaseString } from "matrix-js-sdk/src/randomstring";
|
import { randomUppercaseString, randomLowercaseString } from "matrix-js-sdk/src/randomstring";
|
||||||
|
import SdkConfig from './SdkConfig';
|
||||||
|
import { ensureDMExists, findDMForUser } from './createRoom';
|
||||||
|
|
||||||
export const PROTOCOL_PSTN = 'm.protocol.pstn';
|
export const PROTOCOL_PSTN = 'm.protocol.pstn';
|
||||||
export const PROTOCOL_PSTN_PREFIXED = 'im.vector.protocol.pstn';
|
export const PROTOCOL_PSTN_PREFIXED = 'im.vector.protocol.pstn';
|
||||||
|
@ -167,6 +168,11 @@ export default class CallHandler {
|
||||||
private invitedRoomsAreVirtual = new Map<string, boolean>();
|
private invitedRoomsAreVirtual = new Map<string, boolean>();
|
||||||
private invitedRoomCheckInProgress = false;
|
private invitedRoomCheckInProgress = false;
|
||||||
|
|
||||||
|
// Map of the asserted identity users after we've looked them up using the API.
|
||||||
|
// We need to be be able to determine the mapped room synchronously, so we
|
||||||
|
// do the async lookup when we get new information and then store these mappings here
|
||||||
|
private assertedIdentityNativeUsers = new Map<string, string>();
|
||||||
|
|
||||||
static sharedInstance() {
|
static sharedInstance() {
|
||||||
if (!window.mxCallHandler) {
|
if (!window.mxCallHandler) {
|
||||||
window.mxCallHandler = new CallHandler()
|
window.mxCallHandler = new CallHandler()
|
||||||
|
@ -179,8 +185,19 @@ export default class CallHandler {
|
||||||
* Gets the user-facing room associated with a call (call.roomId may be the call "virtual room"
|
* Gets the user-facing room associated with a call (call.roomId may be the call "virtual room"
|
||||||
* if a voip_mxid_translate_pattern is set in the config)
|
* if a voip_mxid_translate_pattern is set in the config)
|
||||||
*/
|
*/
|
||||||
public static roomIdForCall(call: MatrixCall): string {
|
public roomIdForCall(call: MatrixCall): string {
|
||||||
if (!call) return null;
|
if (!call) return null;
|
||||||
|
|
||||||
|
const voipConfig = SdkConfig.get()['voip'];
|
||||||
|
|
||||||
|
if (voipConfig && voipConfig.obeyAssertedIdentity) {
|
||||||
|
const nativeUser = this.assertedIdentityNativeUsers[call.callId];
|
||||||
|
if (nativeUser) {
|
||||||
|
const room = findDMForUser(MatrixClientPeg.get(), nativeUser);
|
||||||
|
if (room) return room.roomId
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return VoipUserMapper.sharedInstance().nativeRoomForVirtualRoom(call.roomId) || call.roomId;
|
return VoipUserMapper.sharedInstance().nativeRoomForVirtualRoom(call.roomId) || call.roomId;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -379,14 +396,14 @@ export default class CallHandler {
|
||||||
// We don't allow placing more than one call per room, but that doesn't mean there
|
// We don't allow placing more than one call per room, but that doesn't mean there
|
||||||
// can't be more than one, eg. in a glare situation. This checks that the given call
|
// can't be more than one, eg. in a glare situation. This checks that the given call
|
||||||
// is the call we consider 'the' call for its room.
|
// is the call we consider 'the' call for its room.
|
||||||
const mappedRoomId = CallHandler.roomIdForCall(call);
|
const mappedRoomId = this.roomIdForCall(call);
|
||||||
|
|
||||||
const callForThisRoom = this.getCallForRoom(mappedRoomId);
|
const callForThisRoom = this.getCallForRoom(mappedRoomId);
|
||||||
return callForThisRoom && call.callId === callForThisRoom.callId;
|
return callForThisRoom && call.callId === callForThisRoom.callId;
|
||||||
}
|
}
|
||||||
|
|
||||||
private setCallListeners(call: MatrixCall) {
|
private setCallListeners(call: MatrixCall) {
|
||||||
const mappedRoomId = CallHandler.roomIdForCall(call);
|
let mappedRoomId = CallHandler.sharedInstance().roomIdForCall(call);
|
||||||
|
|
||||||
call.on(CallEvent.Error, (err: CallError) => {
|
call.on(CallEvent.Error, (err: CallError) => {
|
||||||
if (!this.matchesCallForThisRoom(call)) return;
|
if (!this.matchesCallForThisRoom(call)) return;
|
||||||
|
@ -500,6 +517,42 @@ export default class CallHandler {
|
||||||
this.setCallListeners(newCall);
|
this.setCallListeners(newCall);
|
||||||
this.setCallState(newCall, newCall.state);
|
this.setCallState(newCall, newCall.state);
|
||||||
});
|
});
|
||||||
|
call.on(CallEvent.AssertedIdentityChanged, async () => {
|
||||||
|
if (!this.matchesCallForThisRoom(call)) return;
|
||||||
|
|
||||||
|
console.log(`Call ID ${call.callId} got new asserted identity:`, call.getRemoteAssertedIdentity());
|
||||||
|
|
||||||
|
const newAssertedIdentity = call.getRemoteAssertedIdentity().id;
|
||||||
|
let newNativeAssertedIdentity = newAssertedIdentity;
|
||||||
|
if (newAssertedIdentity) {
|
||||||
|
const response = await this.sipNativeLookup(newAssertedIdentity);
|
||||||
|
if (response.length) newNativeAssertedIdentity = response[0].userid;
|
||||||
|
}
|
||||||
|
console.log(`Asserted identity ${newAssertedIdentity} mapped to ${newNativeAssertedIdentity}`);
|
||||||
|
|
||||||
|
if (newNativeAssertedIdentity) {
|
||||||
|
this.assertedIdentityNativeUsers[call.callId] = newNativeAssertedIdentity;
|
||||||
|
|
||||||
|
// If we don't already have a room with this user, make one. This will be slightly odd
|
||||||
|
// if they called us because we'll be inviting them, but there's not much we can do about
|
||||||
|
// this if we want the actual, native room to exist (which we do). This is why it's
|
||||||
|
// important to only obey asserted identity in trusted environments, since anyone you're
|
||||||
|
// on a call with can cause you to send a room invite to someone.
|
||||||
|
await ensureDMExists(MatrixClientPeg.get(), newNativeAssertedIdentity);
|
||||||
|
|
||||||
|
const newMappedRoomId = this.roomIdForCall(call);
|
||||||
|
console.log(`Old room ID: ${mappedRoomId}, new room ID: ${newMappedRoomId}`);
|
||||||
|
if (newMappedRoomId !== mappedRoomId) {
|
||||||
|
this.removeCallForRoom(mappedRoomId);
|
||||||
|
mappedRoomId = newMappedRoomId;
|
||||||
|
this.calls.set(mappedRoomId, call);
|
||||||
|
dis.dispatch({
|
||||||
|
action: Action.CallChangeRoom,
|
||||||
|
call,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private async logCallStats(call: MatrixCall, mappedRoomId: string) {
|
private async logCallStats(call: MatrixCall, mappedRoomId: string) {
|
||||||
|
@ -551,7 +604,7 @@ export default class CallHandler {
|
||||||
}
|
}
|
||||||
|
|
||||||
private setCallState(call: MatrixCall, status: CallState) {
|
private setCallState(call: MatrixCall, status: CallState) {
|
||||||
const mappedRoomId = CallHandler.roomIdForCall(call);
|
const mappedRoomId = CallHandler.sharedInstance().roomIdForCall(call);
|
||||||
|
|
||||||
console.log(
|
console.log(
|
||||||
`Call state in ${mappedRoomId} changed to ${status}`,
|
`Call state in ${mappedRoomId} changed to ${status}`,
|
||||||
|
@ -639,7 +692,7 @@ export default class CallHandler {
|
||||||
|
|
||||||
const timeUntilTurnCresExpire = MatrixClientPeg.get().getTurnServersExpiry() - Date.now();
|
const timeUntilTurnCresExpire = MatrixClientPeg.get().getTurnServersExpiry() - Date.now();
|
||||||
console.log("Current turn creds expire in " + timeUntilTurnCresExpire + " ms");
|
console.log("Current turn creds expire in " + timeUntilTurnCresExpire + " ms");
|
||||||
const call = createNewMatrixCall(MatrixClientPeg.get(), mappedRoomId);
|
const call = MatrixClientPeg.get().createCall(mappedRoomId);
|
||||||
|
|
||||||
this.calls.set(roomId, call);
|
this.calls.set(roomId, call);
|
||||||
if (transferee) {
|
if (transferee) {
|
||||||
|
@ -772,7 +825,7 @@ export default class CallHandler {
|
||||||
|
|
||||||
const call = payload.call as MatrixCall;
|
const call = payload.call as MatrixCall;
|
||||||
|
|
||||||
const mappedRoomId = CallHandler.roomIdForCall(call);
|
const mappedRoomId = CallHandler.sharedInstance().roomIdForCall(call);
|
||||||
if (this.getCallForRoom(mappedRoomId)) {
|
if (this.getCallForRoom(mappedRoomId)) {
|
||||||
// ignore multiple incoming calls to the same room
|
// ignore multiple incoming calls to the same room
|
||||||
return;
|
return;
|
||||||
|
|
|
@ -57,7 +57,11 @@ export default class VoipUserMapper {
|
||||||
if (!virtualRoom) return null;
|
if (!virtualRoom) return null;
|
||||||
const virtualRoomEvent = virtualRoom.getAccountData(VIRTUAL_ROOM_EVENT_TYPE);
|
const virtualRoomEvent = virtualRoom.getAccountData(VIRTUAL_ROOM_EVENT_TYPE);
|
||||||
if (!virtualRoomEvent || !virtualRoomEvent.getContent()) return null;
|
if (!virtualRoomEvent || !virtualRoomEvent.getContent()) return null;
|
||||||
return virtualRoomEvent.getContent()['native_room'] || null;
|
const nativeRoomID = virtualRoomEvent.getContent()['native_room'];
|
||||||
|
const nativeRoom = MatrixClientPeg.get().getRoom(nativeRoomID);
|
||||||
|
if (!nativeRoom || nativeRoom.getMyMembership() !== 'join') return null;
|
||||||
|
|
||||||
|
return nativeRoomID;
|
||||||
}
|
}
|
||||||
|
|
||||||
public isVirtualRoom(room: Room): boolean {
|
public isVirtualRoom(room: Room): boolean {
|
||||||
|
|
|
@ -27,6 +27,7 @@ import SettingsStore from "../../../settings/SettingsStore";
|
||||||
import { CallEvent, CallState, MatrixCall } from 'matrix-js-sdk/src/webrtc/call';
|
import { CallEvent, CallState, MatrixCall } from 'matrix-js-sdk/src/webrtc/call';
|
||||||
import { MatrixClientPeg } from '../../../MatrixClientPeg';
|
import { MatrixClientPeg } from '../../../MatrixClientPeg';
|
||||||
import {replaceableComponent} from "../../../utils/replaceableComponent";
|
import {replaceableComponent} from "../../../utils/replaceableComponent";
|
||||||
|
import { Action } from '../../../dispatcher/actions';
|
||||||
|
|
||||||
const SHOW_CALL_IN_STATES = [
|
const SHOW_CALL_IN_STATES = [
|
||||||
CallState.Connected,
|
CallState.Connected,
|
||||||
|
@ -142,6 +143,7 @@ export default class CallPreview extends React.Component<IProps, IState> {
|
||||||
switch (payload.action) {
|
switch (payload.action) {
|
||||||
// listen for call state changes to prod the render method, which
|
// listen for call state changes to prod the render method, which
|
||||||
// may hide the global CallView if the call it is tracking is dead
|
// may hide the global CallView if the call it is tracking is dead
|
||||||
|
case Action.CallChangeRoom:
|
||||||
case 'call_state': {
|
case 'call_state': {
|
||||||
const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(
|
const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(
|
||||||
CallHandler.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId),
|
CallHandler.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId),
|
||||||
|
|
|
@ -208,7 +208,7 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
};
|
};
|
||||||
|
|
||||||
private onExpandClick = () => {
|
private onExpandClick = () => {
|
||||||
const userFacingRoomId = CallHandler.roomIdForCall(this.props.call);
|
const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call);
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'view_room',
|
action: 'view_room',
|
||||||
room_id: userFacingRoomId,
|
room_id: userFacingRoomId,
|
||||||
|
@ -337,7 +337,7 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
};
|
};
|
||||||
|
|
||||||
private onRoomAvatarClick = () => {
|
private onRoomAvatarClick = () => {
|
||||||
const userFacingRoomId = CallHandler.roomIdForCall(this.props.call);
|
const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call);
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'view_room',
|
action: 'view_room',
|
||||||
room_id: userFacingRoomId,
|
room_id: userFacingRoomId,
|
||||||
|
@ -345,7 +345,7 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
}
|
}
|
||||||
|
|
||||||
private onSecondaryRoomAvatarClick = () => {
|
private onSecondaryRoomAvatarClick = () => {
|
||||||
const userFacingRoomId = CallHandler.roomIdForCall(this.props.secondaryCall);
|
const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.secondaryCall);
|
||||||
|
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'view_room',
|
action: 'view_room',
|
||||||
|
@ -354,7 +354,7 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
}
|
}
|
||||||
|
|
||||||
private onCallResumeClick = () => {
|
private onCallResumeClick = () => {
|
||||||
const userFacingRoomId = CallHandler.roomIdForCall(this.props.call);
|
const userFacingRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call);
|
||||||
CallHandler.sharedInstance().setActiveCallRoomId(userFacingRoomId);
|
CallHandler.sharedInstance().setActiveCallRoomId(userFacingRoomId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -365,8 +365,8 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
|
|
||||||
public render() {
|
public render() {
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
const callRoomId = CallHandler.roomIdForCall(this.props.call);
|
const callRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.call);
|
||||||
const secondaryCallRoomId = CallHandler.roomIdForCall(this.props.secondaryCall);
|
const secondaryCallRoomId = CallHandler.sharedInstance().roomIdForCall(this.props.secondaryCall);
|
||||||
const callRoom = client.getRoom(callRoomId);
|
const callRoom = client.getRoom(callRoomId);
|
||||||
const secCallRoom = this.props.secondaryCall ? client.getRoom(secondaryCallRoomId) : null;
|
const secCallRoom = this.props.secondaryCall ? client.getRoom(secondaryCallRoomId) : null;
|
||||||
|
|
||||||
|
@ -482,11 +482,13 @@ export default class CallView extends React.Component<IProps, IState> {
|
||||||
const isOnHold = this.state.isLocalOnHold || this.state.isRemoteOnHold;
|
const isOnHold = this.state.isLocalOnHold || this.state.isRemoteOnHold;
|
||||||
let holdTransferContent;
|
let holdTransferContent;
|
||||||
if (transfereeCall) {
|
if (transfereeCall) {
|
||||||
const transferTargetRoom = MatrixClientPeg.get().getRoom(CallHandler.roomIdForCall(this.props.call));
|
const transferTargetRoom = MatrixClientPeg.get().getRoom(
|
||||||
|
CallHandler.sharedInstance().roomIdForCall(this.props.call),
|
||||||
|
);
|
||||||
const transferTargetName = transferTargetRoom ? transferTargetRoom.name : _t("unknown person");
|
const transferTargetName = transferTargetRoom ? transferTargetRoom.name : _t("unknown person");
|
||||||
|
|
||||||
const transfereeRoom = MatrixClientPeg.get().getRoom(
|
const transfereeRoom = MatrixClientPeg.get().getRoom(
|
||||||
CallHandler.roomIdForCall(transfereeCall),
|
CallHandler.sharedInstance().roomIdForCall(transfereeCall),
|
||||||
);
|
);
|
||||||
const transfereeName = transfereeRoom ? transfereeRoom.name : _t("unknown person");
|
const transfereeName = transfereeRoom ? transfereeRoom.name : _t("unknown person");
|
||||||
|
|
||||||
|
|
|
@ -22,6 +22,7 @@ import dis from '../../../dispatcher/dispatcher';
|
||||||
import {Resizable} from "re-resizable";
|
import {Resizable} from "re-resizable";
|
||||||
import ResizeNotifier from "../../../utils/ResizeNotifier";
|
import ResizeNotifier from "../../../utils/ResizeNotifier";
|
||||||
import {replaceableComponent} from "../../../utils/replaceableComponent";
|
import {replaceableComponent} from "../../../utils/replaceableComponent";
|
||||||
|
import { Action } from '../../../dispatcher/actions';
|
||||||
|
|
||||||
interface IProps {
|
interface IProps {
|
||||||
// What room we should display the call for
|
// What room we should display the call for
|
||||||
|
@ -62,6 +63,7 @@ export default class CallViewForRoom extends React.Component<IProps, IState> {
|
||||||
|
|
||||||
private onAction = (payload) => {
|
private onAction = (payload) => {
|
||||||
switch (payload.action) {
|
switch (payload.action) {
|
||||||
|
case Action.CallChangeRoom:
|
||||||
case 'call_state': {
|
case 'call_state': {
|
||||||
const newCall = this.getCall();
|
const newCall = this.getCall();
|
||||||
if (newCall !== this.state.call) {
|
if (newCall !== this.state.call) {
|
||||||
|
|
|
@ -72,7 +72,7 @@ export default class IncomingCallBox extends React.Component<IProps, IState> {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'answer',
|
action: 'answer',
|
||||||
room_id: CallHandler.roomIdForCall(this.state.incomingCall),
|
room_id: CallHandler.sharedInstance().roomIdForCall(this.state.incomingCall),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -80,7 +80,7 @@ export default class IncomingCallBox extends React.Component<IProps, IState> {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
dis.dispatch({
|
dis.dispatch({
|
||||||
action: 'reject',
|
action: 'reject',
|
||||||
room_id: CallHandler.roomIdForCall(this.state.incomingCall),
|
room_id: CallHandler.sharedInstance().roomIdForCall(this.state.incomingCall),
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -91,7 +91,7 @@ export default class IncomingCallBox extends React.Component<IProps, IState> {
|
||||||
|
|
||||||
let room = null;
|
let room = null;
|
||||||
if (this.state.incomingCall) {
|
if (this.state.incomingCall) {
|
||||||
room = MatrixClientPeg.get().getRoom(CallHandler.roomIdForCall(this.state.incomingCall));
|
room = MatrixClientPeg.get().getRoom(CallHandler.sharedInstance().roomIdForCall(this.state.incomingCall));
|
||||||
}
|
}
|
||||||
|
|
||||||
const caller = room ? room.name : _t("Unknown caller");
|
const caller = room ? room.name : _t("Unknown caller");
|
||||||
|
|
|
@ -114,6 +114,9 @@ export enum Action {
|
||||||
*/
|
*/
|
||||||
VirtualRoomSupportUpdated = "virtual_room_support_updated",
|
VirtualRoomSupportUpdated = "virtual_room_support_updated",
|
||||||
|
|
||||||
|
// Probably would be better to have a VoIP states in a store and have the store emit changes
|
||||||
|
CallChangeRoom = "call_change_room",
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fired when an upload has started. Should be used with UploadStartedPayload.
|
* Fired when an upload has started. Should be used with UploadStartedPayload.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -55,6 +55,15 @@ export default class DMRoomMap {
|
||||||
return DMRoomMap.sharedInstance;
|
return DMRoomMap.sharedInstance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Set the shared instance to the instance supplied
|
||||||
|
* Used by tests
|
||||||
|
* @param inst the new shared instance
|
||||||
|
*/
|
||||||
|
public static setShared(inst: DMRoomMap) {
|
||||||
|
DMRoomMap.sharedInstance = inst;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a shared instance of the class
|
* Returns a shared instance of the class
|
||||||
* that uses the singleton matrix client
|
* that uses the singleton matrix client
|
||||||
|
|
216
test/CallHandler-test.ts
Normal file
216
test/CallHandler-test.ts
Normal file
|
@ -0,0 +1,216 @@
|
||||||
|
/*
|
||||||
|
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 CallHandler, { PlaceCallType } from '../src/CallHandler';
|
||||||
|
import { stubClient, mkStubRoom } from './test-utils';
|
||||||
|
import { MatrixClientPeg } from '../src/MatrixClientPeg';
|
||||||
|
import dis from '../src/dispatcher/dispatcher';
|
||||||
|
import { CallEvent, CallState } from 'matrix-js-sdk/src/webrtc/call';
|
||||||
|
import DMRoomMap from '../src/utils/DMRoomMap';
|
||||||
|
import EventEmitter from 'events';
|
||||||
|
import { Action } from '../src/dispatcher/actions';
|
||||||
|
import SdkConfig from '../src/SdkConfig';
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
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();
|
||||||
|
|
||||||
|
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;
|
||||||
|
MatrixClientPeg.unset();
|
||||||
|
|
||||||
|
document.body.removeChild(audioElement);
|
||||||
|
SdkConfig.unset();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should move calls between rooms when remote asserted identity changes', async () => {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
dis.dispatch({
|
||||||
|
action: 'place_call',
|
||||||
|
type: PlaceCallType.Voice,
|
||||||
|
room_id: REAL_ROOM_ID,
|
||||||
|
}, true);
|
||||||
|
|
||||||
|
let dispatchHandle;
|
||||||
|
// wait for the call to be set up
|
||||||
|
await new Promise<void>(resolve => {
|
||||||
|
dispatchHandle = dis.register(payload => {
|
||||||
|
if (payload.action === 'call_state') {
|
||||||
|
resolve();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
dis.unregister(dispatchHandle);
|
||||||
|
|
||||||
|
// 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 => {
|
||||||
|
dispatchHandle = dis.register(payload => {
|
||||||
|
if (payload.action === Action.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;
|
||||||
|
dis.unregister(dispatchHandle);
|
||||||
|
|
||||||
|
// 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);
|
||||||
|
});
|
||||||
|
});
|
|
@ -64,6 +64,11 @@ export function createTestClient() {
|
||||||
getRoomIdForAlias: jest.fn().mockResolvedValue(undefined),
|
getRoomIdForAlias: jest.fn().mockResolvedValue(undefined),
|
||||||
getRoomDirectoryVisibility: jest.fn().mockResolvedValue(undefined),
|
getRoomDirectoryVisibility: jest.fn().mockResolvedValue(undefined),
|
||||||
getProfileInfo: jest.fn().mockResolvedValue({}),
|
getProfileInfo: jest.fn().mockResolvedValue({}),
|
||||||
|
getThirdpartyProtocols: jest.fn().mockResolvedValue({}),
|
||||||
|
getClientWellKnown: jest.fn().mockReturnValue(null),
|
||||||
|
supportsVoip: jest.fn().mockReturnValue(true),
|
||||||
|
getTurnServersExpiry: jest.fn().mockReturnValue(2^32),
|
||||||
|
getThirdpartyUser: jest.fn().mockResolvedValue([]),
|
||||||
getAccountData: (type) => {
|
getAccountData: (type) => {
|
||||||
return mkEvent({
|
return mkEvent({
|
||||||
type,
|
type,
|
||||||
|
|
Loading…
Reference in a new issue