2022-02-02 14:49:49 +05:30
|
|
|
/*
|
|
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
|
|
|
|
|
|
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 {calculateRoomName} from "./members/Heroes";
|
|
|
|
import {createRoomEncryptionEvent} from "../e2ee/common";
|
2022-02-03 22:27:35 +05:30
|
|
|
import {MediaRepository} from "../net/MediaRepository";
|
2022-02-02 14:49:49 +05:30
|
|
|
import {EventEmitter} from "../../utils/EventEmitter";
|
|
|
|
|
|
|
|
import type {StateEvent} from "../storage/types";
|
|
|
|
import type {HomeServerApi} from "../net/HomeServerApi";
|
|
|
|
import type {ILogItem} from "../../logging/types";
|
|
|
|
|
|
|
|
type CreateRoomPayload = {
|
|
|
|
is_direct?: boolean;
|
|
|
|
preset?: string;
|
|
|
|
name?: string;
|
|
|
|
topic?: string;
|
|
|
|
invite?: string[];
|
|
|
|
initial_state?: StateEvent[]
|
|
|
|
}
|
|
|
|
|
|
|
|
export enum RoomType {
|
|
|
|
DirectMessage,
|
|
|
|
Private,
|
|
|
|
Public
|
|
|
|
}
|
|
|
|
|
|
|
|
function defaultE2EEStatusForType(type: RoomType): boolean {
|
|
|
|
switch (type) {
|
|
|
|
case RoomType.DirectMessage:
|
|
|
|
case RoomType.Private:
|
|
|
|
return true;
|
|
|
|
case RoomType.Public:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function presetForType(type: RoomType): string {
|
|
|
|
switch (type) {
|
|
|
|
case RoomType.DirectMessage:
|
|
|
|
return "trusted_private_chat";
|
|
|
|
case RoomType.Private:
|
|
|
|
return "private_chat";
|
|
|
|
case RoomType.Public:
|
|
|
|
return "public_chat";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-03 22:27:35 +05:30
|
|
|
export class RoomBeingCreated extends EventEmitter<{change: never}> {
|
2022-02-02 14:49:49 +05:30
|
|
|
private _roomId?: string;
|
|
|
|
private profiles: Profile[] = [];
|
|
|
|
|
|
|
|
public readonly isEncrypted: boolean;
|
2022-02-03 22:27:35 +05:30
|
|
|
private _name: string;
|
2022-02-07 21:00:44 +05:30
|
|
|
private _error?: Error;
|
2022-02-02 14:49:49 +05:30
|
|
|
|
|
|
|
constructor(
|
2022-02-03 22:27:35 +05:30
|
|
|
public readonly localId: string,
|
2022-02-02 14:49:49 +05:30
|
|
|
private readonly type: RoomType,
|
|
|
|
isEncrypted: boolean | undefined,
|
|
|
|
private readonly explicitName: string | undefined,
|
|
|
|
private readonly topic: string | undefined,
|
|
|
|
private readonly inviteUserIds: string[] | undefined,
|
2022-02-03 22:27:35 +05:30
|
|
|
private readonly updateCallback,
|
|
|
|
public readonly mediaRepository: MediaRepository,
|
2022-02-02 14:49:49 +05:30
|
|
|
log: ILogItem
|
|
|
|
) {
|
|
|
|
super();
|
|
|
|
this.isEncrypted = isEncrypted === undefined ? defaultE2EEStatusForType(this.type) : isEncrypted;
|
|
|
|
if (explicitName) {
|
2022-02-03 22:27:35 +05:30
|
|
|
this._name = explicitName;
|
2022-02-02 14:49:49 +05:30
|
|
|
} else {
|
|
|
|
const summaryData = {
|
|
|
|
joinCount: 1, // ourselves
|
|
|
|
inviteCount: (this.inviteUserIds?.length || 0)
|
|
|
|
};
|
2022-02-03 22:27:35 +05:30
|
|
|
const userIdProfiles = (inviteUserIds || []).map(userId => new UserIdProfile(userId));
|
|
|
|
this._name = calculateRoomName(userIdProfiles, summaryData, log);
|
2022-02-02 14:49:49 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public async start(hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
|
|
|
await Promise.all([
|
|
|
|
this.loadProfiles(hsApi, log),
|
|
|
|
this.create(hsApi, log),
|
|
|
|
]);
|
|
|
|
}
|
|
|
|
|
|
|
|
private async create(hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
|
|
|
const options: CreateRoomPayload = {
|
|
|
|
is_direct: this.type === RoomType.DirectMessage,
|
|
|
|
preset: presetForType(this.type)
|
|
|
|
};
|
|
|
|
if (this.explicitName) {
|
|
|
|
options.name = this.explicitName;
|
|
|
|
}
|
|
|
|
if (this.topic) {
|
|
|
|
options.topic = this.topic;
|
|
|
|
}
|
|
|
|
if (this.inviteUserIds) {
|
|
|
|
options.invite = this.inviteUserIds;
|
|
|
|
}
|
|
|
|
if (this.isEncrypted) {
|
|
|
|
options.initial_state = [createRoomEncryptionEvent()];
|
|
|
|
}
|
2022-02-07 21:00:44 +05:30
|
|
|
try {
|
|
|
|
const response = await hsApi.createRoom(options, {log}).response();
|
|
|
|
this._roomId = response["room_id"];
|
|
|
|
} catch (err) {
|
|
|
|
this._error = err;
|
|
|
|
}
|
2022-02-04 22:19:10 +05:30
|
|
|
this.emitChange(undefined, log);
|
2022-02-02 14:49:49 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
private async loadProfiles(hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
|
|
|
// only load profiles if we need it for the room name and avatar
|
|
|
|
if (!this.explicitName && this.inviteUserIds) {
|
|
|
|
this.profiles = await loadProfiles(this.inviteUserIds, hsApi, log);
|
2022-02-03 22:27:35 +05:30
|
|
|
const summaryData = {
|
|
|
|
joinCount: 1, // ourselves
|
|
|
|
inviteCount: this.inviteUserIds.length
|
|
|
|
};
|
|
|
|
this._name = calculateRoomName(this.profiles, summaryData, log);
|
|
|
|
this.emitChange();
|
2022-02-02 14:49:49 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-07 21:00:44 +05:30
|
|
|
private emitChange(params?, log?: ILogItem) {
|
2022-02-04 22:19:10 +05:30
|
|
|
this.updateCallback(this, params, log);
|
2022-02-03 22:27:35 +05:30
|
|
|
this.emit("change");
|
|
|
|
}
|
|
|
|
|
2022-02-07 21:00:44 +05:30
|
|
|
get avatarColorId(): string { return this.inviteUserIds?.[0] ?? this._roomId ?? this.localId; }
|
|
|
|
get avatarUrl(): string | undefined { return this.profiles[0]?.avatarUrl; }
|
|
|
|
get roomId(): string | undefined { return this._roomId; }
|
2022-02-03 22:27:35 +05:30
|
|
|
get name() { return this._name; }
|
|
|
|
get isBeingCreated(): boolean { return true; }
|
2022-02-07 21:00:44 +05:30
|
|
|
get error(): Error | undefined { return this._error; }
|
|
|
|
|
|
|
|
cancel() {
|
|
|
|
// remove from collection somehow
|
|
|
|
}
|
2022-02-02 14:49:49 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
export async function loadProfiles(userIds: string[], hsApi: HomeServerApi, log: ILogItem): Promise<Profile[]> {
|
|
|
|
const profiles = await Promise.all(userIds.map(async userId => {
|
|
|
|
const response = await hsApi.profile(userId, {log}).response();
|
|
|
|
return new Profile(userId, response.displayname as string, response.avatar_url as string);
|
|
|
|
}));
|
|
|
|
profiles.sort((a, b) => a.name.localeCompare(b.name));
|
|
|
|
return profiles;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface IProfile {
|
|
|
|
get userId(): string;
|
2022-02-03 22:27:35 +05:30
|
|
|
get displayName(): string | undefined;
|
|
|
|
get avatarUrl(): string | undefined;
|
2022-02-02 14:49:49 +05:30
|
|
|
get name(): string;
|
|
|
|
}
|
|
|
|
|
|
|
|
export class Profile implements IProfile {
|
|
|
|
constructor(
|
|
|
|
public readonly userId: string,
|
|
|
|
public readonly displayName: string,
|
2022-02-03 22:27:35 +05:30
|
|
|
public readonly avatarUrl: string | undefined
|
2022-02-02 14:49:49 +05:30
|
|
|
) {}
|
|
|
|
|
|
|
|
get name() { return this.displayName || this.userId; }
|
|
|
|
}
|
2022-02-03 22:27:35 +05:30
|
|
|
|
|
|
|
class UserIdProfile implements IProfile {
|
|
|
|
constructor(public readonly userId: string) {}
|
|
|
|
get displayName() { return undefined; }
|
|
|
|
get name() { return this.userId; }
|
|
|
|
get avatarUrl() { return undefined; }
|
|
|
|
}
|