summaryrefslogtreecommitdiff
path: root/packages/backend/src/core/QueryService.ts
diff options
context:
space:
mode:
authorHazelnoot <acomputerdog@gmail.com>2025-06-04 12:40:13 +0000
committerHazelnoot <acomputerdog@gmail.com>2025-06-04 12:40:13 +0000
commitdae544b353ebbc6edd85c032786b9901807460e8 (patch)
treef8495c993b0c29f911220ade7df8a5f3f076e811 /packages/backend/src/core/QueryService.ts
parentmerge: Fix error caused by activity type confusion (!1090) (diff)
parentrestore join to note.channel in channel/timeline.ts (diff)
downloadsharkey-dae544b353ebbc6edd85c032786b9901807460e8.tar.gz
sharkey-dae544b353ebbc6edd85c032786b9901807460e8.tar.bz2
sharkey-dae544b353ebbc6edd85c032786b9901807460e8.zip
merge: Rework queries and add indexes to improve timeline performance (!1091)
View MR for information: https://activitypub.software/TransFem-org/Sharkey/-/merge_requests/1091 Approved-by: dakkar <dakkar@thenautilus.net> Approved-by: Marie <github@yuugi.dev>
Diffstat (limited to 'packages/backend/src/core/QueryService.ts')
-rw-r--r--packages/backend/src/core/QueryService.ts468
1 files changed, 303 insertions, 165 deletions
diff --git a/packages/backend/src/core/QueryService.ts b/packages/backend/src/core/QueryService.ts
index cf2419a9eb..bf5b0b359f 100644
--- a/packages/backend/src/core/QueryService.ts
+++ b/packages/backend/src/core/QueryService.ts
@@ -4,13 +4,14 @@
*/
import { Inject, Injectable } from '@nestjs/common';
-import { Brackets, ObjectLiteral } from 'typeorm';
+import { Brackets, Not, WhereExpressionBuilder } from 'typeorm';
import { DI } from '@/di-symbols.js';
import type { MiUser } from '@/models/User.js';
-import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository, RenoteMutingsRepository, MiMeta } from '@/models/_.js';
+import { MiInstance } from '@/models/Instance.js';
+import type { UserProfilesRepository, FollowingsRepository, ChannelFollowingsRepository, BlockingsRepository, NoteThreadMutingsRepository, MutingsRepository, RenoteMutingsRepository, MiMeta, InstancesRepository } from '@/models/_.js';
import { bindThis } from '@/decorators.js';
import { IdService } from '@/core/IdService.js';
-import type { SelectQueryBuilder } from 'typeorm';
+import type { SelectQueryBuilder, ObjectLiteral } from 'typeorm';
@Injectable()
export class QueryService {
@@ -36,6 +37,9 @@ export class QueryService {
@Inject(DI.renoteMutingsRepository)
private renoteMutingsRepository: RenoteMutingsRepository,
+ @Inject(DI.instancesRepository)
+ private readonly instancesRepository: InstancesRepository,
+
@Inject(DI.meta)
private meta: MiMeta,
@@ -72,215 +76,349 @@ export class QueryService {
// ここでいうBlockedは被Blockedの意
@bindThis
- public generateBlockedUserQueryForNotes(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }): void {
- const blockingQuery = this.blockingsRepository.createQueryBuilder('blocking')
- .select('blocking.blockerId')
- .where('blocking.blockeeId = :blockeeId', { blockeeId: me.id });
-
+ public generateBlockedUserQueryForNotes<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }): SelectQueryBuilder<E> {
// 投稿の作者にブロックされていない かつ
// 投稿の返信先の作者にブロックされていない かつ
// 投稿の引用元の作者にブロックされていない
- q
- .andWhere(`note.userId NOT IN (${ blockingQuery.getQuery() })`)
- .andWhere(new Brackets(qb => {
- qb
- .where('note.replyUserId IS NULL')
- .orWhere(`note.replyUserId NOT IN (${ blockingQuery.getQuery() })`);
- }))
- .andWhere(new Brackets(qb => {
- qb
- .where('note.renoteUserId IS NULL')
- .orWhere(`note.renoteUserId NOT IN (${ blockingQuery.getQuery() })`);
- }));
-
- q.setParameters(blockingQuery.getParameters());
+ return this
+ .andNotBlockingUser(q, 'note.userId', ':meId')
+ .andWhere(new Brackets(qb => this
+ .orNotBlockingUser(qb, 'note.replyUserId', ':meId')
+ .orWhere('note.replyUserId IS NULL')))
+ .andWhere(new Brackets(qb => this
+ .orNotBlockingUser(qb, 'note.renoteUserId', ':meId')
+ .orWhere('note.renoteUserId IS NULL')))
+ .setParameters({ meId: me.id });
}
@bindThis
- public generateBlockQueryForUsers(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }): void {
- const blockingQuery = this.blockingsRepository.createQueryBuilder('blocking')
- .select('blocking.blockeeId')
- .where('blocking.blockerId = :blockerId', { blockerId: me.id });
-
- const blockedQuery = this.blockingsRepository.createQueryBuilder('blocking')
- .select('blocking.blockerId')
- .where('blocking.blockeeId = :blockeeId', { blockeeId: me.id });
-
- q.andWhere(`user.id NOT IN (${ blockingQuery.getQuery() })`);
- q.setParameters(blockingQuery.getParameters());
-
- q.andWhere(`user.id NOT IN (${ blockedQuery.getQuery() })`);
- q.setParameters(blockedQuery.getParameters());
+ public generateBlockQueryForUsers<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }): SelectQueryBuilder<E> {
+ this.andNotBlockingUser(q, ':meId', 'user.id');
+ this.andNotBlockingUser(q, 'user.id', ':me.id');
+ return q.setParameters({ meId: me.id });
}
@bindThis
- public generateMutedNoteThreadQuery(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }): void {
- const mutedQuery = this.noteThreadMutingsRepository.createQueryBuilder('threadMuted')
- .select('threadMuted.threadId')
- .where('threadMuted.userId = :userId', { userId: me.id });
-
- q.andWhere(`note.id NOT IN (${ mutedQuery.getQuery() })`);
- q.andWhere(new Brackets(qb => {
- qb
- .where('note.threadId IS NULL')
- .orWhere(`note.threadId NOT IN (${ mutedQuery.getQuery() })`);
- }));
-
- q.setParameters(mutedQuery.getParameters());
+ public generateMutedNoteThreadQuery<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }): SelectQueryBuilder<E> {
+ return this
+ .andNotMutingThread(q, ':meId', 'note.id')
+ .andWhere(new Brackets(qb => this
+ .orNotMutingThread(qb, ':meId', 'note.threadId')
+ .orWhere('note.threadId IS NULL')))
+ .setParameters({ meId: me.id });
}
@bindThis
- public generateMutedUserQueryForNotes(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }, exclude?: { id: MiUser['id'] }): void {
- const mutingQuery = this.mutingsRepository.createQueryBuilder('muting')
- .select('muting.muteeId')
- .where('muting.muterId = :muterId', { muterId: me.id });
-
- if (exclude) {
- mutingQuery.andWhere('muting.muteeId != :excludeId', { excludeId: exclude.id });
- }
-
- const mutingInstanceQuery = this.userProfilesRepository.createQueryBuilder('user_profile')
- .select('user_profile.mutedInstances')
- .where('user_profile.userId = :muterId', { muterId: me.id });
-
+ public generateMutedUserQueryForNotes<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }, exclude?: { id: MiUser['id'] }): SelectQueryBuilder<E> {
// 投稿の作者をミュートしていない かつ
// 投稿の返信先の作者をミュートしていない かつ
// 投稿の引用元の作者をミュートしていない
- q
- .andWhere(`note.userId NOT IN (${ mutingQuery.getQuery() })`)
- .andWhere(new Brackets(qb => {
- qb
- .where('note.replyUserId IS NULL')
- .orWhere(`note.replyUserId NOT IN (${ mutingQuery.getQuery() })`);
- }))
- .andWhere(new Brackets(qb => {
- qb
- .where('note.renoteUserId IS NULL')
- .orWhere(`note.renoteUserId NOT IN (${ mutingQuery.getQuery() })`);
- }))
+ return this
+ .andNotMutingUser(q, ':meId', 'note.userId', exclude)
+ .andWhere(new Brackets(qb => this
+ .orNotMutingUser(qb, ':meId', 'note.replyUserId', exclude)
+ .orWhere('note.replyUserId IS NULL')))
+ .andWhere(new Brackets(qb => this
+ .orNotMutingUser(qb, ':meId', 'note.renoteUserId', exclude)
+ .orWhere('note.renoteUserId IS NULL')))
+ // TODO exclude should also pass a host to skip these instances
// mute instances
- .andWhere(new Brackets(qb => {
- qb
- .andWhere('note.userHost IS NULL')
- .orWhere(`NOT ((${ mutingInstanceQuery.getQuery() })::jsonb ? note.userHost)`);
- }))
- .andWhere(new Brackets(qb => {
- qb
- .where('note.replyUserHost IS NULL')
- .orWhere(`NOT ((${ mutingInstanceQuery.getQuery() })::jsonb ? note.replyUserHost)`);
- }))
- .andWhere(new Brackets(qb => {
- qb
- .where('note.renoteUserHost IS NULL')
- .orWhere(`NOT ((${ mutingInstanceQuery.getQuery() })::jsonb ? note.renoteUserHost)`);
- }));
-
- q.setParameters(mutingQuery.getParameters());
- q.setParameters(mutingInstanceQuery.getParameters());
+ .andWhere(new Brackets(qb => this
+ .andNotMutingInstance(qb, ':meId', 'note.userHost')
+ .orWhere('note.userHost IS NULL')))
+ .andWhere(new Brackets(qb => this
+ .orNotMutingInstance(qb, ':meId', 'note.replyUserHost')
+ .orWhere('note.replyUserHost IS NULL')))
+ .andWhere(new Brackets(qb => this
+ .orNotMutingInstance(qb, ':meId', 'note.renoteUserHost')
+ .orWhere('note.renoteUserHost IS NULL')))
+ .setParameters({ meId: me.id });
}
@bindThis
- public generateMutedUserQueryForUsers(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }): void {
- const mutingQuery = this.mutingsRepository.createQueryBuilder('muting')
- .select('muting.muteeId')
- .where('muting.muterId = :muterId', { muterId: me.id });
-
- q.andWhere(`user.id NOT IN (${ mutingQuery.getQuery() })`);
-
- q.setParameters(mutingQuery.getParameters());
+ public generateMutedUserQueryForUsers<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }): SelectQueryBuilder<E> {
+ return this
+ .andNotMutingUser(q, ':meId', 'user.id')
+ .setParameters({ meId: me.id });
}
+ // This intentionally skips isSuspended, isDeleted, makeNotesFollowersOnlyBefore, makeNotesHiddenBefore, and requireSigninToViewContents.
+ // NoteEntityService checks these automatically and calls hideNote() to hide them without breaking threads.
+ // For moderation purposes, you can set isSilenced to forcibly hide existing posts by a user.
@bindThis
- public generateVisibilityQuery(q: SelectQueryBuilder<any>, me?: { id: MiUser['id'] } | null): void {
+ public generateVisibilityQuery<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me?: { id: MiUser['id'] } | null): SelectQueryBuilder<E> {
// This code must always be synchronized with the checks in Notes.isVisibleForMe.
- if (me == null) {
- q.andWhere(new Brackets(qb => {
- qb
- .where('note.visibility = \'public\'')
- .orWhere('note.visibility = \'home\'');
- }));
- } else {
- const followingQuery = this.followingsRepository.createQueryBuilder('following')
- .select('following.followeeId')
- .where('following.followerId = :meId');
+ return q.andWhere(new Brackets(qb => {
+ // Public post
+ qb.orWhere('note.visibility = \'public\'')
+ .orWhere('note.visibility = \'home\'');
- q.andWhere(new Brackets(qb => {
+ if (me != null) {
qb
- // 公開投稿である
- .where(new Brackets(qb => {
- qb
- .where('note.visibility = \'public\'')
- .orWhere('note.visibility = \'home\'');
- }))
- // または 自分自身
- .orWhere('note.userId = :meId')
- // または 自分宛て
- .orWhere(':meIdAsList <@ note.visibleUserIds')
- .orWhere(new Brackets(qb => {
- qb
- // または フォロワー宛ての投稿であり、
- .where('note.visibility = \'followers\'')
- .andWhere(new Brackets(qb => {
- qb
- // 自分がフォロワーである
- .where(`note.userId IN (${ followingQuery.getQuery() })`)
- // または 自分の投稿へのリプライ
- .orWhere('note.replyUserId = :meId')
- .orWhere(':meIdAsList <@ note.mentions');
- }));
- }));
- }));
+ // My post
+ .orWhere(':meId = note.userId')
+ // Reply to me
+ .orWhere(':meId = note.replyUserId')
+ // DM to me
+ .orWhere(':meId = ANY (note.visibleUserIds)')
+ // Mentions me
+ .orWhere(':meId = ANY (note.mentions)')
+ // Followers-only post
+ .orWhere(new Brackets(qb => this
+ .andFollowingUser(qb, ':meId', 'note.userId')
+ .andWhere('note.visibility = \'followers\'')));
- q.setParameters({ meId: me.id, meIdAsList: [me.id] });
- }
+ q.setParameters({ meId: me.id });
+ }
+ }));
}
@bindThis
- public generateMutedUserRenotesQueryForNotes(q: SelectQueryBuilder<any>, me: { id: MiUser['id'] }): void {
- const mutingQuery = this.renoteMutingsRepository.createQueryBuilder('renote_muting')
- .select('renote_muting.muteeId')
- .where('renote_muting.muterId = :muterId', { muterId: me.id });
-
- q.andWhere(new Brackets(qb => {
- qb
+ public generateMutedUserRenotesQueryForNotes<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me: { id: MiUser['id'] }): SelectQueryBuilder<E> {
+ return q
+ .andWhere(new Brackets(qb => this
+ .orNotMutingRenote(qb, ':meId', 'note.userId')
.orWhere('note.renoteId IS NULL')
.orWhere('note.text IS NOT NULL')
.orWhere('note.cw IS NOT NULL')
.orWhere('note.replyId IS NOT NULL')
- .orWhere('note.hasPoll = false')
- .orWhere('note.fileIds != \'{}\'')
- .orWhere(`note.userId NOT IN (${ mutingQuery.getQuery() })`);
- }));
-
- q.setParameters(mutingQuery.getParameters());
+ .orWhere('note.hasPoll = true')
+ .orWhere('note.fileIds != \'{}\'')))
+ .setParameters({ meId: me.id });
}
@bindThis
- public generateBlockedHostQueryForNote(q: SelectQueryBuilder<any>, excludeAuthor?: boolean, allowSilenced = true): void {
- function checkFor(key: 'user' | 'replyUser' | 'renoteUser') {
- q.leftJoin(`note.${key}Instance`, `${key}Instance`);
- q.andWhere(new Brackets(qb => {
- qb.orWhere(`note.${key}Id IS NULL`) // no corresponding user
- .orWhere(`note.${key}Host IS NULL`); // local
+ public generateExcludedRenotesQueryForNotes<E extends ObjectLiteral>(q: SelectQueryBuilder<E>): SelectQueryBuilder<E> {
+ return q
+ .andWhere(new Brackets(qb => qb
+ .orWhere('note.renoteId IS NULL')
+ .orWhere('note.text IS NOT NULL')
+ .orWhere('note.cw IS NOT NULL')
+ .orWhere('note.replyId IS NOT NULL')
+ .orWhere('note.hasPoll = true')
+ .orWhere('note.fileIds != \'{}\'')));
+ }
- if (allowSilenced) {
- qb.orWhere(`${key}Instance.isBlocked = false`); // not blocked
- } else {
- qb.orWhere(new Brackets(qbb => qbb
- .andWhere(`${key}Instance.isBlocked = false`) // not blocked
- .andWhere(`${key}Instance.isSilenced = false`))); // not silenced
- }
+ @bindThis
+ public generateBlockedHostQueryForNote<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, excludeAuthor?: boolean): SelectQueryBuilder<E> {
+ const checkFor = (key: 'user' | 'replyUser' | 'renoteUser') => this
+ .leftJoinInstance(q, `note.${key}Instance`, `${key}Instance`)
+ .andWhere(new Brackets(qb => {
+ qb
+ .orWhere(`"${key}Instance" IS NULL`) // local
+ .orWhere(`"${key}Instance"."isBlocked" = false`); // not blocked
if (excludeAuthor) {
qb.orWhere(`note.userId = note.${key}Id`); // author
}
}));
- }
if (!excludeAuthor) {
checkFor('user');
}
checkFor('replyUser');
checkFor('renoteUser');
+
+ return q;
+ }
+
+ @bindThis
+ public generateSilencedUserQueryForNotes<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, me?: { id: MiUser['id'] } | null): SelectQueryBuilder<E> {
+ if (!me) {
+ return q.andWhere('user.isSilenced = false');
+ }
+
+ return this
+ .leftJoinInstance(q, 'note.userInstance', 'userInstance')
+ .andWhere(new Brackets(qb => this
+ // case 1: we are following the user
+ .orFollowingUser(qb, ':meId', 'note.userId')
+ // case 2: user not silenced AND instance not silenced
+ .orWhere(new Brackets(qbb => qbb
+ .andWhere(new Brackets(qbbb => qbbb
+ .orWhere('"userInstance"."isSilenced" = false')
+ .orWhere('"userInstance" IS NULL')))
+ .andWhere('user.isSilenced = false')))))
+ .setParameters({ meId: me.id });
+ }
+
+ /**
+ * Left-joins an instance in to the query with a given alias and optional condition.
+ * These calls are de-duplicated - multiple uses of the same alias are skipped.
+ */
+ @bindThis
+ public leftJoinInstance<E extends ObjectLiteral>(q: SelectQueryBuilder<E>, relation: string | typeof MiInstance, alias: string, condition?: string): SelectQueryBuilder<E> {
+ // Skip if it's already joined, otherwise we'll get an error
+ if (!q.expressionMap.joinAttributes.some(j => j.alias.name === alias)) {
+ q.leftJoin(relation, alias, condition);
+ }
+
+ return q;
+ }
+
+ /**
+ * Adds OR condition that followerProp (user ID) is following followeeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orFollowingUser<Q extends WhereExpressionBuilder>(q: Q, followerProp: string, followeeProp: string): Q {
+ return this.addFollowingUser(q, followerProp, followeeProp, 'orWhere');
+ }
+
+ /**
+ * Adds AND condition that followerProp (user ID) is following followeeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andFollowingUser<Q extends WhereExpressionBuilder>(q: Q, followerProp: string, followeeProp: string): Q {
+ return this.addFollowingUser(q, followerProp, followeeProp, 'andWhere');
+ }
+
+ private addFollowingUser<Q extends WhereExpressionBuilder>(q: Q, followerProp: string, followeeProp: string, join: 'andWhere' | 'orWhere'): Q {
+ const followingQuery = this.followingsRepository.createQueryBuilder('following')
+ .select('1')
+ .andWhere(`following.followerId = ${followerProp}`)
+ .andWhere(`following.followeeId = ${followeeProp}`);
+
+ return q[join](`EXISTS (${followingQuery.getQuery()})`, followingQuery.getParameters());
+ };
+
+ /**
+ * Adds OR condition that blockerProp (user ID) is not blocking blockeeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orNotBlockingUser<Q extends WhereExpressionBuilder>(q: Q, blockerProp: string, blockeeProp: string): Q {
+ return this.excludeBlockingUser(q, blockerProp, blockeeProp, 'orWhere');
+ }
+
+ /**
+ * Adds AND condition that blockerProp (user ID) is not blocking blockeeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andNotBlockingUser<Q extends WhereExpressionBuilder>(q: Q, blockerProp: string, blockeeProp: string): Q {
+ return this.excludeBlockingUser(q, blockerProp, blockeeProp, 'andWhere');
+ }
+
+ private excludeBlockingUser<Q extends WhereExpressionBuilder>(q: Q, blockerProp: string, blockeeProp: string, join: 'andWhere' | 'orWhere'): Q {
+ const blockingQuery = this.blockingsRepository.createQueryBuilder('blocking')
+ .select('1')
+ .andWhere(`blocking.blockerId = ${blockerProp}`)
+ .andWhere(`blocking.blockeeId = ${blockeeProp}`);
+
+ return q[join](`NOT EXISTS (${blockingQuery.getQuery()})`, blockingQuery.getParameters());
+ };
+
+ /**
+ * Adds OR condition that muterProp (user ID) is not muting muteeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orNotMutingUser<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, exclude?: { id: MiUser['id'] }): Q {
+ return this.excludeMutingUser(q, muterProp, muteeProp, 'orWhere', exclude);
+ }
+
+ /**
+ * Adds AND condition that muterProp (user ID) is not muting muteeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andNotMutingUser<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, exclude?: { id: MiUser['id'] }): Q {
+ return this.excludeMutingUser(q, muterProp, muteeProp, 'andWhere', exclude);
+ }
+
+ private excludeMutingUser<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, join: 'andWhere' | 'orWhere', exclude?: { id: MiUser['id'] }): Q {
+ const mutingQuery = this.mutingsRepository.createQueryBuilder('muting')
+ .select('1')
+ .andWhere(`muting.muterId = ${muterProp}`)
+ .andWhere(`muting.muteeId = ${muteeProp}`);
+
+ if (exclude) {
+ mutingQuery.andWhere({ muteeId: Not(exclude.id) });
+ }
+
+ return q[join](`NOT EXISTS (${mutingQuery.getQuery()})`, mutingQuery.getParameters());
+ }
+
+ /**
+ * Adds OR condition that muterProp (user ID) is not muting renotes by muteeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orNotMutingRenote<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingRenote(q, muterProp, muteeProp, 'orWhere');
+ }
+
+ /**
+ * Adds AND condition that muterProp (user ID) is not muting renotes by muteeProp (user ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andNotMutingRenote<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingRenote(q, muterProp, muteeProp, 'andWhere');
+ }
+
+ private excludeMutingRenote<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, join: 'andWhere' | 'orWhere'): Q {
+ const mutingQuery = this.renoteMutingsRepository.createQueryBuilder('renote_muting')
+ .select('1')
+ .andWhere(`renote_muting.muterId = ${muterProp}`)
+ .andWhere(`renote_muting.muteeId = ${muteeProp}`);
+
+ return q[join](`NOT EXISTS (${mutingQuery.getQuery()})`, mutingQuery.getParameters());
+ };
+
+ /**
+ * Adds OR condition that muterProp (user ID) is not muting muteeProp (instance host).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orNotMutingInstance<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingInstance(q, muterProp, muteeProp, 'orWhere');
+ }
+
+ /**
+ * Adds AND condition that muterProp (user ID) is not muting muteeProp (instance host).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andNotMutingInstance<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingInstance(q, muterProp, muteeProp, 'andWhere');
+ }
+
+ private excludeMutingInstance<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, join: 'andWhere' | 'orWhere'): Q {
+ const mutingInstanceQuery = this.userProfilesRepository.createQueryBuilder('user_profile')
+ .select('1')
+ .andWhere(`user_profile.userId = ${muterProp}`)
+ .andWhere(`"user_profile"."mutedInstances"::jsonb ? ${muteeProp}`);
+
+ return q[join](`NOT EXISTS (${mutingInstanceQuery.getQuery()})`, mutingInstanceQuery.getParameters());
+ }
+
+ /**
+ * Adds OR condition that muterProp (user ID) is not muting muteeProp (note ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public orNotMutingThread<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingThread(q, muterProp, muteeProp, 'orWhere');
+ }
+
+ /**
+ * Adds AND condition that muterProp (user ID) is not muting muteeProp (note ID).
+ * Both props should be expressions, not raw values.
+ */
+ @bindThis
+ public andNotMutingThread<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string): Q {
+ return this.excludeMutingThread(q, muterProp, muteeProp, 'andWhere');
+ }
+
+ private excludeMutingThread<Q extends WhereExpressionBuilder>(q: Q, muterProp: string, muteeProp: string, join: 'andWhere' | 'orWhere'): Q {
+ const threadMutedQuery = this.noteThreadMutingsRepository.createQueryBuilder('threadMuted')
+ .select('1')
+ .andWhere(`threadMuted.userId = ${muterProp}`)
+ .andWhere(`threadMuted.threadId = ${muteeProp}`);
+
+ return q[join](`NOT EXISTS (${threadMutedQuery.getQuery()})`, threadMutedQuery.getParameters());
}
}