summaryrefslogtreecommitdiff
path: root/packages/backend/src/core/UserCacheService.ts
diff options
context:
space:
mode:
authorsyuilo <Syuilotan@yahoo.co.jp>2022-09-18 03:27:08 +0900
committerGitHub <noreply@github.com>2022-09-18 03:27:08 +0900
commitb75184ec8e3436200bacdcd832e3324702553d20 (patch)
tree8b7e316f29e95df921db57289c8b8da476d18f07 /packages/backend/src/core/UserCacheService.ts
parentUpdate ROADMAP.md (diff)
downloadsharkey-b75184ec8e3436200bacdcd832e3324702553d20.tar.gz
sharkey-b75184ec8e3436200bacdcd832e3324702553d20.tar.bz2
sharkey-b75184ec8e3436200bacdcd832e3324702553d20.zip
なんかもうめっちゃ変えた
Diffstat (limited to 'packages/backend/src/core/UserCacheService.ts')
-rw-r--r--packages/backend/src/core/UserCacheService.ts74
1 files changed, 74 insertions, 0 deletions
diff --git a/packages/backend/src/core/UserCacheService.ts b/packages/backend/src/core/UserCacheService.ts
new file mode 100644
index 0000000000..8212abf7bb
--- /dev/null
+++ b/packages/backend/src/core/UserCacheService.ts
@@ -0,0 +1,74 @@
+import { Inject, Injectable } from '@nestjs/common';
+import Redis from 'ioredis';
+import { UsersRepository } from '@/models/index.js';
+import { Cache } from '@/misc/cache.js';
+import type { CacheableLocalUser, CacheableUser, ILocalUser } from '@/models/entities/User.js';
+import { DI } from '@/di-symbols.js';
+import { UserEntityService } from './entities/UserEntityService.js';
+import type { OnApplicationShutdown } from '@nestjs/common';
+
+@Injectable()
+export class UserCacheService implements OnApplicationShutdown {
+ public userByIdCache: Cache<CacheableUser>;
+ public localUserByNativeTokenCache: Cache<CacheableLocalUser | null>;
+ public localUserByIdCache: Cache<CacheableLocalUser>;
+ public uriPersonCache: Cache<CacheableUser | null>;
+
+ constructor(
+ @Inject(DI.redisSubscriber)
+ private redisSubscriber: Redis.Redis,
+
+ @Inject(DI.usersRepository)
+ private usersRepository: UsersRepository,
+
+ private userEntityService: UserEntityService,
+ ) {
+ this.onMessage = this.onMessage.bind(this);
+
+ this.userByIdCache = new Cache<CacheableUser>(Infinity);
+ this.localUserByNativeTokenCache = new Cache<CacheableLocalUser | null>(Infinity);
+ this.localUserByIdCache = new Cache<CacheableLocalUser>(Infinity);
+ this.uriPersonCache = new Cache<CacheableUser | null>(Infinity);
+
+ this.redisSubscriber.on('message', this.onMessage);
+ }
+
+ private async onMessage(_, data) {
+ const obj = JSON.parse(data);
+
+ if (obj.channel === 'internal') {
+ const { type, body } = obj.message;
+ switch (type) {
+ case 'userChangeSuspendedState':
+ case 'userChangeSilencedState':
+ case 'userChangeModeratorState':
+ case 'remoteUserUpdated': {
+ const user = await this.usersRepository.findOneByOrFail({ id: body.id });
+ this.userByIdCache.set(user.id, user);
+ for (const [k, v] of this.uriPersonCache.cache.entries()) {
+ if (v.value?.id === user.id) {
+ this.uriPersonCache.set(k, user);
+ }
+ }
+ if (this.userEntityService.isLocalUser(user)) {
+ this.localUserByNativeTokenCache.set(user.token, user);
+ this.localUserByIdCache.set(user.id, user);
+ }
+ break;
+ }
+ case 'userTokenRegenerated': {
+ const user = await this.usersRepository.findOneByOrFail({ id: body.id }) as ILocalUser;
+ this.localUserByNativeTokenCache.delete(body.oldToken);
+ this.localUserByNativeTokenCache.set(body.newToken, user);
+ break;
+ }
+ default:
+ break;
+ }
+ }
+ }
+
+ public onApplicationShutdown(signal?: string | undefined) {
+ this.redisSubscriber.off('message', this.onMessage);
+ }
+}