summaryrefslogtreecommitdiff
path: root/packages/backend/src
diff options
context:
space:
mode:
authorおさむのひと <46447427+samunohito@users.noreply.github.com>2024-03-12 14:31:34 +0900
committerGitHub <noreply@github.com>2024-03-12 14:31:34 +0900
commit5c1d86b796d6ab878bc4f9bd2faf4207998e71cf (patch)
treeedf98d9b2a6b8d1e44a510c718c83308a8572b17 /packages/backend/src
parentfix: URL preview popup for local URL appears in the upper left corner (#13555) (diff)
downloadsharkey-5c1d86b796d6ab878bc4f9bd2faf4207998e71cf.tar.gz
sharkey-5c1d86b796d6ab878bc4f9bd2faf4207998e71cf.tar.bz2
sharkey-5c1d86b796d6ab878bc4f9bd2faf4207998e71cf.zip
refactor(backend): UserEntityService.packMany()の高速化 (#13550)
* refactor(backend): UserEntityService.packMany()の高速化 * 修正
Diffstat (limited to 'packages/backend/src')
-rw-r--r--packages/backend/src/core/entities/UserEntityService.ts229
-rw-r--r--packages/backend/src/server/api/endpoints/users/relation.ts8
2 files changed, 201 insertions, 36 deletions
diff --git a/packages/backend/src/core/entities/UserEntityService.ts b/packages/backend/src/core/entities/UserEntityService.ts
index 14761357a5..df2b27d709 100644
--- a/packages/backend/src/core/entities/UserEntityService.ts
+++ b/packages/backend/src/core/entities/UserEntityService.ts
@@ -7,6 +7,7 @@ import { Inject, Injectable } from '@nestjs/common';
import * as Redis from 'ioredis';
import _Ajv from 'ajv';
import { ModuleRef } from '@nestjs/core';
+import { In } from 'typeorm';
import { DI } from '@/di-symbols.js';
import type { Config } from '@/config.js';
import type { Packed } from '@/misc/json-schema.js';
@@ -14,9 +15,30 @@ import type { Promiseable } from '@/misc/prelude/await-all.js';
import { awaitAll } from '@/misc/prelude/await-all.js';
import { USER_ACTIVE_THRESHOLD, USER_ONLINE_THRESHOLD } from '@/const.js';
import type { MiLocalUser, MiPartialLocalUser, MiPartialRemoteUser, MiRemoteUser, MiUser } from '@/models/User.js';
-import { birthdaySchema, descriptionSchema, localUsernameSchema, locationSchema, nameSchema, passwordSchema } from '@/models/User.js';
-import { MiNotification } from '@/models/Notification.js';
-import type { UsersRepository, UserSecurityKeysRepository, FollowingsRepository, FollowRequestsRepository, BlockingsRepository, MutingsRepository, DriveFilesRepository, NoteUnreadsRepository, UserNotePiningsRepository, UserProfilesRepository, AnnouncementReadsRepository, AnnouncementsRepository, MiUserProfile, RenoteMutingsRepository, UserMemoRepository } from '@/models/_.js';
+import {
+ birthdaySchema,
+ descriptionSchema,
+ localUsernameSchema,
+ locationSchema,
+ nameSchema,
+ passwordSchema,
+} from '@/models/User.js';
+import type {
+ BlockingsRepository,
+ FollowingsRepository,
+ FollowRequestsRepository,
+ MiFollowing,
+ MiUserNotePining,
+ MiUserProfile,
+ MutingsRepository,
+ NoteUnreadsRepository,
+ RenoteMutingsRepository,
+ UserMemoRepository,
+ UserNotePiningsRepository,
+ UserProfilesRepository,
+ UserSecurityKeysRepository,
+ UsersRepository,
+} from '@/models/_.js';
import { bindThis } from '@/decorators.js';
import { RoleService } from '@/core/RoleService.js';
import { ApPersonService } from '@/core/activitypub/models/ApPersonService.js';
@@ -46,11 +68,23 @@ function isRemoteUser(user: MiUser | { host: MiUser['host'] }): boolean {
return !isLocalUser(user);
}
+export type UserRelation = {
+ id: MiUser['id']
+ following: MiFollowing | null,
+ isFollowing: boolean
+ isFollowed: boolean
+ hasPendingFollowRequestFromYou: boolean
+ hasPendingFollowRequestToYou: boolean
+ isBlocking: boolean
+ isBlocked: boolean
+ isMuted: boolean
+ isRenoteMuted: boolean
+}
+
@Injectable()
export class UserEntityService implements OnModuleInit {
private apPersonService: ApPersonService;
private noteEntityService: NoteEntityService;
- private driveFileEntityService: DriveFileEntityService;
private pageEntityService: PageEntityService;
private customEmojiService: CustomEmojiService;
private announcementService: AnnouncementService;
@@ -89,9 +123,6 @@ export class UserEntityService implements OnModuleInit {
@Inject(DI.renoteMutingsRepository)
private renoteMutingsRepository: RenoteMutingsRepository,
- @Inject(DI.driveFilesRepository)
- private driveFilesRepository: DriveFilesRepository,
-
@Inject(DI.noteUnreadsRepository)
private noteUnreadsRepository: NoteUnreadsRepository,
@@ -101,12 +132,6 @@ export class UserEntityService implements OnModuleInit {
@Inject(DI.userProfilesRepository)
private userProfilesRepository: UserProfilesRepository,
- @Inject(DI.announcementReadsRepository)
- private announcementReadsRepository: AnnouncementReadsRepository,
-
- @Inject(DI.announcementsRepository)
- private announcementsRepository: AnnouncementsRepository,
-
@Inject(DI.userMemosRepository)
private userMemosRepository: UserMemoRepository,
) {
@@ -115,7 +140,6 @@ export class UserEntityService implements OnModuleInit {
onModuleInit() {
this.apPersonService = this.moduleRef.get('ApPersonService');
this.noteEntityService = this.moduleRef.get('NoteEntityService');
- this.driveFileEntityService = this.moduleRef.get('DriveFileEntityService');
this.pageEntityService = this.moduleRef.get('PageEntityService');
this.customEmojiService = this.moduleRef.get('CustomEmojiService');
this.announcementService = this.moduleRef.get('AnnouncementService');
@@ -138,7 +162,7 @@ export class UserEntityService implements OnModuleInit {
public isRemoteUser = isRemoteUser;
@bindThis
- public async getRelation(me: MiUser['id'], target: MiUser['id']) {
+ public async getRelation(me: MiUser['id'], target: MiUser['id']): Promise<UserRelation> {
const [
following,
isFollowed,
@@ -212,6 +236,59 @@ export class UserEntityService implements OnModuleInit {
}
@bindThis
+ public async getRelations(me: MiUser['id'], targets: MiUser['id'][]): Promise<Map<MiUser['id'], UserRelation>> {
+ const [
+ followers,
+ followees,
+ followersRequests,
+ followeesRequests,
+ blockers,
+ blockees,
+ muters,
+ renoteMuters,
+ ] = await Promise.all([
+ this.followingsRepository.findBy({ followerId: me })
+ .then(f => new Map(f.map(it => [it.followeeId, it]))),
+ this.followingsRepository.findBy({ followeeId: me })
+ .then(it => it.map(it => it.followerId)),
+ this.followRequestsRepository.findBy({ followerId: me })
+ .then(it => it.map(it => it.followeeId)),
+ this.followRequestsRepository.findBy({ followeeId: me })
+ .then(it => it.map(it => it.followerId)),
+ this.blockingsRepository.findBy({ blockerId: me })
+ .then(it => it.map(it => it.blockeeId)),
+ this.blockingsRepository.findBy({ blockeeId: me })
+ .then(it => it.map(it => it.blockerId)),
+ this.mutingsRepository.findBy({ muterId: me })
+ .then(it => it.map(it => it.muteeId)),
+ this.renoteMutingsRepository.findBy({ muterId: me })
+ .then(it => it.map(it => it.muteeId)),
+ ]);
+
+ return new Map(
+ targets.map(target => {
+ const following = followers.get(target) ?? null;
+
+ return [
+ target,
+ {
+ id: target,
+ following: following,
+ isFollowing: following != null,
+ isFollowed: followees.includes(target),
+ hasPendingFollowRequestFromYou: followersRequests.includes(target),
+ hasPendingFollowRequestToYou: followeesRequests.includes(target),
+ isBlocking: blockers.includes(target),
+ isBlocked: blockees.includes(target),
+ isMuted: muters.includes(target),
+ isRenoteMuted: renoteMuters.includes(target),
+ },
+ ];
+ }),
+ );
+ }
+
+ @bindThis
public async getHasUnreadAntenna(userId: MiUser['id']): Promise<boolean> {
/*
const myAntennas = (await this.antennaService.getAntennas()).filter(a => a.userId === userId);
@@ -303,6 +380,9 @@ export class UserEntityService implements OnModuleInit {
schema?: S,
includeSecrets?: boolean,
userProfile?: MiUserProfile,
+ userRelations?: Map<MiUser['id'], UserRelation>,
+ userMemos?: Map<MiUser['id'], string | null>,
+ pinNotes?: Map<MiUser['id'], MiUserNotePining[]>,
},
): Promise<Packed<S>> {
const opts = Object.assign({
@@ -317,13 +397,41 @@ export class UserEntityService implements OnModuleInit {
const isMe = meId === user.id;
const iAmModerator = me ? await this.roleService.isModerator(me as MiUser) : false;
- const relation = meId && !isMe && isDetailed ? await this.getRelation(meId, user.id) : null;
- const pins = isDetailed ? await this.userNotePiningsRepository.createQueryBuilder('pin')
- .where('pin.userId = :userId', { userId: user.id })
- .innerJoinAndSelect('pin.note', 'note')
- .orderBy('pin.id', 'DESC')
- .getMany() : [];
- const profile = isDetailed ? (opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id })) : null;
+ const profile = isDetailed
+ ? (opts.userProfile ?? await this.userProfilesRepository.findOneByOrFail({ userId: user.id }))
+ : null;
+
+ let relation: UserRelation | null = null;
+ if (meId && !isMe && isDetailed) {
+ if (opts.userRelations) {
+ relation = opts.userRelations.get(user.id) ?? null;
+ } else {
+ relation = await this.getRelation(meId, user.id);
+ }
+ }
+
+ let memo: string | null = null;
+ if (isDetailed && meId) {
+ if (opts.userMemos) {
+ memo = opts.userMemos.get(user.id) ?? null;
+ } else {
+ memo = await this.userMemosRepository.findOneBy({ userId: meId, targetUserId: user.id })
+ .then(row => row?.memo ?? null);
+ }
+ }
+
+ let pins: MiUserNotePining[] = [];
+ if (isDetailed) {
+ if (opts.pinNotes) {
+ pins = opts.pinNotes.get(user.id) ?? [];
+ } else {
+ pins = await this.userNotePiningsRepository.createQueryBuilder('pin')
+ .where('pin.userId = :userId', { userId: user.id })
+ .innerJoinAndSelect('pin.note', 'note')
+ .orderBy('pin.id', 'DESC')
+ .getMany();
+ }
+ }
const followingCount = profile == null ? null :
(profile.followingVisibility === 'public') || isMe ? user.followingCount :
@@ -416,9 +524,7 @@ export class UserEntityService implements OnModuleInit {
twoFactorEnabled: profile!.twoFactorEnabled,
usePasswordLessLogin: profile!.usePasswordLessLogin,
securityKeys: profile!.twoFactorEnabled
- ? this.userSecurityKeysRepository.countBy({
- userId: user.id,
- }).then(result => result >= 1)
+ ? this.userSecurityKeysRepository.countBy({ userId: user.id }).then(result => result >= 1)
: false,
roles: this.roleService.getUserRoles(user.id).then(roles => roles.filter(role => role.isPublic).sort((a, b) => b.displayOrder - a.displayOrder).map(role => ({
id: role.id,
@@ -430,10 +536,7 @@ export class UserEntityService implements OnModuleInit {
isAdministrator: role.isAdministrator,
displayOrder: role.displayOrder,
}))),
- memo: meId == null ? null : await this.userMemosRepository.findOneBy({
- userId: meId,
- targetUserId: user.id,
- }).then(row => row?.memo ?? null),
+ memo: memo,
moderationNote: iAmModerator ? (profile!.moderationNote ?? '') : undefined,
} : {}),
@@ -514,7 +617,7 @@ export class UserEntityService implements OnModuleInit {
return await awaitAll(packed);
}
- public packMany<S extends 'MeDetailed' | 'UserDetailedNotMe' | 'UserDetailed' | 'UserLite' = 'UserLite'>(
+ public async packMany<S extends 'MeDetailed' | 'UserDetailedNotMe' | 'UserDetailed' | 'UserLite' = 'UserLite'>(
users: (MiUser['id'] | MiUser)[],
me?: { id: MiUser['id'] } | null | undefined,
options?: {
@@ -522,6 +625,70 @@ export class UserEntityService implements OnModuleInit {
includeSecrets?: boolean,
},
): Promise<Packed<S>[]> {
- return Promise.all(users.map(u => this.pack(u, me, options)));
+ // -- IDのみの要素を補完して完全なエンティティ一覧を作る
+
+ const _users = users.filter((user): user is MiUser => typeof user !== 'string');
+ if (_users.length !== users.length) {
+ _users.push(
+ ...await this.usersRepository.findBy({
+ id: In(users.filter((user): user is string => typeof user === 'string')),
+ }),
+ );
+ }
+ const _userIds = _users.map(u => u.id);
+
+ // -- 特に前提条件のない値群を取得
+
+ const profilesMap = await this.userProfilesRepository.findBy({ userId: In(_userIds) })
+ .then(profiles => new Map(profiles.map(p => [p.userId, p])));
+
+ // -- 実行者の有無や指定スキーマの種別によって要否が異なる値群を取得
+
+ let userRelations: Map<MiUser['id'], UserRelation> = new Map();
+ let userMemos: Map<MiUser['id'], string | null> = new Map();
+ let pinNotes: Map<MiUser['id'], MiUserNotePining[]> = new Map();
+
+ if (options?.schema !== 'UserLite') {
+ const meId = me ? me.id : null;
+ if (meId) {
+ userMemos = await this.userMemosRepository.findBy({ userId: meId })
+ .then(memos => new Map(memos.map(memo => [memo.targetUserId, memo.memo])));
+
+ if (_userIds.length > 0) {
+ userRelations = await this.getRelations(meId, _userIds);
+ pinNotes = await this.userNotePiningsRepository.createQueryBuilder('pin')
+ .where('pin.userId IN (:...userIds)', { userIds: _userIds })
+ .innerJoinAndSelect('pin.note', 'note')
+ .getMany()
+ .then(pinsNotes => {
+ const map = new Map<MiUser['id'], MiUserNotePining[]>();
+ for (const note of pinsNotes) {
+ const notes = map.get(note.userId) ?? [];
+ notes.push(note);
+ map.set(note.userId, notes);
+ }
+ for (const [, notes] of map.entries()) {
+ // pack側ではDESCで取得しているので、それに合わせて降順に並び替えておく
+ notes.sort((a, b) => b.id.localeCompare(a.id));
+ }
+ return map;
+ });
+ }
+ }
+ }
+
+ return Promise.all(
+ _users.map(u => this.pack(
+ u,
+ me,
+ {
+ ...options,
+ userProfile: profilesMap.get(u.id),
+ userRelations: userRelations,
+ userMemos: userMemos,
+ pinNotes: pinNotes,
+ },
+ )),
+ );
}
}
diff --git a/packages/backend/src/server/api/endpoints/users/relation.ts b/packages/backend/src/server/api/endpoints/users/relation.ts
index 6a5b2262fa..1d75437b81 100644
--- a/packages/backend/src/server/api/endpoints/users/relation.ts
+++ b/packages/backend/src/server/api/endpoints/users/relation.ts
@@ -132,11 +132,9 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
private userEntityService: UserEntityService,
) {
super(meta, paramDef, async (ps, me) => {
- const ids = Array.isArray(ps.userId) ? ps.userId : [ps.userId];
-
- const relations = await Promise.all(ids.map(id => this.userEntityService.getRelation(me.id, id)));
-
- return Array.isArray(ps.userId) ? relations : relations[0];
+ return Array.isArray(ps.userId)
+ ? await this.userEntityService.getRelations(me.id, ps.userId).then(it => [...it.values()])
+ : await this.userEntityService.getRelation(me.id, ps.userId).then(it => [it]);
});
}
}