2023-07-27 14:31:52 +09:00
|
|
|
/*
|
2024-02-12 11:37:45 +09:00
|
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
2023-07-27 14:31:52 +09:00
|
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
*/
|
|
|
|
|
2023-03-03 11:13:12 +09:00
|
|
|
import { setTimeout } from 'node:timers/promises';
|
|
|
|
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
|
2023-04-05 07:52:49 +09:00
|
|
|
import { In } from 'typeorm';
|
2022-09-18 03:27:08 +09:00
|
|
|
import { DI } from '@/di-symbols.js';
|
2023-09-20 11:33:36 +09:00
|
|
|
import type { MiUser } from '@/models/User.js';
|
2023-03-10 14:22:37 +09:00
|
|
|
import type { Packed } from '@/misc/json-schema.js';
|
2023-09-20 11:33:36 +09:00
|
|
|
import type { MiNote } from '@/models/Note.js';
|
2022-09-18 03:27:08 +09:00
|
|
|
import { IdService } from '@/core/IdService.js';
|
|
|
|
import { GlobalEventService } from '@/core/GlobalEventService.js';
|
2023-09-15 14:28:29 +09:00
|
|
|
import type { NoteUnreadsRepository, MutingsRepository, NoteThreadMutingsRepository } from '@/models/_.js';
|
2023-02-01 17:29:28 +09:00
|
|
|
import { bindThis } from '@/decorators.js';
|
2024-01-08 12:28:13 +09:00
|
|
|
import { trackPromise } from '@/misc/promise-tracker.js';
|
2022-09-18 03:27:08 +09:00
|
|
|
|
|
|
|
@Injectable()
|
2023-03-03 11:13:12 +09:00
|
|
|
export class NoteReadService implements OnApplicationShutdown {
|
|
|
|
#shutdownController = new AbortController();
|
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
constructor(
|
|
|
|
@Inject(DI.noteUnreadsRepository)
|
|
|
|
private noteUnreadsRepository: NoteUnreadsRepository,
|
|
|
|
|
|
|
|
@Inject(DI.mutingsRepository)
|
|
|
|
private mutingsRepository: MutingsRepository,
|
|
|
|
|
|
|
|
@Inject(DI.noteThreadMutingsRepository)
|
|
|
|
private noteThreadMutingsRepository: NoteThreadMutingsRepository,
|
|
|
|
|
|
|
|
private idService: IdService,
|
2023-02-04 10:02:03 +09:00
|
|
|
private globalEventService: GlobalEventService,
|
2022-09-18 03:27:08 +09:00
|
|
|
) {
|
|
|
|
}
|
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2023-08-16 17:51:28 +09:00
|
|
|
public async insertNoteUnread(userId: MiUser['id'], note: MiNote, params: {
|
2022-09-18 03:27:08 +09:00
|
|
|
// NOTE: isSpecifiedがtrueならisMentionedは必ずfalse
|
|
|
|
isSpecified: boolean;
|
|
|
|
isMentioned: boolean;
|
|
|
|
}): Promise<void> {
|
|
|
|
//#region ミュートしているなら無視
|
|
|
|
const mute = await this.mutingsRepository.findBy({
|
|
|
|
muterId: userId,
|
|
|
|
});
|
|
|
|
if (mute.map(m => m.muteeId).includes(note.userId)) return;
|
|
|
|
//#endregion
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
// スレッドミュート
|
2024-02-08 16:04:41 +09:00
|
|
|
const isThreadMuted = await this.noteThreadMutingsRepository.exists({
|
2023-07-11 14:58:58 +09:00
|
|
|
where: {
|
|
|
|
userId: userId,
|
|
|
|
threadId: note.threadId ?? note.id,
|
|
|
|
},
|
2022-09-18 03:27:08 +09:00
|
|
|
});
|
2023-07-11 14:58:58 +09:00
|
|
|
if (isThreadMuted) return;
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
const unread = {
|
2023-10-16 10:45:22 +09:00
|
|
|
id: this.idService.gen(),
|
2022-09-18 03:27:08 +09:00
|
|
|
noteId: note.id,
|
|
|
|
userId: userId,
|
|
|
|
isSpecified: params.isSpecified,
|
|
|
|
isMentioned: params.isMentioned,
|
|
|
|
noteUserId: note.userId,
|
|
|
|
};
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
await this.noteUnreadsRepository.insert(unread);
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
// 2秒経っても既読にならなかったら「未読の投稿がありますよ」イベントを発行する
|
2023-03-03 11:13:12 +09:00
|
|
|
setTimeout(2000, 'unread note', { signal: this.#shutdownController.signal }).then(async () => {
|
2024-02-08 16:04:41 +09:00
|
|
|
const exist = await this.noteUnreadsRepository.exists({ where: { id: unread.id } });
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2023-07-11 14:58:58 +09:00
|
|
|
if (!exist) return;
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
if (params.isMentioned) {
|
2023-02-04 10:02:03 +09:00
|
|
|
this.globalEventService.publishMainStream(userId, 'unreadMention', note.id);
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
|
|
|
if (params.isSpecified) {
|
2023-02-04 10:02:03 +09:00
|
|
|
this.globalEventService.publishMainStream(userId, 'unreadSpecifiedNote', note.id);
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
2023-03-03 11:13:12 +09:00
|
|
|
}, () => { /* aborted, ignore it */ });
|
|
|
|
}
|
2022-09-18 03:27:08 +09:00
|
|
|
|
2022-12-04 15:03:09 +09:00
|
|
|
@bindThis
|
2022-09-18 03:27:08 +09:00
|
|
|
public async read(
|
2023-08-16 17:51:28 +09:00
|
|
|
userId: MiUser['id'],
|
|
|
|
notes: (MiNote | Packed<'Note'>)[],
|
2022-09-18 03:27:08 +09:00
|
|
|
): Promise<void> {
|
2023-08-16 17:51:28 +09:00
|
|
|
const readMentions: (MiNote | Packed<'Note'>)[] = [];
|
|
|
|
const readSpecifiedNotes: (MiNote | Packed<'Note'>)[] = [];
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
for (const note of notes) {
|
|
|
|
if (note.mentions && note.mentions.includes(userId)) {
|
|
|
|
readMentions.push(note);
|
|
|
|
} else if (note.visibleUserIds && note.visibleUserIds.includes(userId)) {
|
|
|
|
readSpecifiedNotes.push(note);
|
|
|
|
}
|
|
|
|
}
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2023-04-05 07:52:49 +09:00
|
|
|
if ((readMentions.length > 0) || (readSpecifiedNotes.length > 0)) {
|
2022-09-18 03:27:08 +09:00
|
|
|
// Remove the record
|
|
|
|
await this.noteUnreadsRepository.delete({
|
|
|
|
userId: userId,
|
2023-04-05 07:52:49 +09:00
|
|
|
noteId: In([...readMentions.map(n => n.id), ...readSpecifiedNotes.map(n => n.id)]),
|
2022-09-18 03:27:08 +09:00
|
|
|
});
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2022-09-18 03:27:08 +09:00
|
|
|
// TODO: ↓まとめてクエリしたい
|
2023-07-08 07:08:16 +09:00
|
|
|
|
2024-01-08 12:28:13 +09:00
|
|
|
trackPromise(this.noteUnreadsRepository.countBy({
|
2022-09-18 03:27:08 +09:00
|
|
|
userId: userId,
|
|
|
|
isMentioned: true,
|
|
|
|
}).then(mentionsCount => {
|
|
|
|
if (mentionsCount === 0) {
|
|
|
|
// 全て既読になったイベントを発行
|
2023-02-04 10:02:03 +09:00
|
|
|
this.globalEventService.publishMainStream(userId, 'readAllUnreadMentions');
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
2024-01-08 12:28:13 +09:00
|
|
|
}));
|
2023-07-08 07:08:16 +09:00
|
|
|
|
2024-01-08 12:28:13 +09:00
|
|
|
trackPromise(this.noteUnreadsRepository.countBy({
|
2022-09-18 03:27:08 +09:00
|
|
|
userId: userId,
|
|
|
|
isSpecified: true,
|
|
|
|
}).then(specifiedCount => {
|
|
|
|
if (specifiedCount === 0) {
|
|
|
|
// 全て既読になったイベントを発行
|
2023-02-04 10:02:03 +09:00
|
|
|
this.globalEventService.publishMainStream(userId, 'readAllUnreadSpecifiedNotes');
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
2024-01-08 12:28:13 +09:00
|
|
|
}));
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|
|
|
|
}
|
2023-03-03 11:13:12 +09:00
|
|
|
|
2023-05-29 13:21:26 +09:00
|
|
|
@bindThis
|
|
|
|
public dispose(): void {
|
2023-03-03 11:13:12 +09:00
|
|
|
this.#shutdownController.abort();
|
|
|
|
}
|
2023-05-29 13:21:26 +09:00
|
|
|
|
|
|
|
@bindThis
|
|
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
|
|
this.dispose();
|
|
|
|
}
|
2022-09-18 03:27:08 +09:00
|
|
|
}
|