perf: add blockee cache

This commit is contained in:
Namekuji 2023-08-06 00:40:55 -04:00
parent a9d0d61d59
commit f4d8f82239
No known key found for this signature in database
GPG key ID: 1D62332C07FBA532
7 changed files with 52 additions and 13 deletions

View file

@ -9,6 +9,7 @@ import {
ChannelFollowingsCache, ChannelFollowingsCache,
InstanceMutingsCache, InstanceMutingsCache,
LocalFollowingsCache, LocalFollowingsCache,
UserBlockedCache,
UserMutingsCache, UserMutingsCache,
userWordMuteCache, userWordMuteCache,
} from "@/misc/cache.js"; } from "@/misc/cache.js";
@ -466,3 +467,18 @@ export async function filterMutedNote(
return notes.filter((note) => !getWordHardMute(note, user, mutedWords)); return notes.filter((note) => !getWordHardMute(note, user, mutedWords));
} }
export async function filterBlockedUser(
notes: ScyllaNote[],
user: { id: User["id"] },
): Promise<ScyllaNote[]> {
const cache = await UserBlockedCache.init(user.id);
const blockerIds = await cache.getAll();
return notes.filter(
(note) =>
!blockerIds.includes(note.userId) &&
!(note.replyUserId && blockerIds.includes(note.replyUserId)) &&
!(note.renoteUserId && blockerIds.includes(note.renoteUserId)),
);
}

View file

@ -2,6 +2,7 @@ import { redisClient } from "@/db/redis.js";
import { encode, decode } from "msgpackr"; import { encode, decode } from "msgpackr";
import { ChainableCommander } from "ioredis"; import { ChainableCommander } from "ioredis";
import { import {
Blockings,
ChannelFollowings, ChannelFollowings,
Followings, Followings,
MutedNotes, MutedNotes,
@ -380,4 +381,23 @@ export class InstanceMutingsCache extends SetCache {
} }
} }
export class UserBlockedCache extends SetCache {
private constructor(userId: string) {
const fetcher = () =>
Blockings.find({
select: ["blockerId"],
where: { blockeeId: userId },
}).then((blocks) => blocks.map(({ blockerId }) => blockerId));
super("blocked", userId, fetcher);
}
public static async init(userId: string): Promise<UserBlockedCache> {
const cache = new UserBlockedCache(userId);
await cache.fetch();
return cache;
}
}
export const userWordMuteCache = new Cache<string[][]>("mutedWord", 60 * 30); export const userWordMuteCache = new Cache<string[][]>("mutedWord", 60 * 30);

View file

