2023-03-27 11:07:43 +03:00
|
|
|
/*
|
|
|
|
Copyright 2023 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";
|
2023-04-05 14:13:51 +03:00
|
|
|
import {
|
|
|
|
IMatrixProfile,
|
|
|
|
MatrixClient,
|
|
|
|
MatrixError,
|
|
|
|
MatrixEvent,
|
|
|
|
RoomMember,
|
|
|
|
RoomMemberEvent,
|
|
|
|
} from "matrix-js-sdk/src/matrix";
|
2023-03-27 11:07:43 +03:00
|
|
|
|
|
|
|
import { LruCache } from "../utils/LruCache";
|
|
|
|
|
|
|
|
const cacheSize = 500;
|
|
|
|
|
|
|
|
type StoreProfileValue = IMatrixProfile | undefined | null;
|
|
|
|
|
2023-04-05 14:13:51 +03:00
|
|
|
interface GetOptions {
|
|
|
|
/** Whether calling the function shouuld raise an Error. */
|
|
|
|
shouldThrow: boolean;
|
|
|
|
}
|
|
|
|
|
2023-03-27 11:07:43 +03:00
|
|
|
/**
|
|
|
|
* This store provides cached access to user profiles.
|
|
|
|
* Listens for membership events and invalidates the cache for a profile on update with different profile values.
|
|
|
|
*/
|
|
|
|
export class UserProfilesStore {
|
|
|
|
private profiles = new LruCache<string, IMatrixProfile | null>(cacheSize);
|
2023-04-05 14:13:51 +03:00
|
|
|
private profileLookupErrors = new LruCache<string, MatrixError>(cacheSize);
|
2023-03-27 11:07:43 +03:00
|
|
|
private knownProfiles = new LruCache<string, IMatrixProfile | null>(cacheSize);
|
|
|
|
|
|
|
|
public constructor(private client: MatrixClient) {
|
|
|
|
client.on(RoomMemberEvent.Membership, this.onRoomMembershipEvent);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Synchronously get a profile from the store cache.
|
|
|
|
*
|
|
|
|
* @param userId - User Id of the profile to fetch
|
|
|
|
* @returns The profile, if cached by the store.
|
|
|
|
* Null if the profile does not exist.
|
|
|
|
* Undefined if the profile is not cached by the store.
|
|
|
|
* In this case a profile can be fetched from the API via {@link fetchProfile}.
|
|
|
|
*/
|
|
|
|
public getProfile(userId: string): StoreProfileValue {
|
|
|
|
return this.profiles.get(userId);
|
|
|
|
}
|
|
|
|
|
2023-04-05 14:13:51 +03:00
|
|
|
/**
|
|
|
|
* Async shortcut function that returns the profile from cache or
|
|
|
|
* or fetches it on cache miss.
|
|
|
|
*
|
|
|
|
* @param userId - User Id of the profile to get or fetch
|
|
|
|
* @returns The profile, if cached by the store or fetched from the API.
|
|
|
|
* Null if the profile does not exist or an error occurred during fetch.
|
|
|
|
*/
|
|
|
|
public async getOrFetchProfile(userId: string, options?: GetOptions): Promise<IMatrixProfile | null> {
|
|
|
|
const cachedProfile = this.profiles.get(userId);
|
|
|
|
|
|
|
|
if (cachedProfile) return cachedProfile;
|
|
|
|
|
|
|
|
return this.fetchProfile(userId, options);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get a profile lookup error.
|
|
|
|
*
|
|
|
|
* @param userId - User Id for which to get the lookup error
|
|
|
|
* @returns The lookup error or undefined if there was no error or the profile was not fetched.
|
|
|
|
*/
|
|
|
|
public getProfileLookupError(userId: string): MatrixError | undefined {
|
|
|
|
return this.profileLookupErrors.get(userId);
|
|
|
|
}
|
|
|
|
|
2023-03-27 11:07:43 +03:00
|
|
|
/**
|
|
|
|
* Synchronously get a profile from known users from the store cache.
|
|
|
|
* Known user means that at least one shared room with the user exists.
|
|
|
|
*
|
|
|
|
* @param userId - User Id of the profile to fetch
|
|
|
|
* @returns The profile, if cached by the store.
|
|
|
|
* Null if the profile does not exist.
|
|
|
|
* Undefined if the profile is not cached by the store.
|
|
|
|
* In this case a profile can be fetched from the API via {@link fetchOnlyKnownProfile}.
|
|
|
|
*/
|
|
|
|
public getOnlyKnownProfile(userId: string): StoreProfileValue {
|
|
|
|
return this.knownProfiles.get(userId);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Asynchronousely fetches a profile from the API.
|
|
|
|
* Stores the result in the cache, so that next time {@link getProfile} returns this value.
|
|
|
|
*
|
|
|
|
* @param userId - User Id for which the profile should be fetched for
|
|
|
|
* @returns The profile, if found.
|
|
|
|
* Null if the profile does not exist or there was an error fetching it.
|
|
|
|
*/
|
2023-04-05 14:13:51 +03:00
|
|
|
public async fetchProfile(userId: string, options?: GetOptions): Promise<IMatrixProfile | null> {
|
|
|
|
const profile = await this.fetchProfileFromApi(userId, options);
|
2023-03-27 11:07:43 +03:00
|
|
|
this.profiles.set(userId, profile);
|
|
|
|
return profile;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Asynchronousely fetches a profile from a known user from the API.
|
|
|
|
* Known user means that at least one shared room with the user exists.
|
|
|
|
* Stores the result in the cache, so that next time {@link getOnlyKnownProfile} returns this value.
|
|
|
|
*
|
|
|
|
* @param userId - User Id for which the profile should be fetched for
|
|
|
|
* @returns The profile, if found.
|
|
|
|
* Undefined if the user is unknown.
|
|
|
|
* Null if the profile does not exist or there was an error fetching it.
|
|
|
|
*/
|
|
|
|
public async fetchOnlyKnownProfile(userId: string): Promise<StoreProfileValue> {
|
|
|
|
// Do not look up unknown users. The test for existence in knownProfiles is a performance optimisation.
|
|
|
|
// If the user Id exists in knownProfiles we know them.
|
|
|
|
if (!this.knownProfiles.has(userId) && !this.isUserIdKnown(userId)) return undefined;
|
|
|
|
|
|
|
|
const profile = await this.fetchProfileFromApi(userId);
|
|
|
|
this.knownProfiles.set(userId, profile);
|
|
|
|
return profile;
|
|
|
|
}
|
|
|
|
|
2023-04-05 14:13:51 +03:00
|
|
|
public flush(): void {
|
|
|
|
this.profiles = new LruCache<string, IMatrixProfile | null>(cacheSize);
|
|
|
|
this.profileLookupErrors = new LruCache<string, MatrixError>(cacheSize);
|
|
|
|
this.knownProfiles = new LruCache<string, IMatrixProfile | null>(cacheSize);
|
|
|
|
}
|
|
|
|
|
2023-03-27 11:07:43 +03:00
|
|
|
/**
|
|
|
|
* Looks up a user profile via API.
|
|
|
|
*
|
|
|
|
* @param userId - User Id for which the profile should be fetched for
|
|
|
|
* @returns The profile information or null on errors
|
|
|
|
*/
|
2023-04-05 14:13:51 +03:00
|
|
|
private async fetchProfileFromApi(userId: string, options?: GetOptions): Promise<IMatrixProfile | null> {
|
|
|
|
// invalidate cached profile errors
|
|
|
|
this.profileLookupErrors.delete(userId);
|
|
|
|
|
2023-03-27 11:07:43 +03:00
|
|
|
try {
|
|
|
|
return (await this.client.getProfileInfo(userId)) ?? null;
|
|
|
|
} catch (e) {
|
|
|
|
logger.warn(`Error retrieving profile for userId ${userId}`, e);
|
2023-04-05 14:13:51 +03:00
|
|
|
|
|
|
|
if (e instanceof MatrixError) {
|
|
|
|
this.profileLookupErrors.set(userId, e);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options?.shouldThrow) {
|
|
|
|
throw e;
|
|
|
|
}
|
2023-03-27 11:07:43 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether at least one shared room with the userId exists.
|
|
|
|
*
|
|
|
|
* @param userId
|
|
|
|
* @returns true: at least one room shared with user identified by its Id, else false.
|
|
|
|
*/
|
|
|
|
private isUserIdKnown(userId: string): boolean {
|
|
|
|
return this.client.getRooms().some((room) => {
|
|
|
|
return !!room.getMember(userId);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Simple cache invalidation if a room membership event is received and
|
|
|
|
* at least one profile value differs from the cached one.
|
|
|
|
*/
|
|
|
|
private onRoomMembershipEvent = (event: MatrixEvent, member: RoomMember): void => {
|
|
|
|
const profile = this.profiles.get(member.userId);
|
|
|
|
|
|
|
|
if (
|
|
|
|
profile &&
|
|
|
|
(profile.displayname !== member.rawDisplayName || profile.avatar_url !== member.getMxcAvatarUrl())
|
|
|
|
) {
|
|
|
|
this.profiles.delete(member.userId);
|
|
|
|
}
|
|
|
|
|
|
|
|
const knownProfile = this.knownProfiles.get(member.userId);
|
|
|
|
|
|
|
|
if (
|
|
|
|
knownProfile &&
|
|
|
|
(knownProfile.displayname !== member.rawDisplayName || knownProfile.avatar_url !== member.getMxcAvatarUrl())
|
|
|
|
) {
|
|
|
|
this.knownProfiles.delete(member.userId);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|