2018-06-20 21:35:28 -05:00
|
|
|
import { IUser, isLocalUser, isRemoteUser } from '../../../models/user';
|
|
|
|
import Note, { INote } from '../../../models/note';
|
2018-04-07 12:30:37 -05:00
|
|
|
import NoteReaction from '../../../models/note-reaction';
|
2018-07-07 05:19:00 -05:00
|
|
|
import { publishNoteStream } from '../../../stream';
|
|
|
|
import notify from '../../..//notify';
|
2018-04-07 12:30:37 -05:00
|
|
|
import NoteWatching from '../../../models/note-watching';
|
2018-04-07 03:05:14 -05:00
|
|
|
import watch from '../watch';
|
|
|
|
import renderLike from '../../../remote/activitypub/renderer/like';
|
|
|
|
import { deliver } from '../../../queue';
|
2018-04-13 00:39:08 -05:00
|
|
|
import pack from '../../../remote/activitypub/renderer';
|
2018-04-07 03:05:14 -05:00
|
|
|
|
2018-04-07 12:30:37 -05:00
|
|
|
export default async (user: IUser, note: INote, reaction: string) => new Promise(async (res, rej) => {
|
2018-04-07 03:05:14 -05:00
|
|
|
// Myself
|
2018-04-07 12:30:37 -05:00
|
|
|
if (note.userId.equals(user._id)) {
|
|
|
|
return rej('cannot react to my note');
|
2018-04-07 03:05:14 -05:00
|
|
|
}
|
|
|
|
|
2018-04-15 03:53:25 -05:00
|
|
|
// Create reaction
|
|
|
|
try {
|
|
|
|
await NoteReaction.insert({
|
|
|
|
createdAt: new Date(),
|
|
|
|
noteId: note._id,
|
|
|
|
userId: user._id,
|
|
|
|
reaction
|
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
// duplicate key error
|
2018-04-18 19:17:42 -05:00
|
|
|
if (e.code === 11000) {
|
|
|
|
return res(null);
|
2018-04-15 03:53:25 -05:00
|
|
|
}
|
2018-04-07 03:05:14 -05:00
|
|
|
|
2018-04-15 03:53:25 -05:00
|
|
|
console.error(e);
|
|
|
|
return rej('something happened');
|
2018-04-07 03:05:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
res();
|
|
|
|
|
2018-06-17 05:59:02 -05:00
|
|
|
const inc: {[key: string]: number} = {};
|
2018-04-07 03:05:14 -05:00
|
|
|
inc[`reactionCounts.${reaction}`] = 1;
|
|
|
|
|
|
|
|
// Increment reactions count
|
2018-04-07 12:30:37 -05:00
|
|
|
await Note.update({ _id: note._id }, {
|
2018-04-07 03:05:14 -05:00
|
|
|
$inc: inc
|
|
|
|
});
|
|
|
|
|
2018-04-07 12:30:37 -05:00
|
|
|
publishNoteStream(note._id, 'reacted');
|
2018-04-07 03:05:14 -05:00
|
|
|
|
2018-04-21 20:53:27 -05:00
|
|
|
// リアクションされたユーザーがローカルユーザーなら通知を作成
|
|
|
|
if (isLocalUser(note._user)) {
|
|
|
|
notify(note.userId, user._id, 'reaction', {
|
|
|
|
noteId: note._id,
|
|
|
|
reaction: reaction
|
|
|
|
});
|
|
|
|
}
|
2018-04-07 03:05:14 -05:00
|
|
|
|
|
|
|
// Fetch watchers
|
2018-04-07 12:30:37 -05:00
|
|
|
NoteWatching
|
2018-04-07 03:05:14 -05:00
|
|
|
.find({
|
2018-04-07 12:30:37 -05:00
|
|
|
noteId: note._id,
|
2018-04-07 03:05:14 -05:00
|
|
|
userId: { $ne: user._id }
|
|
|
|
}, {
|
|
|
|
fields: {
|
|
|
|
userId: true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.then(watchers => {
|
|
|
|
watchers.forEach(watcher => {
|
|
|
|
notify(watcher.userId, user._id, 'reaction', {
|
2018-04-07 12:30:37 -05:00
|
|
|
noteId: note._id,
|
2018-04-07 03:05:14 -05:00
|
|
|
reaction: reaction
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// ユーザーがローカルユーザーかつ自動ウォッチ設定がオンならばこの投稿をWatchする
|
2018-04-07 13:58:11 -05:00
|
|
|
if (isLocalUser(user) && user.settings.autoWatch !== false) {
|
2018-04-07 12:30:37 -05:00
|
|
|
watch(user._id, note);
|
2018-04-07 03:05:14 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//#region 配信
|
|
|
|
// リアクターがローカルユーザーかつリアクション対象がリモートユーザーの投稿なら配送
|
2018-04-07 12:30:37 -05:00
|
|
|
if (isLocalUser(user) && isRemoteUser(note._user)) {
|
2018-04-23 01:27:01 -05:00
|
|
|
const content = pack(renderLike(user, note, reaction));
|
2018-04-15 03:53:25 -05:00
|
|
|
deliver(user, content, note._user.inbox);
|
2018-04-07 03:05:14 -05:00
|
|
|
}
|
|
|
|
//#endregion
|
|
|
|
});
|