import $ from 'cafy'; import ID, { transform } from '../../../../misc/cafy-id'; import Note from '../../../../models/note'; import User, { pack } from '../../../../models/user'; import define from '../../define'; import { maximum } from '../../../../prelude/array'; export const meta = { requireCredential: false, params: { userId: { validator: $.type(ID), transform: transform, desc: { 'ja-JP': '対象のユーザーのID', 'en-US': 'Target user ID' } }, limit: { validator: $.num.optional.range(1, 100), default: 10 }, } }; export default define(meta, (ps, me) => new Promise(async (res, rej) => { // Lookup user const user = await User.findOne({ _id: ps.userId }, { fields: { _id: true } }); if (user === null) { return rej('user not found'); } // Fetch recent notes const recentNotes = await Note.find({ userId: user._id, replyId: { $exists: true, $ne: null } }, { sort: { _id: -1 }, limit: 1000, fields: { _id: false, replyId: true } }); // 投稿が少なかったら中断 if (recentNotes.length === 0) { return res([]); } const replyTargetNotes = await Note.find({ _id: { $in: recentNotes.map(p => p.replyId) }, userId: { $ne: user._id } }, { fields: { _id: false, userId: true } }); const repliedUsers: any = {}; // Extract replies from recent notes for (const userId of replyTargetNotes.map(x => x.userId.toString())) { if (repliedUsers[userId]) { repliedUsers[userId]++; } else { repliedUsers[userId] = 1; } } // Calc peak const peak = maximum(Object.values(repliedUsers)); // Sort replies by frequency const repliedUsersSorted = Object.keys(repliedUsers).sort((a, b) => repliedUsers[b] - repliedUsers[a]); // Extract top replied users const topRepliedUsers = repliedUsersSorted.slice(0, ps.limit); // Make replies object (includes weights) const repliesObj = await Promise.all(topRepliedUsers.map(async (user) => ({ user: await pack(user, me, { detail: true }), weight: repliedUsers[user] / peak }))); res(repliesObj); }));