/* * SPDX-FileCopyrightText: syuilo and misskey-project * SPDX-License-Identifier: AGPL-3.0-only */ import { Injectable } from '@nestjs/common'; import { NoteEntityService } from '@/core/entities/NoteEntityService.js'; import { bindThis } from '@/decorators.js'; import { RoleService } from '@/core/RoleService.js'; import type { GlobalEvents } from '@/core/GlobalEventService.js'; import type { JsonObject } from '@/misc/json-value.js'; import { isQuotePacked, isRenotePacked } from '@/misc/is-renote.js'; import Channel, { type MiChannelService } from '../channel.js'; class RoleTimelineChannel extends Channel { public readonly chName = 'roleTimeline'; public static shouldShare = false; public static requireCredential = false as const; private roleId: string; constructor( noteEntityService: NoteEntityService, private roleservice: RoleService, id: string, connection: Channel['connection'], ) { super(id, connection, noteEntityService); //this.onNote = this.onNote.bind(this); } @bindThis public async init(params: JsonObject) { if (typeof params.roleId !== 'string') return; this.roleId = params.roleId; this.subscriber.on(`roleTimelineStream:${this.roleId}`, this.onEvent); } @bindThis private async onEvent(data: GlobalEvents['roleTimeline']['payload']) { if (data.type === 'note') { const note = data.body; const isMe = this.user?.id === note.userId; // TODO this should be cached if (!(await this.roleservice.isExplorable({ id: this.roleId }))) { return; } if (note.visibility !== 'public') return; if (this.isNoteMutedOrBlocked(note)) return; if (note.reply) { const reply = note.reply; // 自分のフォローしていないユーザーの visibility: followers な投稿への返信は弾く if (!this.isNoteVisibleToMe(reply)) return; if (!this.following.get(note.userId)?.withReplies) { // 「チャンネル接続主への返信」でもなければ、「チャンネル接続主が行った返信」でもなければ、「投稿者の投稿者自身への返信」でもない場合 if (reply.userId !== this.user?.id && !isMe && reply.userId !== note.userId) return; } } // 純粋なリノート(引用リノートでないリノート)の場合 if (isRenotePacked(note) && !isQuotePacked(note) && note.renote) { if (note.renote.reply) { const reply = note.renote.reply; // 自分のフォローしていないユーザーの visibility: followers な投稿への返信のリノートは弾く if (!this.isNoteVisibleToMe(reply)) return; } } const clonedNote = await this.assignMyReaction(note); await this.hideNote(clonedNote); this.send('note', clonedNote); } else { this.send(data.type, data.body); } } @bindThis public dispose() { // Unsubscribe events this.subscriber.off(`roleTimelineStream:${this.roleId}`, this.onEvent); } } @Injectable() export class RoleTimelineChannelService implements MiChannelService { public readonly shouldShare = RoleTimelineChannel.shouldShare; public readonly requireCredential = RoleTimelineChannel.requireCredential; public readonly kind = RoleTimelineChannel.kind; constructor( private noteEntityService: NoteEntityService, private roleservice: RoleService, ) { } @bindThis public create(id: string, connection: Channel['connection']): RoleTimelineChannel { return new RoleTimelineChannel( this.noteEntityService, this.roleservice, id, connection, ); } }