2016-09-06 15:39:21 +00:00
|
|
|
/*
|
2021-01-13 14:49:23 +00:00
|
|
|
Copyright 2016, 2019, 2021 The Matrix.org Foundation C.I.C.
|
2016-09-06 15:39:21 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2021-06-17 13:24:53 +00:00
|
|
|
import { uniq } from "lodash";
|
|
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
2022-02-22 12:18:08 +00:00
|
|
|
import { ClientEvent, MatrixClient } from "matrix-js-sdk/src/client";
|
2021-10-22 22:23:32 +00:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
2022-01-10 17:09:35 +00:00
|
|
|
import { EventType } from "matrix-js-sdk/src/@types/event";
|
|
|
|
import { MatrixEvent } from "matrix-js-sdk/src/models/event";
|
2022-05-03 21:04:37 +00:00
|
|
|
import { Optional } from "matrix-events-sdk";
|
2021-06-17 13:24:53 +00:00
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
import { MatrixClientPeg } from "../MatrixClientPeg";
|
2016-09-26 17:02:14 +00:00
|
|
|
|
2016-09-06 15:39:21 +00:00
|
|
|
/**
|
|
|
|
* Class that takes a Matrix Client and flips the m.direct map
|
|
|
|
* so the operation of mapping a room ID to which user it's a DM
|
|
|
|
* with can be performed efficiently.
|
2016-09-26 17:02:14 +00:00
|
|
|
*
|
|
|
|
* With 'start', this can also keep itself up to date over time.
|
2016-09-06 15:39:21 +00:00
|
|
|
*/
|
|
|
|
export default class DMRoomMap {
|
2021-01-13 14:49:23 +00:00
|
|
|
private static sharedInstance: DMRoomMap;
|
|
|
|
|
|
|
|
// TODO: convert these to maps
|
2022-12-12 11:24:14 +00:00
|
|
|
private roomToUser: { [key: string]: string } = null;
|
|
|
|
private userToRooms: { [key: string]: string[] } = null;
|
2021-01-13 14:49:23 +00:00
|
|
|
private hasSentOutPatchDirectAccountDataPatch: boolean;
|
2022-12-12 11:24:14 +00:00
|
|
|
private mDirectEvent: { [key: string]: string[] };
|
2021-01-13 14:49:23 +00:00
|
|
|
|
2021-06-17 14:18:52 +00:00
|
|
|
constructor(private readonly matrixClient: MatrixClient) {
|
2021-01-13 14:49:23 +00:00
|
|
|
// see onAccountData
|
|
|
|
this.hasSentOutPatchDirectAccountDataPatch = false;
|
2016-09-26 17:02:14 +00:00
|
|
|
|
2022-01-10 17:09:35 +00:00
|
|
|
const mDirectEvent = matrixClient.getAccountData(EventType.Direct)?.getContent() ?? {};
|
|
|
|
this.mDirectEvent = { ...mDirectEvent }; // copy as we will mutate
|
2016-09-06 15:39:21 +00:00
|
|
|
}
|
|
|
|
|
2016-09-27 08:56:31 +00:00
|
|
|
/**
|
|
|
|
* Makes and returns a new shared instance that can then be accessed
|
|
|
|
* with shared(). This returned instance is not automatically started.
|
|
|
|
*/
|
2021-01-13 15:37:49 +00:00
|
|
|
public static makeShared(): DMRoomMap {
|
2021-01-13 14:49:23 +00:00
|
|
|
DMRoomMap.sharedInstance = new DMRoomMap(MatrixClientPeg.get());
|
|
|
|
return DMRoomMap.sharedInstance;
|
2016-09-27 08:56:31 +00:00
|
|
|
}
|
|
|
|
|
2021-04-23 13:39:39 +00:00
|
|
|
/**
|
|
|
|
* Set the shared instance to the instance supplied
|
|
|
|
* Used by tests
|
|
|
|
* @param inst the new shared instance
|
|
|
|
*/
|
|
|
|
public static setShared(inst: DMRoomMap) {
|
|
|
|
DMRoomMap.sharedInstance = inst;
|
|
|
|
}
|
|
|
|
|
2016-09-26 17:02:14 +00:00
|
|
|
/**
|
|
|
|
* Returns a shared instance of the class
|
|
|
|
* that uses the singleton matrix client
|
|
|
|
* The shared instance must be started before use.
|
|
|
|
*/
|
2021-01-13 15:37:49 +00:00
|
|
|
public static shared(): DMRoomMap {
|
2021-01-13 14:49:23 +00:00
|
|
|
return DMRoomMap.sharedInstance;
|
2016-09-26 17:02:14 +00:00
|
|
|
}
|
|
|
|
|
2021-01-13 15:37:49 +00:00
|
|
|
public start() {
|
|
|
|
this.populateRoomToUser();
|
2022-02-22 12:18:08 +00:00
|
|
|
this.matrixClient.on(ClientEvent.AccountData, this.onAccountData);
|
2016-09-26 17:02:14 +00:00
|
|
|
}
|
|
|
|
|
2021-01-13 15:37:49 +00:00
|
|
|
public stop() {
|
2022-02-22 12:18:08 +00:00
|
|
|
this.matrixClient.removeListener(ClientEvent.AccountData, this.onAccountData);
|
2016-09-26 17:02:14 +00:00
|
|
|
}
|
|
|
|
|
2022-01-10 17:09:35 +00:00
|
|
|
private onAccountData = (ev: MatrixEvent) => {
|
|
|
|
if (ev.getType() == EventType.Direct) {
|
|
|
|
this.mDirectEvent = { ...ev.getContent() }; // copy as we will mutate
|
2018-09-04 14:00:40 +00:00
|
|
|
this.userToRooms = null;
|
|
|
|
this.roomToUser = null;
|
2016-09-26 17:02:14 +00:00
|
|
|
}
|
2021-06-29 12:11:58 +00:00
|
|
|
};
|
2021-01-13 14:49:23 +00:00
|
|
|
|
2018-08-30 08:53:25 +00:00
|
|
|
/**
|
|
|
|
* some client bug somewhere is causing some DMs to be marked
|
|
|
|
* with ourself, not the other user. Fix it by guessing the other user and
|
|
|
|
* modifying userToRooms
|
|
|
|
*/
|
2022-01-10 17:09:35 +00:00
|
|
|
private patchUpSelfDMs(userToRooms: Record<string, string[]>) {
|
2018-08-30 08:53:25 +00:00
|
|
|
const myUserId = this.matrixClient.getUserId();
|
|
|
|
const selfRoomIds = userToRooms[myUserId];
|
|
|
|
if (selfRoomIds) {
|
2018-08-30 10:36:53 +00:00
|
|
|
// any self-chats that should not be self-chats?
|
2022-12-12 11:24:14 +00:00
|
|
|
const guessedUserIdsThatChanged = selfRoomIds
|
|
|
|
.map((roomId) => {
|
|
|
|
const room = this.matrixClient.getRoom(roomId);
|
|
|
|
if (room) {
|
|
|
|
const userId = room.guessDMUserId();
|
|
|
|
if (userId && userId !== myUserId) {
|
|
|
|
return { userId, roomId };
|
|
|
|
}
|
2018-08-30 10:36:53 +00:00
|
|
|
}
|
2022-12-12 11:24:14 +00:00
|
|
|
})
|
|
|
|
.filter((ids) => !!ids); //filter out
|
2018-08-30 10:36:53 +00:00
|
|
|
// these are actually all legit self-chats
|
|
|
|
// bail out
|
|
|
|
if (!guessedUserIdsThatChanged.length) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
userToRooms[myUserId] = selfRoomIds.filter((roomId) => {
|
2022-12-12 11:24:14 +00:00
|
|
|
return !guessedUserIdsThatChanged.some((ids) => ids.roomId === roomId);
|
2018-08-30 08:53:25 +00:00
|
|
|
});
|
2021-06-29 12:11:58 +00:00
|
|
|
guessedUserIdsThatChanged.forEach(({ userId, roomId }) => {
|
2018-10-12 03:05:59 +00:00
|
|
|
const roomIds = userToRooms[userId];
|
2018-08-30 08:53:25 +00:00
|
|
|
if (!roomIds) {
|
2018-09-04 11:07:24 +00:00
|
|
|
userToRooms[userId] = [roomId];
|
|
|
|
} else {
|
|
|
|
roomIds.push(roomId);
|
2020-08-28 17:53:43 +00:00
|
|
|
userToRooms[userId] = uniq(roomIds);
|
2018-08-30 08:53:25 +00:00
|
|
|
}
|
|
|
|
});
|
2018-08-30 10:36:53 +00:00
|
|
|
return true;
|
2018-08-30 08:53:25 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-26 17:02:14 +00:00
|
|
|
|
2022-01-10 17:09:35 +00:00
|
|
|
public getDMRoomsForUserId(userId: string): string[] {
|
2016-09-09 16:35:35 +00:00
|
|
|
// Here, we return the empty list if there are no rooms,
|
|
|
|
// since the number of conversations you have with this user is zero.
|
2021-01-13 15:37:49 +00:00
|
|
|
return this.getUserToRooms()[userId] || [];
|
2016-09-06 15:39:21 +00:00
|
|
|
}
|
|
|
|
|
2020-01-15 06:32:00 +00:00
|
|
|
/**
|
|
|
|
* Gets the DM room which the given IDs share, if any.
|
|
|
|
* @param {string[]} ids The identifiers (user IDs and email addresses) to look for.
|
2022-05-09 22:52:05 +00:00
|
|
|
* @returns {Room} The DM room which all IDs given share, or falsy if no common room.
|
2020-01-15 06:32:00 +00:00
|
|
|
*/
|
2021-01-13 15:37:49 +00:00
|
|
|
public getDMRoomForIdentifiers(ids: string[]): Room {
|
2020-01-15 06:32:00 +00:00
|
|
|
// TODO: [Canonical DMs] Handle lookups for email addresses.
|
|
|
|
// For now we'll pretend we only get user IDs and end up returning nothing for email addresses
|
|
|
|
|
|
|
|
let commonRooms = this.getDMRoomsForUserId(ids[0]);
|
|
|
|
for (let i = 1; i < ids.length; i++) {
|
|
|
|
const userRooms = this.getDMRoomsForUserId(ids[i]);
|
2022-12-12 11:24:14 +00:00
|
|
|
commonRooms = commonRooms.filter((r) => userRooms.includes(r));
|
2020-01-15 06:32:00 +00:00
|
|
|
}
|
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
const joinedRooms = commonRooms
|
|
|
|
.map((r) => MatrixClientPeg.get().getRoom(r))
|
|
|
|
.filter((r) => r && r.getMyMembership() === "join");
|
2020-01-15 06:32:00 +00:00
|
|
|
|
|
|
|
return joinedRooms[0];
|
|
|
|
}
|
|
|
|
|
2022-05-03 21:04:37 +00:00
|
|
|
public getUserIdForRoomId(roomId: string): Optional<string> {
|
2016-09-09 15:15:01 +00:00
|
|
|
if (this.roomToUser == null) {
|
|
|
|
// we lazily populate roomToUser so you can use
|
|
|
|
// this class just to call getDMRoomsForUserId
|
|
|
|
// which doesn't do very much, but is a fairly
|
|
|
|
// convenient wrapper and there's no point
|
|
|
|
// iterating through the map if getUserIdForRoomId()
|
|
|
|
// is never called.
|
2021-01-13 15:37:49 +00:00
|
|
|
this.populateRoomToUser();
|
2016-09-09 15:15:01 +00:00
|
|
|
}
|
2016-09-09 16:35:35 +00:00
|
|
|
// Here, we return undefined if the room is not in the map:
|
|
|
|
// the room ID you gave is not a DM room for any user.
|
2016-09-12 17:32:44 +00:00
|
|
|
if (this.roomToUser[roomId] === undefined) {
|
|
|
|
// no entry? if the room is an invite, look for the is_direct hint.
|
|
|
|
const room = this.matrixClient.getRoom(roomId);
|
|
|
|
if (room) {
|
2018-08-14 09:43:03 +00:00
|
|
|
return room.getDMInviter();
|
2016-09-12 17:32:44 +00:00
|
|
|
}
|
|
|
|
}
|
2016-09-06 15:39:21 +00:00
|
|
|
return this.roomToUser[roomId];
|
|
|
|
}
|
2016-09-09 15:15:01 +00:00
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
public getUniqueRoomsWithIndividuals(): { [userId: string]: Room } {
|
2020-03-04 21:18:56 +00:00
|
|
|
if (!this.roomToUser) return {}; // No rooms means no map.
|
2020-01-03 00:40:18 +00:00
|
|
|
return Object.keys(this.roomToUser)
|
2022-12-12 11:24:14 +00:00
|
|
|
.map((r) => ({ userId: this.getUserIdForRoomId(r), room: this.matrixClient.getRoom(r) }))
|
|
|
|
.filter((r) => r.userId && r.room && r.room.getInvitedAndJoinedMemberCount() === 2)
|
2020-01-03 00:40:18 +00:00
|
|
|
.reduce((obj, r) => (obj[r.userId] = r.room) && obj, {});
|
|
|
|
}
|
|
|
|
|
2022-12-12 11:24:14 +00:00
|
|
|
private getUserToRooms(): { [key: string]: string[] } {
|
2018-09-04 14:00:40 +00:00
|
|
|
if (!this.userToRooms) {
|
2022-01-10 17:09:35 +00:00
|
|
|
const userToRooms = this.mDirectEvent;
|
2018-09-04 14:00:40 +00:00
|
|
|
const myUserId = this.matrixClient.getUserId();
|
|
|
|
const selfDMs = userToRooms[myUserId];
|
2022-01-10 17:09:35 +00:00
|
|
|
if (selfDMs?.length) {
|
2021-01-13 15:44:33 +00:00
|
|
|
const neededPatching = this.patchUpSelfDMs(userToRooms);
|
2018-09-04 14:00:40 +00:00
|
|
|
// to avoid multiple devices fighting to correct
|
|
|
|
// the account data, only try to send the corrected
|
|
|
|
// version once.
|
2022-12-12 11:24:14 +00:00
|
|
|
logger.warn(
|
|
|
|
`Invalid m.direct account data detected ` + `(self-chats that shouldn't be), patching it up.`,
|
|
|
|
);
|
2021-01-13 14:49:23 +00:00
|
|
|
if (neededPatching && !this.hasSentOutPatchDirectAccountDataPatch) {
|
|
|
|
this.hasSentOutPatchDirectAccountDataPatch = true;
|
2022-01-10 17:09:35 +00:00
|
|
|
this.matrixClient.setAccountData(EventType.Direct, userToRooms);
|
2018-09-04 14:00:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this.userToRooms = userToRooms;
|
|
|
|
}
|
|
|
|
return this.userToRooms;
|
|
|
|
}
|
|
|
|
|
2021-01-13 15:37:49 +00:00
|
|
|
private populateRoomToUser() {
|
2016-09-09 15:15:01 +00:00
|
|
|
this.roomToUser = {};
|
2021-01-13 15:44:33 +00:00
|
|
|
for (const user of Object.keys(this.getUserToRooms())) {
|
2016-09-09 15:15:01 +00:00
|
|
|
for (const roomId of this.userToRooms[user]) {
|
|
|
|
this.roomToUser[roomId] = user;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-09-06 15:39:21 +00:00
|
|
|
}
|