hippofish/packages/backend/src/core/NoteDeleteService.ts

287 lines
9.6 KiB
TypeScript
Raw Normal View History

/*
* SPDX-FileCopyrightText: syuilo and misskey-project
* SPDX-License-Identifier: AGPL-3.0-only
*/
2024-09-30 05:24:22 +02:00
import { Brackets, In, Not } from 'typeorm';
2022-09-17 20:27:08 +02:00
import { Injectable, Inject } from '@nestjs/common';
import type { MiUser, MiLocalUser, MiRemoteUser } from '@/models/User.js';
import type { MiNote, IMentionedRemoteUsers } from '@/models/Note.js';
2024-09-30 05:24:22 +02:00
import { LatestNote } from '@/models/LatestNote.js';
import type { InstancesRepository, LatestNotesRepository, NotesRepository, UsersRepository } from '@/models/_.js';
2022-09-17 20:27:08 +02:00
import { RelayService } from '@/core/RelayService.js';
import { FederatedInstanceService } from '@/core/FederatedInstanceService.js';
import { DI } from '@/di-symbols.js';
2022-09-20 22:33:11 +02:00
import type { Config } from '@/config.js';
2022-09-17 20:27:08 +02:00
import NotesChart from '@/core/chart/charts/notes.js';
import PerUserNotesChart from '@/core/chart/charts/per-user-notes.js';
import InstanceChart from '@/core/chart/charts/instance.js';
import { GlobalEventService } from '@/core/GlobalEventService.js';
2022-12-04 02:16:03 +01:00
import { ApRendererService } from '@/core/activitypub/ApRendererService.js';
import { ApDeliverManagerService } from '@/core/activitypub/ApDeliverManagerService.js';
import { UserEntityService } from '@/core/entities/UserEntityService.js';
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
import { bindThis } from '@/decorators.js';
import { MetaService } from '@/core/MetaService.js';
import { SearchService } from '@/core/SearchService.js';
import { ModerationLogService } from '@/core/ModerationLogService.js';
import { isQuote, isRenote } from '@/misc/is-renote.js';
2022-09-17 20:27:08 +02:00
@Injectable()
export class NoteDeleteService {
constructor(
@Inject(DI.config)
private config: Config,
@Inject(DI.usersRepository)
private usersRepository: UsersRepository,
@Inject(DI.notesRepository)
private notesRepository: NotesRepository,
2024-09-30 05:24:22 +02:00
@Inject(DI.latestNotesRepository)
private latestNotesRepository: LatestNotesRepository,
2022-09-17 20:27:08 +02:00
@Inject(DI.instancesRepository)
private instancesRepository: InstancesRepository,
private userEntityService: UserEntityService,
private noteEntityService: NoteEntityService,
2023-02-04 02:02:03 +01:00
private globalEventService: GlobalEventService,
2022-09-17 20:27:08 +02:00
private relayService: RelayService,
private federatedInstanceService: FederatedInstanceService,
private apRendererService: ApRendererService,
private apDeliverManagerService: ApDeliverManagerService,
private metaService: MetaService,
private searchService: SearchService,
private moderationLogService: ModerationLogService,
2022-09-17 20:27:08 +02:00
private notesChart: NotesChart,
private perUserNotesChart: PerUserNotesChart,
private instanceChart: InstanceChart,
) {}
2022-09-17 20:27:08 +02:00
/**
* 稿
* @param user 稿
* @param note 稿
*/
async delete(user: { id: MiUser['id']; uri: MiUser['uri']; host: MiUser['host']; isBot: MiUser['isBot']; }, note: MiNote, quiet = false, deleter?: MiUser) {
2022-09-17 20:27:08 +02:00
const deletedAt = new Date();
const cascadingNotes = await this.findCascadingNotes(note);
2022-09-17 20:27:08 +02:00
if (note.replyId) {
await this.notesRepository.decrement({ id: note.replyId }, 'repliesCount', 1);
}
2023-10-15 19:43:57 +02:00
if (note.renoteId && note.text == null && !note.hasPoll && (note.fileIds == null || note.fileIds.length === 0)) {
await this.notesRepository.findOneBy({ id: note.renoteId }).then(async (renote) => {
if (!renote) return;
if (renote.userId !== user.id) await this.notesRepository.decrement({ id: renote.id }, 'renoteCount', 1);
});
}
2022-09-17 20:27:08 +02:00
if (!quiet) {
2023-02-04 02:02:03 +01:00
this.globalEventService.publishNoteStream(note.id, 'deleted', {
2022-09-17 20:27:08 +02:00
deletedAt: deletedAt,
});
//#region ローカルの投稿なら削除アクティビティを配送
if (this.userEntityService.isLocalUser(user) && !note.localOnly) {
let renote: MiNote | null = null;
2022-09-17 20:27:08 +02:00
// if deleted note is renote
if (isRenote(note) && !isQuote(note)) {
2022-09-17 20:27:08 +02:00
renote = await this.notesRepository.findOneBy({
id: note.renoteId,
});
}
2023-02-12 10:47:30 +01:00
const content = this.apRendererService.addContext(renote
2022-09-17 20:27:08 +02:00
? this.apRendererService.renderUndo(this.apRendererService.renderAnnounce(renote.uri ?? `${this.config.url}/notes/${renote.id}`, note), user)
: this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${note.id}`), user));
2022-09-18 20:11:50 +02:00
this.deliverToConcerned(user, note, content);
2022-09-17 20:27:08 +02:00
}
// also deliver delete activity to cascaded notes
const federatedLocalCascadingNotes = (cascadingNotes).filter(note => !note.localOnly && note.userHost == null); // filter out local-only notes
for (const cascadingNote of federatedLocalCascadingNotes) {
2022-09-17 20:27:08 +02:00
if (!cascadingNote.user) continue;
if (!this.userEntityService.isLocalUser(cascadingNote.user)) continue;
2023-02-12 10:47:30 +01:00
const content = this.apRendererService.addContext(this.apRendererService.renderDelete(this.apRendererService.renderTombstone(`${this.config.url}/notes/${cascadingNote.id}`), cascadingNote.user));
2022-09-18 20:11:50 +02:00
this.deliverToConcerned(cascadingNote.user, cascadingNote, content);
2022-09-17 20:27:08 +02:00
}
//#endregion
const meta = await this.metaService.fetch();
2022-09-17 20:27:08 +02:00
this.notesChart.update(note, false);
if (meta.enableChartsForRemoteUser || (user.host == null)) {
this.perUserNotesChart.update(user, note, false);
}
2022-09-17 20:27:08 +02:00
if (note.renoteId && note.text) {
// Decrement notes count (user)
this.decNotesCountOfUser(user);
} else if (!note.renoteId) {
// Decrement notes count (user)
this.decNotesCountOfUser(user);
}
2022-09-17 20:27:08 +02:00
if (this.userEntityService.isRemoteUser(user)) {
this.federatedInstanceService.fetch(user.host).then(async i => {
if (note.renoteId && note.text) {
this.instancesRepository.decrement({ id: i.id }, 'notesCount', 1);
} else if (!note.renoteId) {
this.instancesRepository.decrement({ id: i.id }, 'notesCount', 1);
}
if ((await this.metaService.fetch()).enableChartsForFederatedInstances) {
this.instanceChart.updateNote(i.host, note, false);
}
2022-09-17 20:27:08 +02:00
});
}
}
for (const cascadingNote of cascadingNotes) {
this.searchService.unindexNote(cascadingNote);
}
this.searchService.unindexNote(note);
2022-09-17 20:27:08 +02:00
await this.notesRepository.delete({
id: note.id,
userId: user.id,
});
2024-09-30 05:24:22 +02:00
await this.updateLatestNote(note);
if (deleter && (note.userId !== deleter.id)) {
2023-09-25 03:29:12 +02:00
const user = await this.usersRepository.findOneByOrFail({ id: note.userId });
this.moderationLogService.log(deleter, 'deleteNote', {
noteId: note.id,
noteUserId: note.userId,
2023-09-25 03:29:12 +02:00
noteUserUsername: user.username,
noteUserHost: user.host,
note: note,
});
}
2022-09-17 20:27:08 +02:00
}
@bindThis
private decNotesCountOfUser(user: { id: MiUser['id']; }) {
this.usersRepository.createQueryBuilder().update()
.set({
updatedAt: new Date(),
notesCount: () => '"notesCount" - 1',
})
.where('id = :id', { id: user.id })
.execute();
}
@bindThis
private async findCascadingNotes(note: MiNote): Promise<MiNote[]> {
const recursive = async (noteId: string): Promise<MiNote[]> => {
2022-09-17 20:27:08 +02:00
const query = this.notesRepository.createQueryBuilder('note')
.where('note.replyId = :noteId', { noteId })
.orWhere(new Brackets(q => {
q.where('note.renoteId = :noteId', { noteId })
.andWhere('note.text IS NOT NULL');
}))
.leftJoinAndSelect('note.user', 'user');
const replies = await query.getMany();
return [
replies,
...await Promise.all(replies.map(reply => recursive(reply.id))),
].flat();
2022-09-17 20:27:08 +02:00
};
const cascadingNotes: MiNote[] = await recursive(note.id);
2022-09-17 20:27:08 +02:00
return cascadingNotes;
2022-09-17 20:27:08 +02:00
}
@bindThis
private async getMentionedRemoteUsers(note: MiNote) {
2022-09-17 20:27:08 +02:00
const where = [] as any[];
// mention / reply / dm
const uris = (JSON.parse(note.mentionedRemoteUsers) as IMentionedRemoteUsers).map(x => x.uri);
if (uris.length > 0) {
where.push(
{ uri: In(uris) },
);
}
// renote / quote
if (note.renoteUserId) {
where.push({
id: note.renoteUserId,
});
}
if (where.length === 0) return [];
return await this.usersRepository.find({
where,
}) as MiRemoteUser[];
2022-09-17 20:27:08 +02:00
}
@bindThis
private async deliverToConcerned(user: { id: MiLocalUser['id']; host: null; }, note: MiNote, content: any) {
2022-09-17 20:27:08 +02:00
this.apDeliverManagerService.deliverToFollowers(user, content);
this.relayService.deliverToRelays(user, content);
2022-09-18 20:11:50 +02:00
const remoteUsers = await this.getMentionedRemoteUsers(note);
2022-09-17 20:27:08 +02:00
for (const remoteUser of remoteUsers) {
this.apDeliverManagerService.deliverToUser(user, content, remoteUser);
}
}
2024-09-30 05:24:22 +02:00
private async updateLatestNote(note: MiNote) {
// If it's a DM, then it can't possibly be the latest note so we can safely skip this.
if (note.visibility === 'specified') return;
2024-10-02 17:38:21 +02:00
// Check if the deleted note was possibly the latest for the user
const hasLatestNote = await this.latestNotesRepository.existsBy({ userId: note.userId });
if (hasLatestNote) return;
2024-09-30 19:29:15 +02:00
// Find the newest remaining note for the user.
// We exclude DMs and pure renotes.
2024-09-30 05:24:22 +02:00
const nextLatest = await this.notesRepository
2024-09-30 19:29:15 +02:00
.createQueryBuilder('note')
2024-09-30 05:24:22 +02:00
.select()
.where({
userId: note.userId,
visibility: Not('specified'),
})
2024-09-30 19:29:15 +02:00
.andWhere(`
(
note."renoteId" IS NULL
OR note.text IS NOT NULL
OR note.cw IS NOT NULL
OR note."replyId" IS NOT NULL
OR note."hasPoll"
OR note."fileIds" != '{}'
)
`)
2024-09-30 05:24:22 +02:00
.orderBy({ id: 'DESC' })
.getOne();
if (!nextLatest) return;
// Record it as the latest
const latestNote = new LatestNote({
userId: note.userId,
noteId: nextLatest.id,
});
2024-10-02 17:38:21 +02:00
// When inserting the latest note, it's possible that another worker has "raced" the insert and already added a newer note.
// We must use orIgnore() to ensure that the query ignores conflicts, otherwise an exception may be thrown.
await this.latestNotesRepository
.createQueryBuilder('latest')
.insert()
.into(LatestNote)
.values(latestNote)
.orIgnore()
.execute();
2024-09-30 05:24:22 +02:00
}
2022-09-17 20:27:08 +02:00
}