168 lines
5.8 KiB
TypeScript
168 lines
5.8 KiB
TypeScript
/*
|
||
* SPDX-FileCopyrightText: syuilo and misskey-project
|
||
* SPDX-License-Identifier: AGPL-3.0-only
|
||
*/
|
||
|
||
import { bindThis } from '@/decorators.js';
|
||
import { isInstanceMuted } from '@/misc/is-instance-muted.js';
|
||
import { isUserRelated } from '@/misc/is-user-related.js';
|
||
import { isRenotePacked, isQuotePacked, isPackedPureRenote } from '@/misc/is-renote.js';
|
||
import type { Packed } from '@/misc/json-schema.js';
|
||
import type { JsonObject, JsonValue } from '@/misc/json-value.js';
|
||
import { NoteEntityService } from '@/core/entities/NoteEntityService.js';
|
||
import type Connection from './Connection.js';
|
||
|
||
/**
|
||
* Stream channel
|
||
*/
|
||
// eslint-disable-next-line import/no-default-export
|
||
export default abstract class Channel {
|
||
protected readonly noteEntityService: NoteEntityService;
|
||
protected connection: Connection;
|
||
public id: string;
|
||
public abstract readonly chName: string;
|
||
public static readonly shouldShare: boolean;
|
||
public static readonly requireCredential: boolean;
|
||
public static readonly kind?: string | null;
|
||
|
||
protected get user() {
|
||
return this.connection.user;
|
||
}
|
||
|
||
protected get userProfile() {
|
||
return this.connection.userProfile;
|
||
}
|
||
|
||
protected get following() {
|
||
return this.connection.following;
|
||
}
|
||
|
||
protected get userIdsWhoMeMuting() {
|
||
return this.connection.userIdsWhoMeMuting;
|
||
}
|
||
|
||
protected get userIdsWhoMeMutingRenotes() {
|
||
return this.connection.userIdsWhoMeMutingRenotes;
|
||
}
|
||
|
||
protected get userIdsWhoBlockingMe() {
|
||
return this.connection.userIdsWhoBlockingMe;
|
||
}
|
||
|
||
protected get userMutedInstances() {
|
||
return this.connection.userMutedInstances;
|
||
}
|
||
|
||
protected get followingChannels() {
|
||
return this.connection.followingChannels;
|
||
}
|
||
|
||
protected get subscriber() {
|
||
return this.connection.subscriber;
|
||
}
|
||
|
||
/*
|
||
* ミュートとブロックされてるを処理する
|
||
*/
|
||
protected isNoteMutedOrBlocked(note: Packed<'Note'>): boolean {
|
||
// 流れてきたNoteがインスタンスミュートしたインスタンスが関わる
|
||
if (isInstanceMuted(note, new Set<string>(this.userProfile?.mutedInstances ?? [])) && !this.following[note.userId]) return true;
|
||
|
||
// 流れてきたNoteがミュートしているユーザーが関わる
|
||
if (isUserRelated(note, this.userIdsWhoMeMuting)) return true;
|
||
// 流れてきたNoteがブロックされているユーザーが関わる
|
||
if (isUserRelated(note, this.userIdsWhoBlockingMe)) return true;
|
||
|
||
// 流れてきたNoteがリノートをミュートしてるユーザが行ったもの
|
||
if (isRenotePacked(note) && !isQuotePacked(note) && this.userIdsWhoMeMutingRenotes.has(note.user.id)) return true;
|
||
|
||
// If it's a boost (pure renote) then we need to check the target as well
|
||
if (isPackedPureRenote(note) && note.renote && this.isNoteMutedOrBlocked(note.renote)) return true;
|
||
|
||
return false;
|
||
}
|
||
|
||
protected async hideNote(note: Packed<'Note'>): Promise<void> {
|
||
if (note.renote) {
|
||
await this.hideNote(note.renote);
|
||
}
|
||
|
||
if (note.reply) {
|
||
await this.hideNote(note.reply);
|
||
}
|
||
|
||
const meId = this.user?.id ?? null;
|
||
await this.noteEntityService.hideNote(note, meId);
|
||
}
|
||
|
||
constructor(id: string, connection: Connection, noteEntityService: NoteEntityService) {
|
||
this.id = id;
|
||
this.connection = connection;
|
||
this.noteEntityService = noteEntityService;
|
||
}
|
||
|
||
public send(payload: { type: string, body: JsonValue }): void
|
||
public send(type: string, payload: JsonValue): void
|
||
@bindThis
|
||
public send(typeOrPayload: { type: string, body: JsonValue } | string, payload?: JsonValue) {
|
||
const type = payload === undefined ? (typeOrPayload as { type: string, body: JsonValue }).type : (typeOrPayload as string);
|
||
const body = payload === undefined ? (typeOrPayload as { type: string, body: JsonValue }).body : payload;
|
||
|
||
this.connection.sendMessageToWs('channel', {
|
||
id: this.id,
|
||
type: type,
|
||
body: body,
|
||
});
|
||
}
|
||
|
||
public abstract init(params: JsonObject): void;
|
||
|
||
public dispose?(): void;
|
||
|
||
public onMessage?(type: string, body: JsonValue): void;
|
||
|
||
public async assignMyReaction(note: Packed<'Note'>): Promise<Packed<'Note'>> {
|
||
let changed = false;
|
||
// StreamingApiServerService creates a single EventEmitter per server process,
|
||
// so a new note arriving from redis gets de-serialised once per server process,
|
||
// and then that single object is passed to all active channels on each connection.
|
||
// If we didn't clone the notes here, different connections would asynchronously write
|
||
// different values to the same object, resulting in a random value being sent to each frontend. -- Dakkar
|
||
const clonedNote = { ...note };
|
||
if (this.user && isRenotePacked(note) && !isQuotePacked(note)) {
|
||
if (note.renote && Object.keys(note.renote.reactions).length > 0) {
|
||
const myReaction = await this.noteEntityService.populateMyReaction(note.renote, this.user.id);
|
||
if (myReaction) {
|
||
changed = true;
|
||
clonedNote.renote = { ...note.renote };
|
||
clonedNote.renote.myReaction = myReaction;
|
||
}
|
||
}
|
||
if (note.renote?.reply && Object.keys(note.renote.reply.reactions).length > 0) {
|
||
const myReaction = await this.noteEntityService.populateMyReaction(note.renote.reply, this.user.id);
|
||
if (myReaction) {
|
||
changed = true;
|
||
clonedNote.renote = { ...note.renote };
|
||
clonedNote.renote.reply = { ...note.renote.reply };
|
||
clonedNote.renote.reply.myReaction = myReaction;
|
||
}
|
||
}
|
||
}
|
||
if (this.user && note.reply && Object.keys(note.reply.reactions).length > 0) {
|
||
const myReaction = await this.noteEntityService.populateMyReaction(note.reply, this.user.id);
|
||
if (myReaction) {
|
||
changed = true;
|
||
clonedNote.reply = { ...note.reply };
|
||
clonedNote.reply.myReaction = myReaction;
|
||
}
|
||
}
|
||
return changed ? clonedNote : note;
|
||
}
|
||
}
|
||
|
||
export type MiChannelService<T extends boolean> = {
|
||
shouldShare: boolean;
|
||
requireCredential: T;
|
||
kind: T extends true ? string : string | null | undefined;
|
||
create: (id: string, connection: Connection) => Channel;
|
||
}
|