forked from mirrors/iceshrimp
Laura Hausmann
302b112f05
This drastically improves timeline performance due to the many (2-6 per query) database joins that are now no longer required
89 lines
2.2 KiB
TypeScript
89 lines
2.2 KiB
TypeScript
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("note.reply", "reply")
|
|
.leftJoinAndSelect("note.renote", "renote")
|
|
.leftJoinAndSelect("reply.user", "replyUser")
|
|
.leftJoinAndSelect("renote.user", "renoteUser")
|
|
.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);
|
|
});
|