@ -4,6 +4,7 @@ import { ApiError } from "../../error.js";
import { getUser } from "../../common/getters.js"; import { getUser } from "../../common/getters.js";
import { Blockings, NoteWatchings, Users } from "@/models/index.js"; import { Blockings, NoteWatchings, Users } from "@/models/index.js";
import { HOUR } from "@/const.js"; import { HOUR } from "@/const.js";
import { UserBlockedCache } from "@/misc/cache.js";
export const meta = { export const meta = {
tags: ["account"], tags: ["account"],
@ -69,12 +70,8 @@ export default define(meta, paramDef, async (ps, user) => {
}); });
// Check if already blocking // Check if already blocking
const exist = await Blockings.exist({ const cache = await UserBlockedCache.init(blockee.id);
where: { const exist = await cache.has(blocker.id);
blockerId: blocker.id,
blockeeId: blockee.id,
},
});
if (exist) { if (exist) {
throw new ApiError(meta.errors.alreadyBlocking); throw new ApiError(meta.errors.alreadyBlocking);

View file

@ -4,6 +4,7 @@ import { ApiError } from "../../error.js";
import { getUser } from "../../common/getters.js"; import { getUser } from "../../common/getters.js";
import { Blockings, Users } from "@/models/index.js"; import { Blockings, Users } from "@/models/index.js";
import { HOUR } from "@/const.js"; import { HOUR } from "@/const.js";
import { UserBlockedCache } from "@/misc/cache.js";
export const meta = { export const meta = {
tags: ["account"], tags: ["account"],
@ -69,12 +70,8 @@ export default define(meta, paramDef, async (ps, user) => {
}); });
// Check not blocking // Check not blocking
const exist = await Blockings.exist({ const cache = await UserBlockedCache.init(blockee.id);
where: { const exist = await cache.has(blocker.id);
blockerId: blocker.id,
blockeeId: blockee.id,
},
});
if (!exist) { if (!exist) {
throw new ApiError(meta.errors.notBlocking); throw new ApiError(meta.errors.notBlocking);

View file

@ -20,6 +20,7 @@ import {
execTimelineQuery, execTimelineQuery,
filterMutedUser, filterMutedUser,
filterMutedNote, filterMutedNote,
filterBlockedUser,
} from "@/db/scylla.js"; } from "@/db/scylla.js";
import { ChannelFollowingsCache, LocalFollowingsCache } from "@/misc/cache.js"; import { ChannelFollowingsCache, LocalFollowingsCache } from "@/misc/cache.js";
@ -90,6 +91,7 @@ export default define(meta, paramDef, async (ps, user) => {
filtered = await filterVisibility(filtered, user, followingUserIds); filtered = await filterVisibility(filtered, user, followingUserIds);
filtered = await filterMutedUser(filtered, user); filtered = await filterMutedUser(filtered, user);
filtered = await filterMutedNote(filtered, user); filtered = await filterMutedNote(filtered, user);
filtered = await filterBlockedUser(filtered, user);
return filtered; return filtered;
}; };

View file

@ -20,6 +20,7 @@ import { genId } from "@/misc/gen-id.js";
import { IdentifiableError } from "@/misc/identifiable-error.js"; import { IdentifiableError } from "@/misc/identifiable-error.js";
import { getActiveWebhooks } from "@/misc/webhook-cache.js"; import { getActiveWebhooks } from "@/misc/webhook-cache.js";
import { webhookDeliver } from "@/queue/index.js"; import { webhookDeliver } from "@/queue/index.js";
import { UserBlockedCache } from "@/misc/cache";
export default async function (blocker: User, blockee: User) { export default async function (blocker: User, blockee: User) {
await Promise.all([ await Promise.all([
@ -41,6 +42,9 @@ export default async function (blocker: User, blockee: User) {
await Blockings.insert(blocking); await Blockings.insert(blocking);
const cache = await UserBlockedCache.init(blockee.id);
await cache.add(blocker.id);
if (Users.isLocalUser(blocker) && Users.isRemoteUser(blockee)) { if (Users.isLocalUser(blocker) && Users.isRemoteUser(blockee)) {
const content = renderActivity(renderBlock(blocking)); const content = renderActivity(renderBlock(blocking));
deliver(blocker, content, blockee.inbox); deliver(blocker, content, blockee.inbox);

View file

@ -4,8 +4,8 @@ import renderUndo from "@/remote/activitypub/renderer/undo.js";
import { deliver } from "@/queue/index.js"; import { deliver } from "@/queue/index.js";
import Logger from "../logger.js"; import Logger from "../logger.js";
import type { CacheableUser } from "@/models/entities/user.js"; import type { CacheableUser } from "@/models/entities/user.js";
import { User } from "@/models/entities/user.js";
import { Blockings, Users } from "@/models/index.js"; import { Blockings, Users } from "@/models/index.js";
import { UserBlockedCache } from "@/misc/cache.js";
const logger = new Logger("blocking/delete"); const logger = new Logger("blocking/delete");
@ -15,6 +15,9 @@ export default async function (blocker: CacheableUser, blockee: CacheableUser) {
blockeeId: blockee.id, blockeeId: blockee.id,
}); });
const cache = await UserBlockedCache.init(blockee.id);
await cache.delete(blocker.id);
if (blocking == null) { if (blocking == null) {
logger.warn( logger.warn(
"ブロック解除がリクエストされましたがブロックしていませんでした", "ブロック解除がリクエストされましたがブロックしていませんでした",