forked from FoundKeyGang/FoundKey
Johann150
7b8333a21f
Instead of throwing an IdentifiableError which then just always gets converted into an ApiError, the getter can just throw the same ApiError directly. This makes it more convenient to use and thus more endpoints have been refactored to use it to reduce code repetition.
113 lines
3.6 KiB
TypeScript
113 lines
3.6 KiB
TypeScript
import { Brackets } from 'typeorm';
|
|
import { Notes } from '@/models/index.js';
|
|
import define from '@/server/api/define.js';
|
|
import { ApiError } from '@/server/api/error.js';
|
|
import { getUser } from '@/server/api/common/getters.js';
|
|
import { makePaginationQuery } from '@/server/api/common/make-pagination-query.js';
|
|
import { visibilityQuery } from '@/server/api/common/generate-visibility-query.js';
|
|
import { generateMutedUserQuery } from '@/server/api/common/generate-muted-user-query.js';
|
|
import { generateBlockedUserQuery } from '@/server/api/common/generate-block-query.js';
|
|
|
|
export const meta = {
|
|
tags: ['users', 'notes'],
|
|
|
|
description: 'Show all notes that this user created.',
|
|
|
|
res: {
|
|
type: 'array',
|
|
optional: false, nullable: false,
|
|
items: {
|
|
type: 'object',
|
|
optional: false, nullable: false,
|
|
ref: 'Note',
|
|
},
|
|
},
|
|
|
|
errors: ['NO_SUCH_USER'],
|
|
} as const;
|
|
|
|
export const paramDef = {
|
|
type: 'object',
|
|
properties: {
|
|
userId: { type: 'string', format: 'misskey:id' },
|
|
includeReplies: { type: 'boolean', default: true },
|
|
limit: { type: 'integer', minimum: 1, maximum: 100, default: 10 },
|
|
sinceId: { type: 'string', format: 'misskey:id' },
|
|
untilId: { type: 'string', format: 'misskey:id' },
|
|
sinceDate: { type: 'integer' },
|
|
untilDate: { type: 'integer' },
|
|
includeMyRenotes: { type: 'boolean', default: true },
|
|
withFiles: { type: 'boolean', default: false },
|
|
fileType: { type: 'array', items: {
|
|
type: 'string',
|
|
} },
|
|
excludeNsfw: { type: 'boolean', default: false },
|
|
},
|
|
required: ['userId'],
|
|
} as const;
|
|
|
|
// eslint-disable-next-line import/no-default-export
|
|
export default define(meta, paramDef, async (ps, me) => {
|
|
// Lookup user
|
|
const user = await getUser(ps.userId);
|
|
|
|
//#region Construct query
|
|
const query = makePaginationQuery(Notes.createQueryBuilder('note'), ps.sinceId, ps.untilId, ps.sinceDate, ps.untilDate)
|
|
.andWhere('note.userId = :userId', { userId: user.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');
|
|
|
|
if (me) {
|
|
generateMutedUserQuery(query, me);
|
|
generateBlockedUserQuery(query, me);
|
|
}
|
|
|
|
if (ps.withFiles) {
|
|
query.andWhere('note.fileIds != \'{}\'');
|
|
}
|
|
|
|
if (ps.fileType != null) {
|
|
query.andWhere('note.fileIds != \'{}\'');
|
|
query.andWhere(new Brackets(qb => {
|
|
for (const type of ps.fileType!) {
|
|
const i = ps.fileType!.indexOf(type);
|
|
qb.orWhere(`:type${i} = ANY(note.attachedFileTypes)`, { [`type${i}`]: type });
|
|
}
|
|
}));
|
|
|
|
if (ps.excludeNsfw) {
|
|
query.andWhere('note.cw IS NULL');
|
|
query.andWhere('0 = (SELECT COUNT(*) FROM drive_file df WHERE df.id = ANY(note."fileIds") AND df."isSensitive")');
|
|
}
|
|
}
|
|
|
|
if (!ps.includeReplies) {
|
|
query.andWhere('note.replyId IS NULL');
|
|
}
|
|
|
|
if (ps.includeMyRenotes === false) {
|
|
query.andWhere(new Brackets(qb => {
|
|
qb.orWhere('note.userId != :userId', { userId: user.id });
|
|
qb.orWhere('note.renoteId IS NULL');
|
|
qb.orWhere('note.text IS NOT NULL');
|
|
qb.orWhere('note.fileIds != \'{}\'');
|
|
qb.orWhere('0 < (SELECT COUNT(*) FROM poll WHERE poll."noteId" = note.id)');
|
|
}));
|
|
}
|
|
|
|
//#endregion
|
|
|
|
const timeline = await visibilityQuery(query, me).take(ps.limit).getMany();
|
|
|
|
return await Notes.packMany(timeline, me);
|
|
});
|