paricafe/src/server/api/endpoints/users/recommendation.ts

125 lines
2.8 KiB
TypeScript
Raw Normal View History

import * as ms from 'ms';
2017-03-08 12:50:09 -06:00
import $ from 'cafy';
import User, { pack, ILocalUser } from '../../../../models/user';
2018-04-18 22:43:25 -05:00
import { getFriendIds } from '../../common/get-friends';
import * as request from 'request-promise-native';
2018-10-09 12:08:26 -05:00
import config from '../../../../config';
2018-11-01 23:47:44 -05:00
import define from '../../define';
import fetchMeta from '../../../../misc/fetch-meta';
import resolveUser from '../../../../remote/resolve-user';
2019-01-31 18:57:51 -06:00
import { getHideUserIds } from '../../common/get-hide-users';
import { apiLogger } from '../../logger';
2018-07-16 14:36:44 -05:00
export const meta = {
desc: {
2018-08-28 16:59:43 -05:00
'ja-JP': 'おすすめのユーザー一覧を取得します。'
2018-07-16 14:36:44 -05:00
},
tags: ['users'],
2018-07-16 14:36:44 -05:00
requireCredential: true,
2018-11-01 22:49:08 -05:00
kind: 'account-read',
params: {
limit: {
2019-02-13 01:33:07 -06:00
validator: $.optional.num.range(1, 100),
2018-11-01 22:49:08 -05:00
default: 10
},
offset: {
2019-02-13 01:33:07 -06:00
validator: $.optional.num.min(0),
2018-11-01 22:49:08 -05:00
default: 0
}
2019-02-24 04:42:26 -06:00
},
res: {
type: 'array',
items: {
type: 'User',
}
},
2018-07-16 14:36:44 -05:00
};
export default define(meta, async (ps, me) => {
const instance = await fetchMeta();
if (instance.enableExternalUserRecommendation) {
2018-10-09 12:08:26 -05:00
const userName = me.username;
const hostName = config.hostname;
2018-11-01 23:47:44 -05:00
const limit = ps.limit;
const offset = ps.offset;
const timeout = instance.externalUserRecommendationTimeout;
const engine = instance.externalUserRecommendationEngine;
2018-10-06 03:21:27 -05:00
const url = engine
.replace('{{host}}', hostName)
.replace('{{user}}', userName)
2018-11-01 23:47:44 -05:00
.replace('{{limit}}', limit.toString())
.replace('{{offset}}', offset.toString());
2018-10-09 12:08:26 -05:00
const users = await request({
2018-10-12 23:13:15 -05:00
url: url,
proxy: config.proxy,
2018-10-12 23:13:15 -05:00
timeout: timeout,
json: true,
followRedirect: true,
followAllRedirects: true
})
.then(body => convertUsers(body, me));
return users;
2018-10-06 02:03:18 -05:00
} else {
// ID list of the user itself and other users who the user follows
const followingIds = await getFriendIds(me._id);
// 隠すユーザーを取得
const hideUserIds = await getHideUserIds(me);
const users = await User.find({
_id: {
$nin: followingIds.concat(hideUserIds)
},
isLocked: { $ne: true },
updatedAt: {
$gte: new Date(Date.now() - ms('7days'))
},
host: null
}, {
limit: ps.limit,
skip: ps.offset,
sort: {
followersCount: -1
}
});
return await Promise.all(users.map(user => pack(user, me, { detail: true })));
2018-10-06 02:03:18 -05:00
}
});
type IRecommendUser = {
name: string;
username: string;
host: string;
description: string;
avatarUrl: string;
};
/**
* Resolve/Pack dummy users
*/
async function convertUsers(src: IRecommendUser[], me: ILocalUser) {
const packed = await Promise.all(src.map(async x => {
const user = await resolveUser(x.username, x.host)
.catch(() => {
apiLogger.warn(`Can't resolve ${x.username}@${x.host}`);
return null;
});
if (user == null) return x;
return await pack(user, me, { detail: true });
}));
return packed;
}