2018-12-15 09:44:59 -07:00
|
|
|
import { ObjectID } from 'mongodb';
|
2018-08-20 22:48:03 -06:00
|
|
|
import * as Router from 'koa-router';
|
2018-08-14 05:13:32 -06:00
|
|
|
import config from '../../config';
|
2018-12-15 09:44:59 -07:00
|
|
|
import $ from 'cafy';
|
|
|
|
import ID, { transform } from '../../misc/cafy-id';
|
2018-08-14 05:13:32 -06:00
|
|
|
import User from '../../models/user';
|
|
|
|
import pack from '../../remote/activitypub/renderer';
|
|
|
|
import renderOrderedCollection from '../../remote/activitypub/renderer/ordered-collection';
|
|
|
|
import renderOrderedCollectionPage from '../../remote/activitypub/renderer/ordered-collection-page';
|
2018-08-20 22:48:03 -06:00
|
|
|
import { setResponseType } from '../activitypub';
|
2018-08-14 05:13:32 -06:00
|
|
|
|
2018-09-07 14:24:55 -06:00
|
|
|
import Note, { INote } from '../../models/note';
|
2018-08-14 05:13:32 -06:00
|
|
|
import renderNote from '../../remote/activitypub/renderer/note';
|
2018-09-07 14:24:55 -06:00
|
|
|
import renderCreate from '../../remote/activitypub/renderer/create';
|
|
|
|
import renderAnnounce from '../../remote/activitypub/renderer/announce';
|
2018-09-05 11:16:08 -06:00
|
|
|
import { countIf } from '../../prelude/array';
|
2018-08-14 05:13:32 -06:00
|
|
|
|
2018-08-20 22:48:03 -06:00
|
|
|
export default async (ctx: Router.IRouterContext) => {
|
2018-12-15 09:44:59 -07:00
|
|
|
if (!ObjectID.isValid(ctx.params.user)) {
|
|
|
|
ctx.status = 404;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const userId = new ObjectID(ctx.params.user);
|
2018-08-14 05:13:32 -06:00
|
|
|
|
|
|
|
// Get 'sinceId' parameter
|
|
|
|
const [sinceId, sinceIdErr] = $.type(ID).optional.get(ctx.request.query.since_id);
|
|
|
|
|
|
|
|
// Get 'untilId' parameter
|
|
|
|
const [untilId, untilIdErr] = $.type(ID).optional.get(ctx.request.query.until_id);
|
|
|
|
|
|
|
|
// 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
|
2018-09-05 11:16:08 -06:00
|
|
|
if (sinceIdErr || untilIdErr || pageErr || countIf(x => x != null, [sinceId, untilId]) > 1) {
|
2018-08-14 05:13:32 -06:00
|
|
|
ctx.status = 400;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Verify user
|
|
|
|
const user = await User.findOne({
|
|
|
|
_id: userId,
|
|
|
|
host: null
|
|
|
|
});
|
|
|
|
|
|
|
|
if (user === null) {
|
|
|
|
ctx.status = 404;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const limit = 20;
|
|
|
|
const partOf = `${config.url}/users/${userId}/outbox`;
|
|
|
|
|
|
|
|
if (page) {
|
|
|
|
//#region Construct query
|
|
|
|
const sort = {
|
|
|
|
_id: -1
|
|
|
|
};
|
|
|
|
|
|
|
|
const query = {
|
|
|
|
userId: user._id,
|
2018-11-15 13:47:29 -07:00
|
|
|
visibility: { $in: ['public', 'home'] },
|
|
|
|
localOnly: { $ne: true }
|
2018-08-14 05:13:32 -06:00
|
|
|
} as any;
|
|
|
|
|
|
|
|
if (sinceId) {
|
|
|
|
sort._id = 1;
|
|
|
|
query._id = {
|
2018-11-01 12:32:24 -06:00
|
|
|
$gt: transform(sinceId)
|
2018-08-14 05:13:32 -06:00
|
|
|
};
|
|
|
|
} else if (untilId) {
|
|
|
|
query._id = {
|
2018-11-01 12:32:24 -06:00
|
|
|
$lt: transform(untilId)
|
2018-08-14 05:13:32 -06:00
|
|
|
};
|
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
|
|
|
|
// Issue query
|
|
|
|
const notes = await Note
|
|
|
|
.find(query, {
|
|
|
|
limit: limit,
|
|
|
|
sort: sort
|
|
|
|
});
|
|
|
|
|
|
|
|
if (sinceId) notes.reverse();
|
|
|
|
|
2018-09-07 14:24:55 -06:00
|
|
|
const activities = await Promise.all(notes.map(note => packActivity(note)));
|
2018-08-14 05:13:32 -06:00
|
|
|
const rendered = renderOrderedCollectionPage(
|
|
|
|
`${partOf}?page=true${sinceId ? `&since_id=${sinceId}` : ''}${untilId ? `&until_id=${untilId}` : ''}`,
|
2018-09-07 14:24:55 -06:00
|
|
|
user.notesCount, activities, partOf,
|
2018-08-14 05:13:32 -06:00
|
|
|
notes.length > 0 ? `${partOf}?page=true&since_id=${notes[0]._id}` : null,
|
|
|
|
notes.length > 0 ? `${partOf}?page=true&until_id=${notes[notes.length - 1]._id}` : null
|
|
|
|
);
|
|
|
|
|
|
|
|
ctx.body = pack(rendered);
|
2018-09-18 16:17:19 -06:00
|
|
|
ctx.set('Cache-Control', 'private, max-age=0, must-revalidate');
|
2018-08-20 22:48:03 -06:00
|
|
|
setResponseType(ctx);
|
2018-08-14 05:13:32 -06:00
|
|
|
} else {
|
|
|
|
// index page
|
|
|
|
const rendered = renderOrderedCollection(partOf, user.notesCount,
|
|
|
|
`${partOf}?page=true`,
|
|
|
|
`${partOf}?page=true&since_id=000000000000000000000000`
|
|
|
|
);
|
|
|
|
ctx.body = pack(rendered);
|
2018-09-18 16:17:19 -06:00
|
|
|
ctx.set('Cache-Control', 'private, max-age=0, must-revalidate');
|
2018-08-20 22:48:03 -06:00
|
|
|
setResponseType(ctx);
|
2018-08-14 05:13:32 -06:00
|
|
|
}
|
|
|
|
};
|
2018-09-07 14:24:55 -06:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Pack Create<Note> or Announce Activity
|
|
|
|
* @param note Note
|
|
|
|
*/
|
|
|
|
export async function packActivity(note: INote): Promise<object> {
|
2018-09-08 11:59:14 -06:00
|
|
|
if (note.renoteId && note.text == null && note.poll == null && (note.fileIds == null || note.fileIds.length == 0)) {
|
2018-09-07 14:24:55 -06:00
|
|
|
const renote = await Note.findOne(note.renoteId);
|
|
|
|
return renderAnnounce(renote.uri ? renote.uri : `${config.url}/notes/${renote._id}`, note);
|
|
|
|
}
|
|
|
|
|
|
|
|
return renderCreate(await renderNote(note, false), note);
|
|
|
|
}
|