174 lines
4.7 KiB
TypeScript
174 lines
4.7 KiB
TypeScript
import { publishNoteStream } from "@/services/stream.js";
|
|
import { renderLike } from "@/remote/activitypub/renderer/like.js";
|
|
import DeliverManager from "@/remote/activitypub/deliver-manager.js";
|
|
import { renderActivity } from "@/remote/activitypub/renderer/index.js";
|
|
import { toDbReaction, decodeReaction } from "@/misc/reaction-lib.js";
|
|
import type { User, IRemoteUser } from "@/models/entities/user.js";
|
|
import type { Note } from "@/models/entities/note.js";
|
|
import {
|
|
NoteReactions,
|
|
Users,
|
|
NoteWatchings,
|
|
Notes,
|
|
Emojis,
|
|
Blockings,
|
|
} from "@/models/index.js";
|
|
import { IsNull, Not } from "typeorm";
|
|
import { perUserReactionsChart } from "@/services/chart/index.js";
|
|
import { genId } from "@/misc/gen-id.js";
|
|
import { createNotification } from "../../create-notification.js";
|
|
import deleteReaction from "./delete.js";
|
|
import { isDuplicateKeyValueError } from "@/misc/is-duplicate-key-value-error.js";
|
|
import type { NoteReaction } from "@/models/entities/note-reaction.js";
|
|
import { IdentifiableError } from "@/misc/identifiable-error.js";
|
|
|
|
export default async (
|
|
user: { id: User["id"]; host: User["host"] },
|
|
note: Note,
|
|
reaction?: string,
|
|
) => {
|
|
// Check blocking
|
|
if (note.userId !== user.id) {
|
|
const block = await Blockings.findOneBy({
|
|
blockerId: note.userId,
|
|
blockeeId: user.id,
|
|
});
|
|
if (block) {
|
|
throw new IdentifiableError("e70412a4-7197-4726-8e74-f3e0deb92aa7");
|
|
}
|
|
}
|
|
|
|
// check visibility
|
|
if (!(await Notes.isVisibleForMe(note, user.id))) {
|
|
throw new IdentifiableError(
|
|
"68e9d2d1-48bf-42c2-b90a-b20e09fd3d48",
|
|
"Note not accessible for you.",
|
|
);
|
|
}
|
|
|
|
// TODO: cache
|
|
reaction = await toDbReaction(reaction, user.host);
|
|
|
|
const record: NoteReaction = {
|
|
id: genId(),
|
|
createdAt: new Date(),
|
|
noteId: note.id,
|
|
userId: user.id,
|
|
reaction,
|
|
};
|
|
|
|
// Create reaction
|
|
try {
|
|
await NoteReactions.insert(record);
|
|
} catch (e) {
|
|
if (isDuplicateKeyValueError(e)) {
|
|
const exists = await NoteReactions.findOneByOrFail({
|
|
noteId: note.id,
|
|
userId: user.id,
|
|
});
|
|
|
|
if (exists.reaction !== reaction) {
|
|
// 別のリアクションがすでにされていたら置き換える
|
|
await deleteReaction(user, note);
|
|
await NoteReactions.insert(record);
|
|
} else {
|
|
// 同じリアクションがすでにされていたらエラー
|
|
throw new IdentifiableError("51c42bb4-931a-456b-bff7-e5a8a70dd298");
|
|
}
|
|
} 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,
|
|
score: () => '"score" + 1',
|
|
})
|
|
.where("id = :id", { id: note.id })
|
|
.execute();
|
|
|
|
perUserReactionsChart.update(user, note);
|
|
|
|
// カスタム絵文字リアクションだったら絵文字情報も送る
|
|
const decodedReaction = decodeReaction(reaction);
|
|
|
|
const emoji = await Emojis.findOne({
|
|
where: {
|
|
name: decodedReaction.name,
|
|
host: decodedReaction.host ?? IsNull(),
|
|
},
|
|
select: ["name", "host", "originalUrl", "publicUrl"],
|
|
});
|
|
|
|
publishNoteStream(note.id, "reacted", {
|
|
reaction: decodedReaction.reaction,
|
|
emoji:
|
|
emoji != null
|
|
? {
|
|
name: emoji.host
|
|
? `${emoji.name}@${emoji.host}`
|
|
: `${emoji.name}@.`,
|
|
url: emoji.publicUrl || emoji.originalUrl, // || emoji.originalUrl してるのは後方互換性のため
|
|
}
|
|
: null,
|
|
userId: user.id,
|
|
});
|
|
|
|
// Create notification if the reaction target is a local user.
|
|
if (note.userHost === null) {
|
|
createNotification(note.userId, "reaction", {
|
|
notifierId: user.id,
|
|
note: note,
|
|
noteId: note.id,
|
|
reaction: reaction,
|
|
});
|
|
}
|
|
|
|
// Fetch watchers
|
|
NoteWatchings.findBy({
|
|
noteId: note.id,
|
|
userId: Not(user.id),
|
|
}).then((watchers) => {
|
|
for (const watcher of watchers) {
|
|
createNotification(watcher.userId, "reaction", {
|
|
notifierId: user.id,
|
|
note: note,
|
|
noteId: note.id,
|
|
reaction: reaction,
|
|
});
|
|
}
|
|
});
|
|
|
|
//#region deliver
|
|
if (
|
|
Users.isLocalUser(user) &&
|
|
!note.localOnly &&
|
|
note.visibility !== "hidden"
|
|
) {
|
|
const content = renderActivity(await renderLike(record, note));
|
|
const dm = new DeliverManager(user, content);
|
|
if (note.userHost !== null) {
|
|
const reactee = await Users.findOneBy({ id: note.userId });
|
|
dm.addDirectRecipe(reactee as IRemoteUser);
|
|
}
|
|
|
|
if (["public", "home", "followers"].includes(note.visibility)) {
|
|
dm.addFollowersRecipe();
|
|
} else if (note.visibility === "specified") {
|
|
const visibleUsers = await Promise.all(
|
|
note.visibleUserIds.map((id) => Users.findOneBy({ id })),
|
|
);
|
|
for (const u of visibleUsers.filter((u) => u && Users.isRemoteUser(u))) {
|
|
dm.addDirectRecipe(u as IRemoteUser);
|
|
}
|
|
}
|
|
|
|
dm.execute();
|
|
}
|
|
//#endregion
|
|
};
|