2022-02-17 21:28:44 +05:30
|
|
|
/*
|
|
|
|
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";
|
2022-03-10 19:23:31 +05:30
|
|
|
import {Member} from "./Member";
|
2022-03-09 15:59:39 +05:30
|
|
|
import {LocalMedia} from "../LocalMedia";
|
2022-03-10 19:23:31 +05:30
|
|
|
import {RoomMember} from "../../room/members/RoomMember";
|
2022-03-11 19:10:37 +05:30
|
|
|
import {makeId} from "../../common";
|
|
|
|
|
2022-03-10 19:23:31 +05:30
|
|
|
import type {Options as MemberOptions} from "./Member";
|
|
|
|
import type {BaseObservableMap} from "../../../observable/map/BaseObservableMap";
|
2022-03-09 15:59:39 +05:30
|
|
|
import type {Track} from "../../../platform/types/MediaDevices";
|
2022-03-09 23:23:51 +05:30
|
|
|
import type {SignallingMessage, MGroupCallBase} from "../callEventTypes";
|
|
|
|
import type {Room} from "../../room/Room";
|
|
|
|
import type {StateEvent} from "../../storage/types";
|
|
|
|
import type {Platform} from "../../../platform/web/Platform";
|
2022-03-10 19:23:31 +05:30
|
|
|
import type {EncryptedMessage} from "../../e2ee/olm/Encryption";
|
|
|
|
import type {ILogItem} from "../../../logging/types";
|
2022-03-10 22:09:29 +05:30
|
|
|
import type {Storage} from "../../storage/idb/Storage";
|
|
|
|
|
|
|
|
export enum GroupCallState {
|
2022-03-11 19:10:37 +05:30
|
|
|
Fledgling = "fledgling",
|
|
|
|
Creating = "creating",
|
|
|
|
Created = "created",
|
|
|
|
Joining = "joining",
|
|
|
|
Joined = "joined",
|
2022-03-10 22:09:29 +05:30
|
|
|
}
|
2022-03-10 19:23:31 +05:30
|
|
|
|
|
|
|
export type Options = Omit<MemberOptions, "emitUpdate" | "confId" | "encryptDeviceMessage"> & {
|
|
|
|
emitUpdate: (call: GroupCall, params?: any) => void;
|
|
|
|
encryptDeviceMessage: (roomId: string, message: SignallingMessage<MGroupCallBase>, log: ILogItem) => Promise<EncryptedMessage>,
|
2022-03-10 22:09:29 +05:30
|
|
|
storage: Storage,
|
|
|
|
ownDeviceId: string
|
2022-03-10 19:23:31 +05:30
|
|
|
};
|
2022-02-17 21:28:44 +05:30
|
|
|
|
2022-03-09 15:59:39 +05:30
|
|
|
export class GroupCall {
|
2022-03-11 19:10:37 +05:30
|
|
|
public readonly id: string;
|
2022-03-10 19:23:31 +05:30
|
|
|
private readonly _members: ObservableMap<string, Member> = new ObservableMap();
|
2022-03-11 19:10:37 +05:30
|
|
|
private _localMedia?: LocalMedia;
|
2022-03-10 19:23:31 +05:30
|
|
|
private _memberOptions: MemberOptions;
|
2022-03-11 19:10:37 +05:30
|
|
|
private _state: GroupCallState;
|
|
|
|
|
2022-03-09 23:23:51 +05:30
|
|
|
constructor(
|
2022-03-11 19:10:37 +05:30
|
|
|
id: string | undefined,
|
|
|
|
private callContent: Record<string, any> | undefined,
|
|
|
|
private readonly roomId: string,
|
2022-03-10 19:23:31 +05:30
|
|
|
private readonly options: Options
|
2022-03-09 23:23:51 +05:30
|
|
|
) {
|
2022-03-11 19:10:37 +05:30
|
|
|
this.id = id ?? makeId("conf-");
|
|
|
|
this._state = id ? GroupCallState.Created : GroupCallState.Fledgling;
|
2022-03-10 19:23:31 +05:30
|
|
|
this._memberOptions = Object.assign({
|
|
|
|
confId: this.id,
|
|
|
|
emitUpdate: member => this._members.update(member.member.userId, member),
|
|
|
|
encryptDeviceMessage: (message: SignallingMessage<MGroupCallBase>, log) => {
|
2022-03-11 19:10:37 +05:30
|
|
|
return this.options.encryptDeviceMessage(this.roomId, message, log);
|
2022-03-10 19:23:31 +05:30
|
|
|
}
|
|
|
|
}, options);
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
|
|
|
|
2022-03-11 19:10:37 +05:30
|
|
|
get localMedia(): LocalMedia | undefined { return this._localMedia; }
|
2022-03-10 19:23:31 +05:30
|
|
|
get members(): BaseObservableMap<string, Member> { return this._members; }
|
|
|
|
|
|
|
|
get isTerminated(): boolean {
|
2022-03-11 19:10:37 +05:30
|
|
|
return this.callContent?.["m.terminated"] === true;
|
2022-03-10 19:23:31 +05:30
|
|
|
}
|
|
|
|
|
2022-03-11 19:10:37 +05:30
|
|
|
async join(localMedia: LocalMedia) {
|
|
|
|
if (this._state !== GroupCallState.Created) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this._state = GroupCallState.Joining;
|
|
|
|
this._localMedia = localMedia;
|
|
|
|
const memberContent = await this._joinCallMemberContent();
|
2022-02-17 21:28:44 +05:30
|
|
|
// send m.call.member state event
|
2022-03-11 19:10:37 +05:30
|
|
|
const request = this.options.hsApi.sendState(this.roomId, "m.call.member", this.options.ownUserId, memberContent);
|
2022-03-10 19:23:31 +05:30
|
|
|
await request.response();
|
|
|
|
// send invite to all members that are < my userId
|
|
|
|
for (const [,member] of this._members) {
|
2022-03-11 19:10:37 +05:30
|
|
|
member.connect(this._localMedia);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async leave() {
|
|
|
|
const memberContent = await this._leaveCallMemberContent();
|
|
|
|
// send m.call.member state event
|
|
|
|
if (memberContent) {
|
|
|
|
const request = this.options.hsApi.sendState(this.roomId, "m.call.member", this.options.ownUserId, memberContent);
|
|
|
|
await request.response();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @internal */
|
|
|
|
async create(localMedia: LocalMedia, name: string) {
|
|
|
|
if (this._state !== GroupCallState.Fledgling) {
|
|
|
|
return;
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
2022-03-11 19:10:37 +05:30
|
|
|
this._state = GroupCallState.Creating;
|
|
|
|
this.callContent = {
|
|
|
|
"m.type": localMedia.cameraTrack ? "m.video" : "m.voice",
|
|
|
|
"m.name": name,
|
|
|
|
"m.intent": "m.ring"
|
|
|
|
};
|
|
|
|
const request = this.options.hsApi.sendState(this.roomId, "m.call", this.id, this.callContent);
|
|
|
|
await request.response();
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
|
|
|
|
2022-03-10 19:23:31 +05:30
|
|
|
/** @internal */
|
2022-03-11 19:10:37 +05:30
|
|
|
updateCallEvent(callContent: Record<string, any>) {
|
|
|
|
this.callContent = callContent;
|
|
|
|
if (this._state === GroupCallState.Creating) {
|
|
|
|
this._state = GroupCallState.Created;
|
|
|
|
}
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
|
|
|
|
2022-03-10 19:23:31 +05:30
|
|
|
/** @internal */
|
|
|
|
addMember(userId, memberCallInfo) {
|
2022-03-11 19:10:37 +05:30
|
|
|
if (userId === this.options.ownUserId) {
|
|
|
|
if (this._state === GroupCallState.Joining) {
|
|
|
|
this._state = GroupCallState.Joined;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-03-10 19:23:31 +05:30
|
|
|
let member = this._members.get(userId);
|
|
|
|
if (member) {
|
|
|
|
member.updateCallInfo(memberCallInfo);
|
2022-02-17 21:28:44 +05:30
|
|
|
} else {
|
2022-03-11 19:10:37 +05:30
|
|
|
member = new Member(RoomMember.fromUserId(this.roomId, userId, "join"), memberCallInfo, this._memberOptions);
|
2022-03-10 19:23:31 +05:30
|
|
|
this._members.add(userId, member);
|
2022-03-11 19:10:37 +05:30
|
|
|
if (this._state === GroupCallState.Joining || this._state === GroupCallState.Joined) {
|
|
|
|
member.connect(this._localMedia!);
|
|
|
|
}
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-10 19:23:31 +05:30
|
|
|
/** @internal */
|
|
|
|
removeMember(userId) {
|
2022-03-11 19:10:37 +05:30
|
|
|
if (userId === this.options.ownUserId) {
|
|
|
|
if (this._state === GroupCallState.Joined) {
|
|
|
|
this._state = GroupCallState.Created;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2022-03-10 19:23:31 +05:30
|
|
|
this._members.remove(userId);
|
2022-03-09 23:23:51 +05:30
|
|
|
}
|
|
|
|
|
2022-03-10 19:23:31 +05:30
|
|
|
/** @internal */
|
|
|
|
handleDeviceMessage(message: SignallingMessage<MGroupCallBase>, userId: string, deviceId: string, log: ILogItem) {
|
|
|
|
// TODO: return if we are not membering to the call
|
|
|
|
let member = this._members.get(userId);
|
|
|
|
if (member) {
|
|
|
|
member.handleDeviceMessage(message, deviceId, log);
|
|
|
|
} else {
|
|
|
|
// we haven't received the m.call.member yet for this caller. buffer the device messages or create the member/call anyway?
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|
|
|
|
}
|
2022-03-10 22:09:29 +05:30
|
|
|
|
2022-03-11 19:10:37 +05:30
|
|
|
private async _joinCallMemberContent() {
|
2022-03-10 22:09:29 +05:30
|
|
|
const {storage} = this.options;
|
|
|
|
const txn = await storage.readTxn([storage.storeNames.roomState]);
|
2022-03-11 19:10:37 +05:30
|
|
|
const stateEvent = await txn.roomState.get(this.roomId, "m.call.member", this.options.ownUserId);
|
2022-03-10 22:09:29 +05:30
|
|
|
const stateContent = stateEvent?.event?.content ?? {
|
|
|
|
["m.calls"]: []
|
|
|
|
};
|
|
|
|
const callsInfo = stateContent["m.calls"];
|
|
|
|
let callInfo = callsInfo.find(c => c["m.call_id"] === this.id);
|
|
|
|
if (!callInfo) {
|
|
|
|
callInfo = {
|
|
|
|
["m.call_id"]: this.id,
|
|
|
|
["m.devices"]: []
|
|
|
|
};
|
|
|
|
callsInfo.push(callInfo);
|
|
|
|
}
|
|
|
|
const devicesInfo = callInfo["m.devices"];
|
|
|
|
let deviceInfo = devicesInfo.find(d => d["device_id"] === this.options.ownDeviceId);
|
|
|
|
if (!deviceInfo) {
|
|
|
|
deviceInfo = {
|
|
|
|
["device_id"]: this.options.ownDeviceId
|
|
|
|
};
|
|
|
|
devicesInfo.push(deviceInfo);
|
|
|
|
}
|
|
|
|
return stateContent;
|
|
|
|
}
|
2022-03-11 19:10:37 +05:30
|
|
|
|
|
|
|
private async _leaveCallMemberContent(): Promise<Record<string, any> | undefined> {
|
|
|
|
const {storage} = this.options;
|
|
|
|
const txn = await storage.readTxn([storage.storeNames.roomState]);
|
|
|
|
const stateEvent = await txn.roomState.get(this.roomId, "m.call.member", this.options.ownUserId);
|
|
|
|
const callsInfo = stateEvent?.event?.content?.["m.calls"];
|
|
|
|
callsInfo?.filter(c => c["m.call_id"] === this.id);
|
|
|
|
return stateEvent?.event.content;
|
|
|
|
}
|
2022-02-17 21:28:44 +05:30
|
|
|
}
|