import $ from 'cafy'; import ID, { transform } from '../../../../misc/cafy-id'; import Note from '../../../../models/note'; import { getFriends } from '../../common/get-friends'; import { packMany } from '../../../../models/note'; import define from '../../define'; import fetchMeta from '../../../../misc/fetch-meta'; import activeUsersChart from '../../../../services/chart/active-users'; import { getHideUserIds } from '../../common/get-hide-users'; import { ApiError } from '../../error'; export const meta = { desc: { 'ja-JP': 'ハイブリッドタイムラインを取得します。' }, tags: ['notes'], params: { limit: { validator: $.optional.num.range(1, 100), default: 10, desc: { 'ja-JP': '最大数' } }, sinceId: { validator: $.optional.type(ID), transform: transform, desc: { 'ja-JP': '指定すると、この投稿を基点としてより新しい投稿を取得します' } }, untilId: { validator: $.optional.type(ID), transform: transform, desc: { 'ja-JP': '指定すると、この投稿を基点としてより古い投稿を取得します' } }, sinceDate: { validator: $.optional.num, desc: { 'ja-JP': '指定した時間を基点としてより新しい投稿を取得します。数値は、1970年1月1日 00:00:00 UTC から指定した日時までの経過時間をミリ秒単位で表します。' } }, untilDate: { validator: $.optional.num, desc: { 'ja-JP': '指定した時間を基点としてより古い投稿を取得します。数値は、1970年1月1日 00:00:00 UTC から指定した日時までの経過時間をミリ秒単位で表します。' } }, includeMyRenotes: { validator: $.optional.bool, default: true, desc: { 'ja-JP': '自分の行ったRenoteを含めるかどうか' } }, includeRenotedMyNotes: { validator: $.optional.bool, default: true, desc: { 'ja-JP': 'Renoteされた自分の投稿を含めるかどうか' } }, includeLocalRenotes: { validator: $.optional.bool, default: true, desc: { 'ja-JP': 'Renoteされたローカルの投稿を含めるかどうか' } }, withFiles: { validator: $.optional.bool, desc: { 'ja-JP': 'true にすると、ファイルが添付された投稿だけ取得します' } }, mediaOnly: { validator: $.optional.bool, desc: { 'ja-JP': 'true にすると、ファイルが添付された投稿だけ取得します (このパラメータは廃止予定です。代わりに withFiles を使ってください。)' } }, }, res: { type: 'array', items: { type: 'Note', }, }, errors: { stlDisabled: { message: 'Social timeline has been disabled.', code: 'STL_DISABLED', id: '620763f4-f621-4533-ab33-0577a1a3c342' }, } }; export default define(meta, async (ps, user) => { const m = await fetchMeta(); if (m.disableLocalTimeline && !user.isAdmin && !user.isModerator) { throw new ApiError(meta.errors.stlDisabled); } const [followings, hideUserIds] = await Promise.all([ // フォローを取得 // Fetch following getFriends(user._id, true, false), // 隠すユーザーを取得 getHideUserIds(user) ]); //#region Construct query const sort = { _id: -1 }; const followQuery = followings.map(f => ({ userId: f.id, /*// リプライは含めない(ただし投稿者自身の投稿へのリプライ、自分の投稿へのリプライ、自分のリプライは含める) $or: [{ // リプライでない replyId: null }, { // または // リプライだが返信先が投稿者自身の投稿 $expr: { $eq: ['$_reply.userId', '$userId'] } }, { // または // リプライだが返信先が自分(フォロワー)の投稿 '_reply.userId': user._id }, { // または // 自分(フォロワー)が送信したリプライ userId: user._id }]*/ })); const visibleQuery = user == null ? [{ visibility: { $in: ['public', 'home'] } }] : [{ visibility: { $in: ['public', 'home', 'followers'] } }, { // myself (for specified/private) userId: user._id }, { // to me (for specified) visibleUserIds: { $in: [ user._id ] } }]; const query = { $and: [{ deletedAt: null, $or: [{ $and: [{ // フォローしている人の投稿 $or: followQuery }, { // visible for me $or: visibleQuery }] }, { // public only visibility: 'public', // リプライでない //replyId: null, // local '_user.host': null }], // hide userId: { $nin: hideUserIds }, '_reply.userId': { $nin: hideUserIds }, '_renote.userId': { $nin: hideUserIds }, }] } as any; // MongoDBではトップレベルで否定ができないため、De Morganの法則を利用してクエリします。 // つまり、「『自分の投稿かつRenote』ではない」を「『自分の投稿ではない』または『Renoteではない』」と表現します。 // for details: https://en.wikipedia.org/wiki/De_Morgan%27s_laws if (ps.includeMyRenotes === false) { query.$and.push({ $or: [{ userId: { $ne: user._id } }, { renoteId: null }, { text: { $ne: null } }, { fileIds: { $ne: [] } }, { poll: { $ne: null } }] }); } if (ps.includeRenotedMyNotes === false) { query.$and.push({ $or: [{ '_renote.userId': { $ne: user._id } }, { renoteId: null }, { text: { $ne: null } }, { fileIds: { $ne: [] } }, { poll: { $ne: null } }] }); } if (ps.includeLocalRenotes === false) { query.$and.push({ $or: [{ '_renote.user.host': { $ne: null } }, { renoteId: null }, { text: { $ne: null } }, { fileIds: { $ne: [] } }, { poll: { $ne: null } }] }); } if (ps.withFiles || ps.mediaOnly) { query.$and.push({ fileIds: { $exists: true, $ne: [] } }); } 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 }); activeUsersChart.update(user); return await packMany(timeline, user); });