From fe3e455c20814c214c0bc00ac5c3949160d7e3b2 Mon Sep 17 00:00:00 2001 From: Michael Bromley Date: Mon, 15 Jul 2024 11:32:43 +0200 Subject: [PATCH] perf(core): Improve performance of role query with many channels Fixes #2910. In local testing with 505 channels, executing the Admin UI `GetActiveAdministrator` query, the average response time was 29.05s. With this optimization the average response came down to 106ms. --- .../core/src/service/services/role.service.ts | 23 +++++++++++++++---- 1 file changed, 18 insertions(+), 5 deletions(-) diff --git a/packages/core/src/service/services/role.service.ts b/packages/core/src/service/services/role.service.ts index 6a29b6f908..f551de8bdb 100644 --- a/packages/core/src/service/services/role.service.ts +++ b/packages/core/src/service/services/role.service.ts @@ -17,6 +17,7 @@ import { unique } from '@vendure/common/lib/unique'; import { RequestContext } from '../../api/common/request-context'; import { RelationPaths } from '../../api/decorators/relations.decorator'; +import { RequestContextCacheService } from '../../cache/request-context-cache.service'; import { getAllPermissionsMetadata } from '../../common/constants'; import { EntityNotFoundError, @@ -56,6 +57,7 @@ export class RoleService { private listQueryBuilder: ListQueryBuilder, private configService: ConfigService, private eventBus: EventBus, + private requestContextCache: RequestContextCacheService, ) {} async initRoles() { @@ -206,13 +208,24 @@ export class RoleService { ctx: RequestContext, channelId: ID, ): Promise { - if (ctx.activeUserId == null) { + const { activeUserId } = ctx; + if (activeUserId == null) { return []; } - const user = await this.connection.getEntityOrThrow(ctx, User, ctx.activeUserId, { - relations: ['roles', 'roles.channels'], - }); - const userChannels = getUserChannelsPermissions(user); + // For apps with many channels, this is a performance bottleneck as it will be called + // for each channel in certain code paths such as the GetActiveAdministrator query in the + // admin ui. Caching the result prevents unbounded quadratic slowdown. + const userChannels = await this.requestContextCache.get( + ctx, + `RoleService.getActiveUserPermissionsOnChannel.user(${activeUserId})`, + async () => { + const user = await this.connection.getEntityOrThrow(ctx, User, activeUserId, { + relations: ['roles', 'roles.channels'], + }); + return getUserChannelsPermissions(user); + }, + ); + const channel = userChannels.find(c => idsAreEqual(c.id, channelId)); if (!channel) { return [];