/* 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 { logger } from "matrix-js-sdk/src/logger"; import { ClientEvent, MatrixClient } from "matrix-js-sdk/src/matrix"; import defaultDispatcher from "../dispatcher/dispatcher"; import { MatrixClientPeg } from "../MatrixClientPeg"; import { LocalRoom, LocalRoomState } from "../models/LocalRoom"; import { isLocalRoom } from "./localRoom/isLocalRoom"; import { isRoomReady } from "./localRoom/isRoomReady"; /** * Does a room action: * For non-local rooms it calls fn directly. * For local rooms it adds the callback function to the room's afterCreateCallbacks and * dispatches a "local_room_event". * * @async * @template T * @param {string} roomId Room ID of the target room * @param {(actualRoomId: string) => Promise} fn Callback to be called directly or collected at the local room * @param {MatrixClient} [client] * @returns {Promise} Promise that gets resolved after the callback has finished */ export async function doMaybeLocalRoomAction( roomId: string, fn: (actualRoomId: string) => Promise, client?: MatrixClient, ): Promise { if (isLocalRoom(roomId)) { client = client ?? MatrixClientPeg.get(); const room = client.getRoom(roomId) as LocalRoom; if (room.isCreated) { return fn(room.actualRoomId); } return new Promise((resolve, reject) => { room.afterCreateCallbacks.push((newRoomId: string) => { fn(newRoomId).then(resolve).catch(reject); }); defaultDispatcher.dispatch({ action: "local_room_event", roomId: room.roomId, }); }); } return fn(roomId); } /** * Waits until a room is ready and then applies the after-create local room callbacks. * Also implements a stopgap timeout after that a room is assumed to be ready. * * @see isRoomReady * @async * @param {MatrixClient} client * @param {LocalRoom} localRoom * @returns {Promise} Resolved to the actual room id */ export async function waitForRoomReadyAndApplyAfterCreateCallbacks( client: MatrixClient, localRoom: LocalRoom, ): Promise { if (isRoomReady(client, localRoom)) { return applyAfterCreateCallbacks(localRoom, localRoom.actualRoomId).then(() => { localRoom.state = LocalRoomState.CREATED; client.emit(ClientEvent.Room, localRoom); return Promise.resolve(localRoom.actualRoomId); }); } return new Promise((resolve) => { const finish = () => { if (checkRoomStateIntervalHandle) clearInterval(checkRoomStateIntervalHandle); if (stopgapTimeoutHandle) clearTimeout(stopgapTimeoutHandle); applyAfterCreateCallbacks(localRoom, localRoom.actualRoomId).then(() => { localRoom.state = LocalRoomState.CREATED; client.emit(ClientEvent.Room, localRoom); resolve(localRoom.actualRoomId); }); }; const stopgapFinish = () => { logger.warn(`Assuming local room ${localRoom.roomId} is ready after hitting timeout`); finish(); }; const checkRoomStateIntervalHandle = setInterval(() => { if (isRoomReady(client, localRoom)) finish(); }, 500); const stopgapTimeoutHandle = setTimeout(stopgapFinish, 5000); }); } /** * Applies the after-create callback of a local room. * * @async * @param {LocalRoom} localRoom * @param {string} roomId * @returns {Promise} Resolved after all callbacks have been called */ async function applyAfterCreateCallbacks(localRoom: LocalRoom, roomId: string): Promise { for (const afterCreateCallback of localRoom.afterCreateCallbacks) { await afterCreateCallback(roomId); } localRoom.afterCreateCallbacks = []; }