hippofish/src/services/note/reaction/create.ts

93 lines
2.4 KiB
TypeScript
Raw Normal View History

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