import $ from 'cafy'; import ID, { transform } from '../../../../misc/cafy-id'; import Note from '../../../../models/note'; import { packMany } from '../../../../models/note'; import define from '../../define'; import { countIf } from '../../../../prelude/array'; import fetchMeta from '../../../../misc/fetch-meta'; import activeUsersChart from '../../../../services/chart/active-users'; import { getHideUserIds } from '../../common/get-hide-users'; export const meta = { desc: { 'ja-JP': 'ローカルタイムラインを取得します。' }, params: { withFiles: { validator: $.optional.bool, desc: { 'ja-JP': 'ファイルが添付された投稿に限定するか否か' } }, mediaOnly: { validator: $.optional.bool, desc: { 'ja-JP': 'ファイルが添付された投稿に限定するか否か (このパラメータは廃止予定です。代わりに withFiles を使ってください。)' } }, fileType: { validator: $.optional.arr($.str), desc: { 'ja-JP': '指定された種類のファイルが添付された投稿のみを取得します' } }, excludeNsfw: { validator: $.optional.bool, default: false, desc: { 'ja-JP': 'true にすると、NSFW指定されたファイルを除外します(fileTypeが指定されている場合のみ有効)' } }, limit: { validator: $.optional.num.range(1, 100), default: 10 }, sinceId: { validator: $.optional.type(ID), transform: transform, }, untilId: { validator: $.optional.type(ID), transform: transform, }, sinceDate: { validator: $.optional.num, }, untilDate: { validator: $.optional.num, }, } }; export default define(meta, (ps, user) => new Promise(async (res, rej) => { const meta = await fetchMeta(); if (meta.disableLocalTimeline) { if (user == null || (!user.isAdmin && !user.isModerator)) { return rej('local timeline disabled'); } } // Check if only one of sinceId, untilId, sinceDate, untilDate specified if (countIf(x => x != null, [ps.sinceId, ps.untilId, ps.sinceDate, ps.untilDate]) > 1) { return rej('only one of sinceId, untilId, sinceDate, untilDate can be specified'); } // 隠すユーザーを取得 const hideUserIds = await getHideUserIds(user); //#region Construct query const sort = { _id: -1 }; const query = { deletedAt: null, // public only visibility: 'public', // リプライでない //replyId: null, // local '_user.host': null } as any; if (hideUserIds && hideUserIds.length > 0) { query.userId = { $nin: hideUserIds }; query['_reply.userId'] = { $nin: hideUserIds }; query['_renote.userId'] = { $nin: hideUserIds }; } const withFiles = ps.withFiles != null ? ps.withFiles : ps.mediaOnly; if (withFiles) { query.fileIds = { $exists: true, $ne: [] }; } if (ps.fileType) { query.fileIds = { $exists: true, $ne: [] }; query['_files.contentType'] = { $in: ps.fileType }; if (ps.excludeNsfw) { query['_files.metadata.isSensitive'] = { $ne: true }; } } if (ps.sinceId) { sort._id = 1; query._id = { $gt: ps.sinceId }; } else if (ps.untilId) { query._id = { $lt: ps.untilId }; } else if (ps.sinceDate) { sort._id = 1; query.createdAt = { $gt: new Date(ps.sinceDate) }; } else if (ps.untilDate) { query.createdAt = { $lt: new Date(ps.untilDate) }; } //#endregion const timeline = await Note .find(query, { limit: ps.limit, sort: sort }); res(await packMany(timeline, user)); if (user) { activeUsersChart.update(user); } }));