149 lines
4.6 KiB
TypeScript
149 lines
4.6 KiB
TypeScript
import type { CacheableUser, User } from "@/models/entities/user.js";
|
|
import type { UserGroup } from "@/models/entities/user-group.js";
|
|
import type { DriveFile } from "@/models/entities/drive-file.js";
|
|
import {
|
|
MessagingMessages,
|
|
UserGroupJoinings,
|
|
Mutings,
|
|
Users,
|
|
} from "@/models/index.js";
|
|
import { genId } from "@/misc/gen-id.js";
|
|
import type { MessagingMessage } from "@/models/entities/messaging-message.js";
|
|
import {
|
|
publishMessagingStream,
|
|
publishMessagingIndexStream,
|
|
publishMainStream,
|
|
publishGroupMessagingStream,
|
|
} from "@/services/stream.js";
|
|
import { pushNotification } from "@/services/push-notification.js";
|
|
import { Not } from "typeorm";
|
|
import type { Note } from "@/models/entities/note.js";
|
|
import renderNote from "@/remote/activitypub/renderer/note.js";
|
|
import renderCreate from "@/remote/activitypub/renderer/create.js";
|
|
import { renderActivity } from "@/remote/activitypub/renderer/index.js";
|
|
import { deliver } from "@/queue/index.js";
|
|
|
|
export async function createMessage(
|
|
user: { id: User["id"]; host: User["host"] },
|
|
recipientUser: CacheableUser | undefined,
|
|
recipientGroup: UserGroup | undefined,
|
|
text: string | null | undefined,
|
|
file: DriveFile | null,
|
|
uri?: string,
|
|
) {
|
|
const message = {
|
|
id: genId(),
|
|
createdAt: new Date(),
|
|
fileId: file ? file.id : null,
|
|
recipientId: recipientUser ? recipientUser.id : null,
|
|
groupId: recipientGroup ? recipientGroup.id : null,
|
|
text: text ? text.trim() : null,
|
|
userId: user.id,
|
|
isRead: false,
|
|
reads: [] as any[],
|
|
uri,
|
|
} as MessagingMessage;
|
|
|
|
await MessagingMessages.insert(message);
|
|
|
|
const messageObj = await MessagingMessages.pack(message);
|
|
|
|
if (recipientUser) {
|
|
if (Users.isLocalUser(user)) {
|
|
// 自分のストリーム
|
|
publishMessagingStream(
|
|
message.userId,
|
|
recipientUser.id,
|
|
"message",
|
|
messageObj,
|
|
);
|
|
publishMessagingIndexStream(message.userId, "message", messageObj);
|
|
publishMainStream(message.userId, "messagingMessage", messageObj);
|
|
}
|
|
|
|
if (Users.isLocalUser(recipientUser)) {
|
|
// 相手のストリーム
|
|
publishMessagingStream(
|
|
recipientUser.id,
|
|
message.userId,
|
|
"message",
|
|
messageObj,
|
|
);
|
|
publishMessagingIndexStream(recipientUser.id, "message", messageObj);
|
|
publishMainStream(recipientUser.id, "messagingMessage", messageObj);
|
|
}
|
|
} else if (recipientGroup) {
|
|
// グループのストリーム
|
|
publishGroupMessagingStream(recipientGroup.id, "message", messageObj);
|
|
|
|
// メンバーのストリーム
|
|
const joinings = await UserGroupJoinings.findBy({
|
|
userGroupId: recipientGroup.id,
|
|
});
|
|
for (const joining of joinings) {
|
|
publishMessagingIndexStream(joining.userId, "message", messageObj);
|
|
publishMainStream(joining.userId, "messagingMessage", messageObj);
|
|
}
|
|
}
|
|
|
|
// 2秒経っても(今回作成した)メッセージが既読にならなかったら「未読のメッセージがありますよ」イベントを発行する
|
|
setTimeout(async () => {
|
|
const freshMessage = await MessagingMessages.findOneBy({ id: message.id });
|
|
if (freshMessage == null) return; // メッセージが削除されている場合もある
|
|
|
|
if (recipientUser && Users.isLocalUser(recipientUser)) {
|
|
if (freshMessage.isRead) return; // 既読
|
|
|
|
//#region ただしミュートされているなら発行しない
|
|
const mute = await Mutings.findBy({
|
|
muterId: recipientUser.id,
|
|
});
|
|
if (mute.map((m) => m.muteeId).includes(user.id)) return;
|
|
//#endregion
|
|
|
|
publishMainStream(recipientUser.id, "unreadMessagingMessage", messageObj);
|
|
pushNotification(recipientUser.id, "unreadMessagingMessage", messageObj);
|
|
} else if (recipientGroup) {
|
|
const joinings = await UserGroupJoinings.findBy({
|
|
userGroupId: recipientGroup.id,
|
|
userId: Not(user.id),
|
|
});
|
|
for (const joining of joinings) {
|
|
if (freshMessage.reads.includes(joining.userId)) return; // 既読
|
|
publishMainStream(joining.userId, "unreadMessagingMessage", messageObj);
|
|
pushNotification(joining.userId, "unreadMessagingMessage", messageObj);
|
|
}
|
|
}
|
|
}, 2000);
|
|
|
|
if (
|
|
recipientUser &&
|
|
Users.isLocalUser(user) &&
|
|
Users.isRemoteUser(recipientUser)
|
|
) {
|
|
const note = {
|
|
id: message.id,
|
|
createdAt: message.createdAt,
|
|
fileIds: message.fileId ? [message.fileId] : [],
|
|
text: message.text,
|
|
userId: message.userId,
|
|
visibility: "specified",
|
|
mentions: [recipientUser].map((u) => u.id),
|
|
mentionedRemoteUsers: JSON.stringify(
|
|
[recipientUser].map((u) => ({
|
|
uri: u.uri,
|
|
username: u.username,
|
|
host: u.host,
|
|
})),
|
|
),
|
|
} as Note;
|
|
|
|
const activity = renderActivity(
|
|
renderCreate(await renderNote(note, false, true), note),
|
|
);
|
|
|
|
deliver(user, activity, recipientUser.inbox);
|
|
}
|
|
return messageObj;
|
|
}
|