1
1
mirror of https://github.com/kokonect-link/cherrypick synced 2025-01-06 01:43:21 +09:00
cherrypick/packages/backend/src/core/CreateNotificationService.ts

126 lines
5.1 KiB
TypeScript
Raw Normal View History

import { setTimeout } from 'node:timers/promises';
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
2022-09-21 05:33:11 +09:00
import type { MutingsRepository, NotificationsRepository, UserProfilesRepository, UsersRepository } from '@/models/index.js';
2022-09-18 03:27:08 +09:00
import type { User } from '@/models/entities/User.js';
import type { Notification } from '@/models/entities/Notification.js';
import { GlobalEventService } from '@/core/GlobalEventService.js';
import { IdService } from '@/core/IdService.js';
import { DI } from '@/di-symbols.js';
2022-12-04 10:16:03 +09:00
import { NotificationEntityService } from '@/core/entities/NotificationEntityService.js';
import { PushNotificationService } from '@/core/PushNotificationService.js';
import { bindThis } from '@/decorators.js';
2022-09-18 03:27:08 +09:00
@Injectable()
export class CreateNotificationService implements OnApplicationShutdown {
#shutdownController = new AbortController();
2022-09-18 03:27:08 +09:00
constructor(
@Inject(DI.usersRepository)
private usersRepository: UsersRepository,
@Inject(DI.userProfilesRepository)
private userProfilesRepository: UserProfilesRepository,
@Inject(DI.notificationsRepository)
private notificationsRepository: NotificationsRepository,
@Inject(DI.mutingsRepository)
private mutingsRepository: MutingsRepository,
private notificationEntityService: NotificationEntityService,
private idService: IdService,
2023-02-04 10:02:03 +09:00
private globalEventService: GlobalEventService,
2022-09-18 03:27:08 +09:00
private pushNotificationService: PushNotificationService,
) {
}
@bindThis
2022-09-18 03:27:08 +09:00
public async createNotification(
notifieeId: User['id'],
type: Notification['type'],
data: Partial<Notification>,
): Promise<Notification | null> {
if (data.notifierId && (notifieeId === data.notifierId)) {
return null;
}
2022-09-18 03:27:08 +09:00
const profile = await this.userProfilesRepository.findOneBy({ userId: notifieeId });
2022-09-18 03:27:08 +09:00
const isMuted = profile?.mutingNotificationTypes.includes(type);
2022-09-18 03:27:08 +09:00
// Create notification
const notification = await this.notificationsRepository.insert({
id: this.idService.genId(),
createdAt: new Date(),
notifieeId: notifieeId,
type: type,
// 相手がこの通知をミュートしているようなら、既読を予めつけておく
isRead: isMuted,
...data,
} as Partial<Notification>)
.then(x => this.notificationsRepository.findOneByOrFail(x.identifiers[0]));
2022-09-18 03:27:08 +09:00
const packed = await this.notificationEntityService.pack(notification, {});
2022-09-18 03:27:08 +09:00
// Publish notification event
2023-02-04 10:02:03 +09:00
this.globalEventService.publishMainStream(notifieeId, 'notification', packed);
2022-09-18 03:27:08 +09:00
// 2秒経っても(今回作成した)通知が既読にならなかったら「未読の通知がありますよ」イベントを発行する
setTimeout(2000, 'unread note', { signal: this.#shutdownController.signal }).then(async () => {
2022-09-18 03:27:08 +09:00
const fresh = await this.notificationsRepository.findOneBy({ id: notification.id });
if (fresh == null) return; // 既に削除されているかもしれない
if (fresh.isRead) return;
2022-09-18 03:27:08 +09:00
//#region ただしミュートしているユーザーからの通知なら無視
const mutings = await this.mutingsRepository.findBy({
muterId: notifieeId,
});
if (data.notifierId && mutings.map(m => m.muteeId).includes(data.notifierId)) {
return;
}
//#endregion
2023-02-04 10:02:03 +09:00
this.globalEventService.publishMainStream(notifieeId, 'unreadNotification', packed);
2022-09-18 03:27:08 +09:00
this.pushNotificationService.pushNotification(notifieeId, 'notification', packed);
2022-09-19 03:11:50 +09:00
if (type === 'follow') this.emailNotificationFollow(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! }));
if (type === 'receiveFollowRequest') this.emailNotificationReceiveFollowRequest(notifieeId, await this.usersRepository.findOneByOrFail({ id: data.notifierId! }));
}, () => { /* aborted, ignore it */ });
2022-09-18 03:27:08 +09:00
return notification;
}
// TODO
//const locales = await import('../../../../locales/index.js');
// TODO: locale ファイルをクライアント用とサーバー用で分けたい
@bindThis
2022-09-19 03:11:50 +09:00
private async emailNotificationFollow(userId: User['id'], follower: User) {
2022-09-18 03:27:08 +09:00
/*
const userProfile = await UserProfiles.findOneByOrFail({ userId: userId });
if (!userProfile.email || !userProfile.emailNotificationTypes.includes('follow')) return;
const locale = locales[userProfile.lang ?? 'ja-JP'];
const i18n = new I18n(locale);
// TODO: render user information html
sendEmail(userProfile.email, i18n.t('_email._follow.title'), `${follower.name} (@${Acct.toString(follower)})`, `${follower.name} (@${Acct.toString(follower)})`);
*/
}
@bindThis
2022-09-19 03:11:50 +09:00
private async emailNotificationReceiveFollowRequest(userId: User['id'], follower: User) {
2022-09-18 03:27:08 +09:00
/*
const userProfile = await UserProfiles.findOneByOrFail({ userId: userId });
if (!userProfile.email || !userProfile.emailNotificationTypes.includes('receiveFollowRequest')) return;
const locale = locales[userProfile.lang ?? 'ja-JP'];
const i18n = new I18n(locale);
// TODO: render user information html
sendEmail(userProfile.email, i18n.t('_email._receiveFollowRequest.title'), `${follower.name} (@${Acct.toString(follower)})`, `${follower.name} (@${Acct.toString(follower)})`);
*/
}
onApplicationShutdown(signal?: string | undefined): void {
this.#shutdownController.abort();
}
2022-09-18 03:27:08 +09:00
}