65e7204ec9
Related #6813
123 lines
3.5 KiB
TypeScript
123 lines
3.5 KiB
TypeScript
import { publishNoteStream } from '../../stream';
|
|
import { renderLike } from '../../../remote/activitypub/renderer/like';
|
|
import DeliverManager from '../../../remote/activitypub/deliver-manager';
|
|
import { renderActivity } from '../../../remote/activitypub/renderer';
|
|
import { toDbReaction, decodeReaction } from '../../../misc/reaction-lib';
|
|
import { User, IRemoteUser } from '../../../models/entities/user';
|
|
import { Note } from '../../../models/entities/note';
|
|
import { NoteReactions, Users, NoteWatchings, Notes, Emojis } from '../../../models';
|
|
import { Not } from 'typeorm';
|
|
import { perUserReactionsChart } from '../../chart';
|
|
import { genId } from '../../../misc/gen-id';
|
|
import { createNotification } from '../../create-notification';
|
|
import deleteReaction from './delete';
|
|
import { isDuplicateKeyValueError } from '../../../misc/is-duplicate-key-value-error';
|
|
import { NoteReaction } from '../../../models/entities/note-reaction';
|
|
|
|
export default async (user: User, note: Note, reaction?: string) => {
|
|
// TODO: cache
|
|
reaction = await toDbReaction(reaction, user.host);
|
|
|
|
let record: NoteReaction;
|
|
|
|
// Create reaction
|
|
try {
|
|
record = await NoteReactions.save({
|
|
id: genId(),
|
|
createdAt: new Date(),
|
|
noteId: note.id,
|
|
userId: user.id,
|
|
reaction
|
|
});
|
|
} catch (e) {
|
|
if (isDuplicateKeyValueError(e)) {
|
|
record = await NoteReactions.findOneOrFail({
|
|
noteId: note.id,
|
|
userId: user.id,
|
|
});
|
|
|
|
if (record.reaction !== reaction) {
|
|
// 別のリアクションがすでにされていたら置き換える
|
|
await deleteReaction(user, note);
|
|
} else {
|
|
// 同じリアクションがすでにされていたら何もしない
|
|
return;
|
|
}
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
|
|
// Increment reactions count
|
|
const sql = `jsonb_set("reactions", '{${reaction}}', (COALESCE("reactions"->>'${reaction}', '0')::int + 1)::text::jsonb)`;
|
|
await Notes.createQueryBuilder().update()
|
|
.set({
|
|
reactions: () => sql,
|
|
})
|
|
.where('id = :id', { id: note.id })
|
|
.execute();
|
|
|
|
Notes.increment({ id: note.id }, 'score', 1);
|
|
|
|
perUserReactionsChart.update(user, note);
|
|
|
|
// カスタム絵文字リアクションだったら絵文字情報も送る
|
|
const decodedReaction = decodeReaction(reaction);
|
|
|
|
let emoji = await Emojis.findOne({
|
|
where: {
|
|
name: decodedReaction.name,
|
|
host: decodedReaction.host
|
|
},
|
|
select: ['name', 'host', 'url']
|
|
});
|
|
|
|
if (emoji) {
|
|
emoji = {
|
|
name: emoji.host ? `${emoji.name}@${emoji.host}` : `${emoji.name}@.`,
|
|
url: emoji.url
|
|
} as any;
|
|
}
|
|
|
|
publishNoteStream(note.id, 'reacted', {
|
|
reaction: decodedReaction.reaction,
|
|
emoji: emoji,
|
|
userId: user.id
|
|
});
|
|
|
|
// リアクションされたユーザーがローカルユーザーなら通知を作成
|
|
if (note.userHost === null) {
|
|
createNotification(note.userId, 'reaction', {
|
|
notifierId: user.id,
|
|
noteId: note.id,
|
|
reaction: reaction
|
|
});
|
|
}
|
|
|
|
// Fetch watchers
|
|
NoteWatchings.find({
|
|
noteId: note.id,
|
|
userId: Not(user.id)
|
|
}).then(watchers => {
|
|
for (const watcher of watchers) {
|
|
createNotification(watcher.userId, 'reaction', {
|
|
notifierId: user.id,
|
|
noteId: note.id,
|
|
reaction: reaction
|
|
});
|
|
}
|
|
});
|
|
|
|
//#region 配信
|
|
if (Users.isLocalUser(user) && !note.localOnly) {
|
|
const content = renderActivity(await renderLike(record, note));
|
|
const dm = new DeliverManager(user, content);
|
|
if (note.userHost !== null) {
|
|
const reactee = await Users.findOne(note.userId);
|
|
dm.addDirectRecipe(reactee as IRemoteUser);
|
|
}
|
|
dm.addFollowersRecipe();
|
|
dm.execute();
|
|
}
|
|
//#endregion
|
|
};
|