152 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			152 lines
		
	
	
		
			6.7 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 { Room } from 'matrix-js-sdk/src/models/room';
 | |
| import { logger } from "matrix-js-sdk/src/logger";
 | |
| import { EventType } from 'matrix-js-sdk/src/@types/event';
 | |
| 
 | |
| import { ensureVirtualRoomExists } from './createRoom';
 | |
| import { MatrixClientPeg } from "./MatrixClientPeg";
 | |
| import DMRoomMap from "./utils/DMRoomMap";
 | |
| import LegacyCallHandler from './LegacyCallHandler';
 | |
| import { VIRTUAL_ROOM_EVENT_TYPE } from "./call-types";
 | |
| import { findDMForUser } from './utils/dm/findDMForUser';
 | |
| 
 | |
| // Functions for mapping virtual users & rooms. Currently the only lookup
 | |
| // is sip virtual: there could be others in the future.
 | |
| 
 | |
| export default class VoipUserMapper {
 | |
|     // We store mappings of virtual -> native room IDs here until the local echo for the
 | |
|     // account data arrives.
 | |
|     private virtualToNativeRoomIdCache = new Map<string, string>();
 | |
| 
 | |
|     public static sharedInstance(): VoipUserMapper {
 | |
|         if (window.mxVoipUserMapper === undefined) window.mxVoipUserMapper = new VoipUserMapper();
 | |
|         return window.mxVoipUserMapper;
 | |
|     }
 | |
| 
 | |
|     private async userToVirtualUser(userId: string): Promise<string> {
 | |
|         const results = await LegacyCallHandler.instance.sipVirtualLookup(userId);
 | |
|         if (results.length === 0 || !results[0].fields.lookup_success) return null;
 | |
|         return results[0].userid;
 | |
|     }
 | |
| 
 | |
|     private async getVirtualUserForRoom(roomId: string): Promise<string | null> {
 | |
|         const userId = DMRoomMap.shared().getUserIdForRoomId(roomId);
 | |
|         if (!userId) return null;
 | |
| 
 | |
|         const virtualUser = await this.userToVirtualUser(userId);
 | |
|         if (!virtualUser) return null;
 | |
| 
 | |
|         return virtualUser;
 | |
|     }
 | |
| 
 | |
|     public async getOrCreateVirtualRoomForRoom(roomId: string): Promise<string | null> {
 | |
|         const virtualUser = await this.getVirtualUserForRoom(roomId);
 | |
|         if (!virtualUser) return null;
 | |
| 
 | |
|         const virtualRoomId = await ensureVirtualRoomExists(MatrixClientPeg.get(), virtualUser, roomId);
 | |
|         MatrixClientPeg.get().setRoomAccountData(virtualRoomId, VIRTUAL_ROOM_EVENT_TYPE, {
 | |
|             native_room: roomId,
 | |
|         });
 | |
| 
 | |
|         this.virtualToNativeRoomIdCache.set(virtualRoomId, roomId);
 | |
| 
 | |
|         return virtualRoomId;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Gets the ID of the virtual room for a room, or null if the room has no
 | |
|      * virtual room
 | |
|      */
 | |
|     public async getVirtualRoomForRoom(roomId: string): Promise<Room | null> {
 | |
|         const virtualUser = await this.getVirtualUserForRoom(roomId);
 | |
|         if (!virtualUser) return null;
 | |
| 
 | |
|         return findDMForUser(MatrixClientPeg.get(), virtualUser);
 | |
|     }
 | |
| 
 | |
|     public nativeRoomForVirtualRoom(roomId: string): string {
 | |
|         const cachedNativeRoomId = this.virtualToNativeRoomIdCache.get(roomId);
 | |
|         if (cachedNativeRoomId) {
 | |
|             logger.log(
 | |
|                 "Returning native room ID " + cachedNativeRoomId + " for virtual room ID " + roomId + " from cache",
 | |
|             );
 | |
|             return cachedNativeRoomId;
 | |
|         }
 | |
| 
 | |
|         const virtualRoom = MatrixClientPeg.get().getRoom(roomId);
 | |
|         if (!virtualRoom) return null;
 | |
|         const virtualRoomEvent = virtualRoom.getAccountData(VIRTUAL_ROOM_EVENT_TYPE);
 | |
|         if (!virtualRoomEvent || !virtualRoomEvent.getContent()) return 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 {
 | |
|         if (this.nativeRoomForVirtualRoom(room.roomId)) return true;
 | |
| 
 | |
|         if (this.virtualToNativeRoomIdCache.has(room.roomId)) return true;
 | |
| 
 | |
|         // also look in the create event for the claimed native room ID, which is the only
 | |
|         // way we can recognise a virtual room we've created when it first arrives down
 | |
|         // our stream. We don't trust this in general though, as it could be faked by an
 | |
|         // inviter: our main source of truth is the DM state.
 | |
|         const roomCreateEvent = room.currentState.getStateEvents(EventType.RoomCreate, "");
 | |
|         if (!roomCreateEvent || !roomCreateEvent.getContent()) return false;
 | |
|         // we only look at this for rooms we created (so inviters can't just cause rooms
 | |
|         // to be invisible)
 | |
|         if (roomCreateEvent.getSender() !== MatrixClientPeg.get().getUserId()) return false;
 | |
|         const claimedNativeRoomId = roomCreateEvent.getContent()[VIRTUAL_ROOM_EVENT_TYPE];
 | |
|         return Boolean(claimedNativeRoomId);
 | |
|     }
 | |
| 
 | |
|     public async onNewInvitedRoom(invitedRoom: Room): Promise<void> {
 | |
|         if (!LegacyCallHandler.instance.getSupportsVirtualRooms()) return;
 | |
| 
 | |
|         const inviterId = invitedRoom.getDMInviter();
 | |
|         logger.log(`Checking virtual-ness of room ID ${invitedRoom.roomId}, invited by ${inviterId}`);
 | |
|         const result = await LegacyCallHandler.instance.sipNativeLookup(inviterId);
 | |
|         if (result.length === 0) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if (result[0].fields.is_virtual) {
 | |
|             const nativeUser = result[0].userid;
 | |
|             const nativeRoom = findDMForUser(MatrixClientPeg.get(), nativeUser);
 | |
|             if (nativeRoom) {
 | |
|                 // It's a virtual room with a matching native room, so set the room account data. This
 | |
|                 // will make sure we know where how to map calls and also allow us know not to display
 | |
|                 // it in the future.
 | |
|                 MatrixClientPeg.get().setRoomAccountData(invitedRoom.roomId, VIRTUAL_ROOM_EVENT_TYPE, {
 | |
|                     native_room: nativeRoom.roomId,
 | |
|                 });
 | |
|                 // also auto-join the virtual room if we have a matching native room
 | |
|                 // (possibly we should only join if we've also joined the native room, then we'd also have
 | |
|                 // to make sure we joined virtual rooms on joining a native one)
 | |
|                 MatrixClientPeg.get().joinRoom(invitedRoom.roomId);
 | |
|             }
 | |
| 
 | |
|             // also put this room in the virtual room ID cache so isVirtualRoom return the right answer
 | |
|             // in however long it takes for the echo of setAccountData to come down the sync
 | |
|             this.virtualToNativeRoomIdCache.set(invitedRoom.roomId, nativeRoom.roomId);
 | |
|         }
 | |
|     }
 | |
| }
 |