import define from "../../define.js"; import { ClipNotes, Clips, Notes } from "@/models/index.js"; import { makePaginationQuery } from "../../common/make-pagination-query.js"; import { generateVisibilityQuery } from "../../common/generate-visibility-query.js"; import { generateMutedUserQuery } from "../../common/generate-muted-user-query.js"; import { ApiError } from "../../error.js"; import { generateBlockedUserQuery } from "../../common/generate-block-query.js"; export const meta = { tags: ["account", "notes", "clips"], requireCredential: false, requireCredentialPrivateMode: true, kind: "read:account", errors: { noSuchClip: { message: "No such clip.", code: "NO_SUCH_CLIP", id: "1d7645e6-2b6d-4635-b0fe-fe22b0e72e00", }, }, res: { type: "array", optional: false, nullable: false, items: { type: "object", optional: false, nullable: false, ref: "Note", }, }, } as const; export const paramDef = { type: "object", properties: { clipId: { type: "string", format: "misskey:id" }, limit: { type: "integer", minimum: 1, maximum: 100, default: 10 }, sinceId: { type: "string", format: "misskey:id" }, untilId: { type: "string", format: "misskey:id" }, }, required: ["clipId"], } as const; export default define(meta, paramDef, async (ps, user) => { const clip = await Clips.findOneBy({ id: ps.clipId, }); if (clip == null) { throw new ApiError(meta.errors.noSuchClip); } if (!clip.isPublic && (user == null || clip.userId !== user.id)) { throw new ApiError(meta.errors.noSuchClip); } const query = makePaginationQuery( Notes.createQueryBuilder("note"), ps.sinceId, ps.untilId, ) .innerJoin( ClipNotes.metadata.targetName, "clipNote", "clipNote.noteId = note.id", ) .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") .andWhere("clipNote.clipId = :clipId", { clipId: clip.id }); if (user) { generateVisibilityQuery(query, user); generateMutedUserQuery(query, user); generateBlockedUserQuery(query, user); } const notes = await query.take(ps.limit).getMany(); return await Notes.packMany(notes, user); });