import { IsNull, Not } from "typeorm"; import { Users, Followings } from "@/models/index.js"; import type { ILocalUser, IRemoteUser, User } from "@/models/entities/user.js"; import { deliver } from "@/queue/index.js"; import { skippedInstances } from "@/misc/skipped-instances.js"; //#region types interface IRecipe { type: string; } interface IFollowersRecipe extends IRecipe { type: "Followers"; } interface IDirectRecipe extends IRecipe { type: "Direct"; to: IRemoteUser; } const isFollowers = (recipe: any): recipe is IFollowersRecipe => recipe.type === "Followers"; const isDirect = (recipe: any): recipe is IDirectRecipe => recipe.type === "Direct"; //#endregion export default class DeliverManager { private actor: { id: User["id"]; host: null }; private activity: any; private recipes: IRecipe[] = []; /** * Constructor * @param actor Actor * @param activity Activity to deliver */ constructor(actor: { id: User["id"]; host: null }, activity: any) { this.actor = actor; this.activity = activity; } /** * Add recipe for followers deliver */ public addFollowersRecipe() { const deliver = { type: "Followers", } as IFollowersRecipe; this.addRecipe(deliver); } /** * Add recipe for direct deliver * @param to To */ public addDirectRecipe(to: IRemoteUser) { const recipe = { type: "Direct", to, } as IDirectRecipe; this.addRecipe(recipe); } /** * Add recipe * @param recipe Recipe */ public addRecipe(recipe: IRecipe) { this.recipes.push(recipe); } /** * Execute delivers */ public async execute() { if (!Users.isLocalUser(this.actor)) return; const inboxes = new Set(); /* build inbox list Process follower recipes first to avoid duplication when processing direct recipes later. */ if (this.recipes.some((r) => isFollowers(r))) { // followers deliver // TODO: SELECT DISTINCT ON ("followerSharedInbox") "followerSharedInbox" みたいな問い合わせにすればよりパフォーマンス向上できそう // ただ、sharedInboxがnullなリモートユーザーも稀におり、その対応ができなさそう? const followers = (await Followings.find({ where: { followeeId: this.actor.id, followerHost: Not(IsNull()), }, select: { followerSharedInbox: true, followerInbox: true, }, })) as { followerSharedInbox: string | null; followerInbox: string; }[]; for (const following of followers) { const inbox = following.followerSharedInbox || following.followerInbox; inboxes.add(inbox); } } this.recipes .filter( (recipe): recipe is IDirectRecipe => // followers recipes have already been processed isDirect(recipe) && // check that shared inbox has not been added yet !(recipe.to.sharedInbox && inboxes.has(recipe.to.sharedInbox)) && // check that they actually have an inbox recipe.to.inbox != null, ) .forEach((recipe) => inboxes.add(recipe.to.inbox!)); const instancesToSkip = await skippedInstances( // get (unique) list of hosts Array.from( new Set(Array.from(inboxes).map((inbox) => new URL(inbox).host)), ), ); // deliver for (const inbox of inboxes) { // skip instances as indicated if (instancesToSkip.includes(new URL(inbox).host)) continue; deliver(this.actor, this.activity, inbox); } } } //#region Utilities /** * Deliver activity to followers * @param activity Activity * @param from Followee */ export async function deliverToFollowers( actor: { id: ILocalUser["id"]; host: null }, activity: any, ) { const manager = new DeliverManager(actor, activity); manager.addFollowersRecipe(); await manager.execute(); } /** * Deliver activity to user * @param activity Activity * @param to Target user */ export async function deliverToUser( actor: { id: ILocalUser["id"]; host: null }, activity: any, to: IRemoteUser, ) { const manager = new DeliverManager(actor, activity); manager.addDirectRecipe(to); await manager.execute(); } //#endregion