2022-02-14 17:14:21 +01: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 {ObservableMap} from "../../observable/map/ObservableMap";
|
|
|
|
|
|
|
|
import type {Room} from "../room/Room";
|
|
|
|
import type {StateEvent} from "../storage/types";
|
|
|
|
import type {ILogItem} from "../../logging/types";
|
|
|
|
|
|
|
|
const GROUP_CALL_TYPE = "m.call";
|
|
|
|
const GROUP_CALL_MEMBER_TYPE = "m.call.member";
|
|
|
|
|
|
|
|
enum CallSetupMessageType {
|
|
|
|
Invite = "m.call.invite",
|
|
|
|
Answer = "m.call.answer",
|
|
|
|
Candidates = "m.call.candidates",
|
|
|
|
Hangup = "m.call.hangup",
|
|
|
|
}
|
|
|
|
|
|
|
|
const CALL_ID = "m.call_id";
|
|
|
|
const CALL_TERMINATED = "m.terminated";
|
|
|
|
|
|
|
|
export class CallHandler {
|
2022-02-15 17:05:20 +01:00
|
|
|
// group calls by call id
|
2022-02-14 17:14:21 +01:00
|
|
|
public readonly groupCalls: ObservableMap<string, GroupCall> = new ObservableMap<string, GroupCall>();
|
|
|
|
|
|
|
|
constructor() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
handleRoomState(room: Room, events: StateEvent[], log: ILogItem) {
|
|
|
|
// first update call events
|
|
|
|
for (const event of events) {
|
|
|
|
if (event.type === GROUP_CALL_TYPE) {
|
|
|
|
const callId = event.state_key;
|
|
|
|
let call = this.groupCalls.get(callId);
|
|
|
|
if (call) {
|
|
|
|
call.updateCallEvent(event);
|
|
|
|
if (call.isTerminated) {
|
|
|
|
this.groupCalls.remove(call.id);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
call = new GroupCall(event, room);
|
|
|
|
this.groupCalls.set(call.id, call);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// then update participants
|
|
|
|
for (const event of events) {
|
|
|
|
if (event.type === GROUP_CALL_MEMBER_TYPE) {
|
|
|
|
const participant = event.state_key;
|
|
|
|
const sources = event.content["m.sources"];
|
|
|
|
for (const source of sources) {
|
|
|
|
const call = this.groupCalls.get(source[CALL_ID]);
|
|
|
|
if (call && !call.isTerminated) {
|
|
|
|
call.addParticipant(participant, source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
handlesDeviceMessageEventType(eventType: string | undefined): boolean {
|
|
|
|
return eventType === CallSetupMessageType.Invite ||
|
|
|
|
eventType === CallSetupMessageType.Candidates ||
|
|
|
|
eventType === CallSetupMessageType.Answer ||
|
|
|
|
eventType === CallSetupMessageType.Hangup;
|
|
|
|
}
|
|
|
|
|
|
|
|
handleDeviceMessage(senderUserId: string, senderDeviceId: string, eventType: string, content: Record<string, any>, log: ILogItem) {
|
|
|
|
const callId = content[CALL_ID];
|
|
|
|
const call = this.groupCalls.get(callId);
|
|
|
|
call?.handleDeviceMessage(senderUserId, senderDeviceId, eventType, content, log);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function peerCallKey(senderUserId: string, senderDeviceId: string) {
|
|
|
|
return JSON.stringify(senderUserId) + JSON.stringify(senderDeviceId);
|
|
|
|
}
|
|
|
|
|
|
|
|
class GroupCall {
|
|
|
|
private peerCalls: Map<string, PeerCall>
|
|
|
|
|
|
|
|
constructor(private callEvent: StateEvent, private readonly room: Room) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
updateCallEvent(callEvent: StateEvent) {
|
|
|
|
this.callEvent = callEvent;
|
|
|
|
}
|
|
|
|
|
|
|
|
addParticipant(userId, source) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
handleDeviceMessage(senderUserId: string, senderDeviceId: string, eventType: string, content: Record<string, any>, log: ILogItem) {
|
|
|
|
const peerCall = this.peerCalls.get(peerCallKey(senderUserId, senderDeviceId));
|
|
|
|
peerCall?.handleIncomingSignallingMessage()
|
|
|
|
}
|
|
|
|
|
|
|
|
get id(): string {
|
|
|
|
return this.callEvent.state_key;
|
|
|
|
}
|
|
|
|
|
|
|
|
get isTerminated(): boolean {
|
|
|
|
return !!this.callEvent.content[CALL_TERMINATED];
|
|
|
|
}
|
|
|
|
|
|
|
|
private createPeerCall(userId: string, deviceId: string): PeerCall {
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Does WebRTC signalling for a single PeerConnection, and deals with WebRTC wrappers from platform
|
|
|
|
* */
|
|
|
|
|
|
|
|
|
|
|
|
// when sending, we need to encrypt message with olm. I think the flow of room => roomEncryption => olmEncryption as we already
|
|
|
|
// do for sharing keys will be best as that already deals with room tracking.
|
|
|
|
type SendSignallingMessageCallback = (type: CallSetupMessageType, content: Record<string, any>) => Promise<void>;
|
|
|
|
|
|
|
|
class PeerCall {
|
|
|
|
constructor(private readonly sendSignallingMessage: SendSignallingMessageCallback) {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
handleIncomingSignallingMessage(type: CallSetupMessageType, content: Record<string, any>) {
|
|
|
|
switch (type) {
|
|
|
|
case CallSetupMessageType.Invite:
|
|
|
|
case CallSetupMessageType.Answer:
|
|
|
|
case CallSetupMessageType.Candidates:
|
|
|
|
case CallSetupMessageType.Hangup:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MediaSource {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class PeerConnection {
|
|
|
|
|
|
|
|
}
|