7e481080b5
Signed-off-by: RMidhunSuresh <rmidhunsuresh@gmail.com>
138 lines
4.7 KiB
JavaScript
138 lines
4.7 KiB
JavaScript
/*
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
Copyright 2020 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 {RoomMember} from "./RoomMember.js";
|
|
|
|
async function loadMembers({roomId, storage}) {
|
|
const txn = await storage.readTxn([
|
|
storage.storeNames.roomMembers,
|
|
]);
|
|
const memberDatas = await txn.roomMembers.getAll(roomId);
|
|
return memberDatas.map(d => new RoomMember(d));
|
|
}
|
|
|
|
async function fetchMembers({summary, syncToken, roomId, hsApi, storage, setChangedMembersMap}, log) {
|
|
// if any members are changed by sync while we're fetching members,
|
|
// they will end up here, so we check not to override them
|
|
const changedMembersDuringSync = new Map();
|
|
setChangedMembersMap(changedMembersDuringSync);
|
|
|
|
const memberResponse = await hsApi.members(roomId, {at: syncToken}, {log}).response();
|
|
|
|
const txn = await storage.readWriteTxn([
|
|
storage.storeNames.roomSummary,
|
|
storage.storeNames.roomMembers,
|
|
]);
|
|
|
|
let summaryChanges;
|
|
let members;
|
|
|
|
try {
|
|
summaryChanges = summary.writeHasFetchedMembers(true, txn);
|
|
const {roomMembers} = txn;
|
|
const memberEvents = memberResponse.chunk;
|
|
if (!Array.isArray(memberEvents)) {
|
|
throw new Error("malformed");
|
|
}
|
|
log.set("members", memberEvents.length);
|
|
members = await Promise.all(memberEvents.map(async memberEvent => {
|
|
const userId = memberEvent?.state_key;
|
|
if (!userId) {
|
|
throw new Error("malformed");
|
|
}
|
|
// this member was changed during a sync that happened while calling /members
|
|
// and thus is more recent, so don't overwrite
|
|
const changedMember = changedMembersDuringSync.get(userId);
|
|
if (changedMember) {
|
|
return changedMember;
|
|
} else {
|
|
const member = RoomMember.fromMemberEvent(roomId, memberEvent);
|
|
if (member) {
|
|
roomMembers.set(member.serialize());
|
|
}
|
|
return member;
|
|
}
|
|
}));
|
|
} catch (err) {
|
|
// abort txn on any error
|
|
txn.abort();
|
|
throw err;
|
|
} finally {
|
|
// important this gets cleared
|
|
// or otherwise Room remains in "fetching-members" mode
|
|
setChangedMembersMap(null);
|
|
}
|
|
await txn.complete();
|
|
summary.applyChanges(summaryChanges);
|
|
return members;
|
|
}
|
|
|
|
export async function fetchOrLoadMembers(options, logger) {
|
|
const {summary} = options;
|
|
if (!summary.data.hasFetchedMembers) {
|
|
// we only want to log if we fetch members, so start or continue the optional log operation here
|
|
return logger.wrapOrRun(options.log, "fetchMembers", log => fetchMembers(options, log));
|
|
} else {
|
|
return loadMembers(options);
|
|
}
|
|
}
|
|
|
|
export async function fetchOrLoadMember(options, logger) {
|
|
const member = await loadMember(options);
|
|
const {summary} = options;
|
|
if (!summary.data.hasFetchedMembers && !member) {
|
|
// We haven't fetched the memberlist yet; so ping the hs to see if this member does exist
|
|
return logger.wrapOrRun(options.log, "fetchMember", log => fetchMember(options, log));
|
|
}
|
|
return member;
|
|
}
|
|
|
|
async function loadMember({roomId, userId, storage}) {
|
|
const txn = await storage.readTxn([storage.storeNames.roomMembers,]);
|
|
const member = await txn.roomMembers.get(roomId, userId);
|
|
return member? new RoomMember(member) : null;
|
|
}
|
|
|
|
async function fetchMember({roomId, userId, hsApi, storage}, log) {
|
|
let memberData;
|
|
try {
|
|
memberData = await hsApi.state(roomId, "m.room.member", userId, { log }).response();
|
|
}
|
|
catch (error) {
|
|
if (error.name === "HomeServerError" && error.errcode === "M_NOT_FOUND") {
|
|
return null;
|
|
}
|
|
throw error;
|
|
}
|
|
const member = new RoomMember({
|
|
roomId,
|
|
userId,
|
|
membership: memberData.membership,
|
|
avatarUrl: memberData.avatar_url,
|
|
displayName: memberData.displayname,
|
|
});
|
|
const txn = await storage.readWriteTxn([storage.storeNames.roomMembers]);
|
|
try {
|
|
txn.roomMembers.set(member.serialize());
|
|
}
|
|
catch(e) {
|
|
txn.abort();
|
|
throw e;
|
|
}
|
|
await txn.complete();
|
|
return member;
|
|
}
|