2018-12-16 01:44:59 +09:00
|
|
|
import { ObjectID } from 'mongodb';
|
2018-08-21 13:48:03 +09:00
|
|
|
import * as Router from 'koa-router';
|
2018-08-14 20:13:32 +09:00
|
|
|
import config from '../../config';
|
2019-02-05 11:48:08 +09:00
|
|
|
import $ from 'cafy';
|
|
|
|
import ID, { transform } from '../../misc/cafy-id';
|
2018-08-14 20:13:32 +09:00
|
|
|
import User from '../../models/user';
|
|
|
|
import Following from '../../models/following';
|
2019-01-31 02:29:36 +09:00
|
|
|
import { renderActivity } from '../../remote/activitypub/renderer';
|
2018-08-14 20:13:32 +09:00
|
|
|
import renderOrderedCollection from '../../remote/activitypub/renderer/ordered-collection';
|
|
|
|
import renderOrderedCollectionPage from '../../remote/activitypub/renderer/ordered-collection-page';
|
|
|
|
import renderFollowUser from '../../remote/activitypub/renderer/follow-user';
|
2018-08-21 13:48:03 +09:00
|
|
|
import { setResponseType } from '../activitypub';
|
2018-08-14 20:13:32 +09:00
|
|
|
|
2018-08-21 13:48:03 +09:00
|
|
|
export default async (ctx: Router.IRouterContext) => {
|
2018-12-16 01:44:59 +09:00
|
|
|
if (!ObjectID.isValid(ctx.params.user)) {
|
|
|
|
ctx.status = 404;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const userId = new ObjectID(ctx.params.user);
|
2018-08-14 20:13:32 +09:00
|
|
|
|
|
|
|
// Get 'cursor' parameter
|
|
|
|
const [cursor = null, cursorErr] = $.type(ID).optional.get(ctx.request.query.cursor);
|
|
|
|
|
|
|
|
// Get 'page' parameter
|
|
|
|
const pageErr = !$.str.optional.or(['true', 'false']).ok(ctx.request.query.page);
|
|
|
|
const page: boolean = ctx.request.query.page === 'true';
|
|
|
|
|
|
|
|
// Validate parameters
|
|
|
|
if (cursorErr || pageErr) {
|
|
|
|
ctx.status = 400;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify user
|
|
|
|
const user = await User.findOne({
|
|
|
|
_id: userId,
|
|
|
|
host: null
|
|
|
|
});
|
|
|
|
|
|
|
|
if (user === null) {
|
|
|
|
ctx.status = 404;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const limit = 10;
|
|
|
|
const partOf = `${config.url}/users/${userId}/followers`;
|
|
|
|
|
|
|
|
if (page) {
|
|
|
|
// Construct query
|
|
|
|
const query = {
|
|
|
|
followeeId: user._id
|
|
|
|
} as any;
|
|
|
|
|
|
|
|
// カーソルが指定されている場合
|
|
|
|
if (cursor) {
|
|
|
|
query._id = {
|
2018-11-02 03:32:24 +09:00
|
|
|
$lt: transform(cursor)
|
2018-08-14 20:13:32 +09:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get followers
|
|
|
|
const followings = await Following
|
|
|
|
.find(query, {
|
|
|
|
limit: limit + 1,
|
|
|
|
sort: { _id: -1 }
|
|
|
|
});
|
|
|
|
|
|
|
|
// 「次のページ」があるかどうか
|
|
|
|
const inStock = followings.length === limit + 1;
|
|
|
|
if (inStock) followings.pop();
|
|
|
|
|
|
|
|
const renderedFollowers = await Promise.all(followings.map(following => renderFollowUser(following.followerId)));
|
|
|
|
const rendered = renderOrderedCollectionPage(
|
|
|
|
`${partOf}?page=true${cursor ? `&cursor=${cursor}` : ''}`,
|
|
|
|
user.followersCount, renderedFollowers, partOf,
|
|
|
|
null,
|
|
|
|
inStock ? `${partOf}?page=true&cursor=${followings[followings.length - 1]._id}` : null
|
|
|
|
);
|
|
|
|
|
2019-01-31 02:29:36 +09:00
|
|
|
ctx.body = renderActivity(rendered);
|
2018-08-21 13:48:03 +09:00
|
|
|
setResponseType(ctx);
|
2018-08-14 20:13:32 +09:00
|
|
|
} else {
|
|
|
|
// index page
|
|
|
|
const rendered = renderOrderedCollection(partOf, user.followersCount, `${partOf}?page=true`, null);
|
2019-01-31 02:29:36 +09:00
|
|
|
ctx.body = renderActivity(rendered);
|
2018-09-19 07:17:19 +09:00
|
|
|
ctx.set('Cache-Control', 'private, max-age=0, must-revalidate');
|
2018-08-21 13:48:03 +09:00
|
|
|
setResponseType(ctx);
|
2018-08-14 20:13:32 +09:00
|
|
|
}
|
|
|
|
};
|