import { Brackets } from "typeorm"; import read from "@/services/note/read.js"; import { Notes, Followings } from "@/models/index.js"; import define from "../../define.js"; import { generateVisibilityQuery } from "../../common/generate-visibility-query.js"; import { generateMutedUserQuery } from "../../common/generate-muted-user-query.js"; import { makePaginationQuery } from "../../common/make-pagination-query.js"; import { generateBlockedUserQuery } from "../../common/generate-block-query.js"; import { generateMutedNoteThreadQuery } from "../../common/generate-muted-note-thread-query.js"; export const meta = { tags: ["notes"], requireCredential: true, res: { type: "array", optional: false, nullable: false, items: { type: "object", optional: false, nullable: false, ref: "Note", }, }, } as const; export const paramDef = { type: "object", properties: { following: { type: "boolean", default: false }, limit: { type: "integer", minimum: 1, maximum: 100, default: 10 }, sinceId: { type: "string", format: "misskey:id" }, untilId: { type: "string", format: "misskey:id" }, visibility: { type: "string" }, }, required: [], } as const; export default define(meta, paramDef, async (ps, user) => { const followingQuery = Followings.createQueryBuilder("following") .select("following.followeeId") .where("following.followerId = :followerId", { followerId: user.id }); const query = makePaginationQuery( Notes.createQueryBuilder("note"), ps.sinceId, ps.untilId, ) .andWhere( new Brackets((qb) => { qb.where(`'{"${user.id}"}' <@ note.mentions`).orWhere( `'{"${user.id}"}' <@ note.visibleUserIds`, ); }), ) .innerJoinAndSelect("note.user", "user") .leftJoinAndSelect("user.avatar", "avatar") .leftJoinAndSelect("user.banner", "banner") .leftJoinAndSelect("note.reply", "reply") .leftJoinAndSelect("note.renote", "renote") .leftJoinAndSelect("reply.user", "replyUser") .leftJoinAndSelect("replyUser.avatar", "replyUserAvatar") .leftJoinAndSelect("replyUser.banner", "replyUserBanner") .leftJoinAndSelect("renote.user", "renoteUser") .leftJoinAndSelect("renoteUser.avatar", "renoteUserAvatar") .leftJoinAndSelect("renoteUser.banner", "renoteUserBanner"); generateVisibilityQuery(query, user); generateMutedUserQuery(query, user); generateMutedNoteThreadQuery(query, user); generateBlockedUserQuery(query, user); if (ps.visibility) { query.andWhere("note.visibility = :visibility", { visibility: ps.visibility, }); } if (ps.following) { query.andWhere( `((note.userId IN (${followingQuery.getQuery()})) OR (note.userId = :meId))`, { meId: user.id }, ); query.setParameters(followingQuery.getParameters()); } // We fetch more than requested because some may be filtered out, and if there's less than // requested, the pagination stops. const found = []; const take = Math.floor(ps.limit * 1.5); let skip = 0; while (found.length < ps.limit) { const notes = await query.take(take).skip(skip).getMany(); found.push(...await Notes.packMany(notes, user)) skip += take; if (notes.length < take) break; } if (found.length > ps.limit) { found.length = ps.limit; } read(user.id, found); return found; });