2022-02-02 10:19:49 +01:00
|
|
|
/*
|
|
|
|
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 17:57:35 +01:00
|
|
|
import {MediaRepository} from "../net/MediaRepository";
|
2022-02-02 10:19:49 +01:00
|
|
|
import {EventEmitter} from "../../utils/EventEmitter";
|
2022-02-09 18:58:30 +01:00
|
|
|
import {AttachmentUpload} from "./AttachmentUpload";
|
2022-02-10 16:39:54 +01:00
|
|
|
import {loadProfiles, Profile, UserIdProfile} from "../profile";
|
|
|
|
import {RoomType} from "./common";
|
2022-02-02 10:19:49 +01:00
|
|
|
|
|
|
|
import type {HomeServerApi} from "../net/HomeServerApi";
|
|
|
|
import type {ILogItem} from "../../logging/types";
|
2022-02-09 18:58:30 +01:00
|
|
|
import type {Platform} from "../../platform/web/Platform";
|
|
|
|
import type {IBlobHandle} from "../../platform/types/types";
|
2022-02-10 19:54:15 +01:00
|
|
|
import type {User} from "../User";
|
|
|
|
import type {Storage} from "../storage/idb/Storage";
|
2022-02-02 10:19:49 +01:00
|
|
|
|
|
|
|
type CreateRoomPayload = {
|
|
|
|
is_direct?: boolean;
|
|
|
|
preset?: string;
|
|
|
|
name?: string;
|
|
|
|
topic?: string;
|
|
|
|
invite?: string[];
|
2022-02-09 18:58:30 +01:00
|
|
|
room_alias_name?: string;
|
2022-02-10 14:09:18 +01:00
|
|
|
creation_content?: {"m.federate": boolean};
|
2022-02-09 18:58:30 +01:00
|
|
|
initial_state: {type: string; state_key: string; content: Record<string, any>}[]
|
|
|
|
}
|
|
|
|
|
|
|
|
type ImageInfo = {
|
|
|
|
w: number;
|
|
|
|
h: number;
|
|
|
|
mimetype: string;
|
|
|
|
size: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
type Avatar = {
|
|
|
|
info: ImageInfo;
|
|
|
|
blob: IBlobHandle;
|
|
|
|
name: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
type Options = {
|
|
|
|
type: RoomType;
|
|
|
|
isEncrypted?: boolean;
|
2022-02-10 14:09:18 +01:00
|
|
|
isFederationDisabled?: boolean;
|
2022-02-09 18:58:30 +01:00
|
|
|
name?: string;
|
|
|
|
topic?: string;
|
|
|
|
invites?: string[];
|
|
|
|
avatar?: Avatar;
|
|
|
|
alias?: string;
|
2022-02-02 10:19:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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 17:57:35 +01:00
|
|
|
export class RoomBeingCreated extends EventEmitter<{change: never}> {
|
2022-02-02 10:19:49 +01:00
|
|
|
private _roomId?: string;
|
|
|
|
private profiles: Profile[] = [];
|
|
|
|
|
|
|
|
public readonly isEncrypted: boolean;
|
2022-02-09 18:58:30 +01:00
|
|
|
private _calculatedName: string;
|
2022-02-07 16:30:44 +01:00
|
|
|
private _error?: Error;
|
2022-02-10 11:07:29 +01:00
|
|
|
private _isCancelled = false;
|
2022-02-02 10:19:49 +01:00
|
|
|
|
|
|
|
constructor(
|
2022-02-10 11:03:52 +01:00
|
|
|
public readonly id: string,
|
2022-02-09 18:58:30 +01:00
|
|
|
private readonly options: Options,
|
2022-02-08 14:58:29 +01:00
|
|
|
private readonly updateCallback: (self: RoomBeingCreated, params: string | undefined) => void,
|
2022-02-03 17:57:35 +01:00
|
|
|
public readonly mediaRepository: MediaRepository,
|
2022-02-09 18:58:30 +01:00
|
|
|
public readonly platform: Platform,
|
2022-02-02 10:19:49 +01:00
|
|
|
log: ILogItem
|
|
|
|
) {
|
|
|
|
super();
|
2022-02-09 18:58:30 +01:00
|
|
|
this.isEncrypted = options.isEncrypted === undefined ? defaultE2EEStatusForType(options.type) : options.isEncrypted;
|
|
|
|
if (options.name) {
|
|
|
|
this._calculatedName = options.name;
|
2022-02-02 10:19:49 +01:00
|
|
|
} else {
|
|
|
|
const summaryData = {
|
|
|
|
joinCount: 1, // ourselves
|
2022-02-09 18:58:30 +01:00
|
|
|
inviteCount: (options.invites?.length || 0)
|
2022-02-02 10:19:49 +01:00
|
|
|
};
|
2022-02-09 18:58:30 +01:00
|
|
|
const userIdProfiles = (options.invites || []).map(userId => new UserIdProfile(userId));
|
|
|
|
this._calculatedName = calculateRoomName(userIdProfiles, summaryData, log);
|
2022-02-02 10:19:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-08 14:58:29 +01:00
|
|
|
/** @internal */
|
2022-02-07 18:58:43 +01:00
|
|
|
async create(hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
2022-02-07 16:30:44 +01:00
|
|
|
try {
|
2022-02-10 11:06:20 +01:00
|
|
|
let avatarEventContent;
|
|
|
|
if (this.options.avatar) {
|
|
|
|
const {avatar} = this.options;
|
|
|
|
const attachment = new AttachmentUpload({filename: avatar.name, blob: avatar.blob, platform: this.platform});
|
|
|
|
await attachment.upload(hsApi, () => {}, log);
|
|
|
|
avatarEventContent = {
|
|
|
|
info: avatar.info
|
|
|
|
};
|
|
|
|
attachment.applyToContent("url", avatarEventContent);
|
|
|
|
}
|
|
|
|
const createOptions: CreateRoomPayload = {
|
|
|
|
is_direct: this.options.type === RoomType.DirectMessage,
|
|
|
|
preset: presetForType(this.options.type),
|
|
|
|
initial_state: []
|
|
|
|
};
|
|
|
|
if (this.options.name) {
|
|
|
|
createOptions.name = this.options.name;
|
|
|
|
}
|
|
|
|
if (this.options.topic) {
|
|
|
|
createOptions.topic = this.options.topic;
|
|
|
|
}
|
|
|
|
if (this.options.invites) {
|
|
|
|
createOptions.invite = this.options.invites;
|
|
|
|
}
|
|
|
|
if (this.options.alias) {
|
|
|
|
createOptions.room_alias_name = this.options.alias;
|
|
|
|
}
|
2022-02-10 14:09:18 +01:00
|
|
|
if (this.options.isFederationDisabled === true) {
|
|
|
|
createOptions.creation_content = {
|
|
|
|
"m.federate": false
|
|
|
|
};
|
|
|
|
}
|
2022-02-10 11:06:20 +01:00
|
|
|
if (this.isEncrypted) {
|
|
|
|
createOptions.initial_state.push(createRoomEncryptionEvent());
|
|
|
|
}
|
|
|
|
if (avatarEventContent) {
|
|
|
|
createOptions.initial_state.push({
|
|
|
|
type: "m.room.avatar",
|
|
|
|
state_key: "",
|
|
|
|
content: avatarEventContent
|
|
|
|
});
|
|
|
|
}
|
2022-02-09 18:58:30 +01:00
|
|
|
const response = await hsApi.createRoom(createOptions, {log}).response();
|
2022-02-07 16:30:44 +01:00
|
|
|
this._roomId = response["room_id"];
|
|
|
|
} catch (err) {
|
|
|
|
this._error = err;
|
|
|
|
}
|
2022-02-08 14:58:29 +01:00
|
|
|
this.emitChange();
|
2022-02-02 10:19:49 +01:00
|
|
|
}
|
|
|
|
|
2022-02-07 18:58:43 +01:00
|
|
|
/** requests the profiles of the invitees if needed to give an accurate
|
|
|
|
* estimated room name in case an explicit room name is not set.
|
|
|
|
* The room is being created in the background whether this is called
|
|
|
|
* or not, and this just gives a more accurate name while that request
|
|
|
|
* is running. */
|
2022-02-08 14:58:29 +01:00
|
|
|
/** @internal */
|
2022-02-07 18:58:43 +01:00
|
|
|
async loadProfiles(hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
2022-02-10 11:06:20 +01:00
|
|
|
try {
|
|
|
|
// only load profiles if we need it for the room name and avatar
|
|
|
|
if (!this.options.name && this.options.invites) {
|
|
|
|
this.profiles = await loadProfiles(this.options.invites, hsApi, log);
|
|
|
|
const summaryData = {
|
|
|
|
joinCount: 1, // ourselves
|
|
|
|
inviteCount: this.options.invites.length
|
|
|
|
};
|
|
|
|
this._calculatedName = calculateRoomName(this.profiles, summaryData, log);
|
|
|
|
this.emitChange();
|
|
|
|
}
|
|
|
|
} catch (err) {} // swallow error, loading profiles is not essential
|
2022-02-02 10:19:49 +01:00
|
|
|
}
|
|
|
|
|
2022-02-08 14:58:29 +01:00
|
|
|
private emitChange(params?: string) {
|
|
|
|
this.updateCallback(this, params);
|
2022-02-03 17:57:35 +01:00
|
|
|
this.emit("change");
|
|
|
|
}
|
|
|
|
|
2022-02-10 11:03:52 +01:00
|
|
|
get avatarColorId(): string { return this.options.invites?.[0] ?? this._roomId ?? this.id; }
|
2022-02-10 11:07:29 +01:00
|
|
|
get avatarUrl(): string | undefined { return this.profiles?.[0]?.avatarUrl; }
|
2022-02-09 18:58:30 +01:00
|
|
|
get avatarBlobUrl(): string | undefined { return this.options.avatar?.blob?.url; }
|
2022-02-07 16:30:44 +01:00
|
|
|
get roomId(): string | undefined { return this._roomId; }
|
2022-02-09 18:58:30 +01:00
|
|
|
get name() { return this._calculatedName; }
|
2022-02-03 17:57:35 +01:00
|
|
|
get isBeingCreated(): boolean { return true; }
|
2022-02-07 16:30:44 +01:00
|
|
|
get error(): Error | undefined { return this._error; }
|
2022-02-10 11:07:29 +01:00
|
|
|
|
2022-02-07 16:30:44 +01:00
|
|
|
cancel() {
|
2022-02-10 11:07:29 +01:00
|
|
|
if (!this._isCancelled) {
|
|
|
|
this.dispose();
|
|
|
|
this._isCancelled = true;
|
|
|
|
this.emitChange("isCancelled");
|
|
|
|
}
|
2022-02-07 16:30:44 +01:00
|
|
|
}
|
2022-02-10 11:07:29 +01:00
|
|
|
// called from Session when updateCallback is invoked to remove it from the collection
|
|
|
|
get isCancelled() { return this._isCancelled; }
|
2022-02-09 18:58:30 +01:00
|
|
|
|
2022-02-10 11:07:29 +01:00
|
|
|
/** @internal */
|
2022-02-09 18:58:30 +01:00
|
|
|
dispose() {
|
|
|
|
if (this.options.avatar) {
|
|
|
|
this.options.avatar.blob.dispose();
|
|
|
|
}
|
|
|
|
}
|
2022-02-10 19:54:15 +01:00
|
|
|
|
|
|
|
async adjustDirectMessageMapIfNeeded(user: User, storage: Storage, hsApi: HomeServerApi, log: ILogItem): Promise<void> {
|
|
|
|
if (!this.options.invites || this.options.type !== RoomType.DirectMessage) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const userId = this.options.invites[0];
|
|
|
|
const DM_MAP_TYPE = "m.direct";
|
|
|
|
await log.wrap("set " + DM_MAP_TYPE, async log => {
|
|
|
|
try {
|
|
|
|
const txn = await storage.readWriteTxn([storage.storeNames.accountData]);
|
|
|
|
let mapEntry;
|
|
|
|
try {
|
|
|
|
mapEntry = await txn.accountData.get(DM_MAP_TYPE);
|
|
|
|
if (!mapEntry) {
|
|
|
|
mapEntry = {type: DM_MAP_TYPE, content: {}};
|
|
|
|
}
|
|
|
|
const map = mapEntry.content;
|
|
|
|
const userRooms = map[userId];
|
|
|
|
// this is a new room id so no need to check if it's already there
|
|
|
|
userRooms.push(this._roomId);
|
|
|
|
txn.accountData.set(mapEntry);
|
|
|
|
await txn.complete();
|
|
|
|
} catch (err) {
|
|
|
|
txn.abort();
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
await hsApi.setAccountData(user.id, DM_MAP_TYPE, mapEntry.content, {log}).response();
|
|
|
|
} catch (err) {
|
|
|
|
// we can't really do anything else but logging here
|
|
|
|
log.catch(err);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2022-02-02 10:19:49 +01:00
|
|
|
}
|