/* * SPDX-FileCopyrightText: syuilo and misskey-project * SPDX-License-Identifier: AGPL-3.0-only */ import { Inject, Injectable } from '@nestjs/common'; import * as Redis from 'ioredis'; import * as Reversi from 'misskey-reversi'; import type { MiChannel } from '@/models/Channel.js'; import type { MiUser } from '@/models/User.js'; import type { MiUserProfile } from '@/models/UserProfile.js'; import type { MiNote } from '@/models/Note.js'; import type { MiAntenna } from '@/models/Antenna.js'; import type { MiDriveFile } from '@/models/DriveFile.js'; import type { MiDriveFolder } from '@/models/DriveFolder.js'; import type { MiUserList } from '@/models/UserList.js'; import type { MiAbuseUserReport } from '@/models/AbuseUserReport.js'; import type { MiSignin } from '@/models/Signin.js'; import type { MiPage } from '@/models/Page.js'; import type { MiWebhook } from '@/models/Webhook.js'; import type { MiSystemWebhook } from '@/models/SystemWebhook.js'; import type { MiMeta } from '@/models/Meta.js'; import { MiAvatarDecoration, MiReversiGame, MiRole, MiRoleAssignment } from '@/models/_.js'; import type { Packed } from '@/misc/json-schema.js'; import { DI } from '@/di-symbols.js'; import type { Config } from '@/config.js'; import { bindThis } from '@/decorators.js'; import { Serialized } from '@/types.js'; import type Emitter from 'strict-event-emitter-types'; import type { EventEmitter } from 'events'; //#region Stream type-body definitions export interface BroadcastTypes { emojiAdded: { emoji: Packed<'EmojiDetailed'>; }; emojiUpdated: { emojis: Packed<'EmojiDetailed'>[]; }; emojiDeleted: { emojis: { id?: string; name: string; [other: string]: any; }[]; }; announcementCreated: { announcement: Packed<'Announcement'>; }; } export interface MainEventTypes { notification: Packed<'Notification'>; mention: Packed<'Note'>; reply: Packed<'Note'>; renote: Packed<'Note'>; follow: Packed<'UserDetailedNotMe'>; followed: Packed<'UserLite'>; unfollow: Packed<'UserDetailedNotMe'>; meUpdated: Packed<'MeDetailed'>; pageEvent: { pageId: MiPage['id']; event: string; var: any; userId: MiUser['id']; user: Packed<'UserDetailed'>; }; urlUploadFinished: { marker?: string | null; file: Packed<'DriveFile'>; }; readAllNotifications: undefined; notificationFlushed: undefined; unreadNotification: Packed<'Notification'>; unreadMention: MiNote['id']; readAllUnreadMentions: undefined; unreadSpecifiedNote: MiNote['id']; readAllUnreadSpecifiedNotes: undefined; readAllAntennas: undefined; unreadAntenna: MiAntenna; readAllAnnouncements: undefined; myTokenRegenerated: undefined; signin: { id: MiSignin['id']; createdAt: string; ip: string; headers: Record; success: boolean; }; registryUpdated: { scope?: string[]; key: string; value: any | null; }; driveFileCreated: Packed<'DriveFile'>; readAntenna: MiAntenna; receiveFollowRequest: Packed<'UserLite'>; announcementCreated: { announcement: Packed<'Announcement'>; }; } export interface DriveEventTypes { fileCreated: Packed<'DriveFile'>; fileDeleted: MiDriveFile['id']; fileUpdated: Packed<'DriveFile'>; folderCreated: Packed<'DriveFolder'>; folderDeleted: MiDriveFolder['id']; folderUpdated: Packed<'DriveFolder'>; } export interface NoteEventTypes { pollVoted: { choice: number; userId: MiUser['id']; }; deleted: { deletedAt: Date; }; updated: { cw: string | null; text: string; }; reacted: { reaction: string; emoji?: { name: string; url: string; } | null; userId: MiUser['id']; }; unreacted: { reaction: string; userId: MiUser['id']; }; } type NoteStreamEventTypes = { [key in keyof NoteEventTypes]: { id: MiNote['id']; body: NoteEventTypes[key]; }; }; export interface UserListEventTypes { userAdded: Packed<'UserLite'>; userRemoved: Packed<'UserLite'>; } export interface AntennaEventTypes { note: MiNote; } export interface RoleTimelineEventTypes { note: Packed<'Note'>; } export interface AdminEventTypes { newAbuseUserReport: { id: MiAbuseUserReport['id']; targetUserId: MiUser['id'], reporterId: MiUser['id'], comment: string; }; } export interface ReversiEventTypes { matched: { game: Packed<'ReversiGameDetailed'>; }; invited: { user: Packed<'User'>; }; } export interface ReversiGameEventTypes { changeReadyStates: { user1: boolean; user2: boolean; }; updateSettings: { userId: MiUser['id']; key: string; value: any; }; log: Reversi.Serializer.Log & { id: string | null }; started: { game: Packed<'ReversiGameDetailed'>; }; ended: { winnerId: MiUser['id'] | null; game: Packed<'ReversiGameDetailed'>; }; canceled: { userId: MiUser['id']; }; } //#endregion // 辞書(interface or type)から{ type, body }ユニオンを定義 // https://stackoverflow.com/questions/49311989/can-i-infer-the-type-of-a-value-using-extends-keyof-type // VS Codeの展開を防止するためにEvents型を定義 type Events = { [K in keyof T]: { type: K; body: T[K]; } }; type EventUnionFromDictionary< T extends object, U = Events > = U[keyof U]; type SerializedAll = { [K in keyof T]: Serialized; }; type UndefinedAsNullAll = { [K in keyof T]: T[K] extends undefined ? null : T[K]; } export interface InternalEventTypes { userChangeSuspendedState: { id: MiUser['id']; isSuspended: MiUser['isSuspended']; }; userChangeDeletedState: { id: MiUser['id']; isDeleted: MiUser['isDeleted']; }; userTokenRegenerated: { id: MiUser['id']; oldToken: string; newToken: string; }; remoteUserUpdated: { id: MiUser['id']; }; localUserUpdated: { id: MiUser['id']; }; follow: { followerId: MiUser['id']; followeeId: MiUser['id']; }; unfollow: { followerId: MiUser['id']; followeeId: MiUser['id']; }; blockingCreated: { blockerId: MiUser['id']; blockeeId: MiUser['id']; }; blockingDeleted: { blockerId: MiUser['id']; blockeeId: MiUser['id']; }; policiesUpdated: MiRole['policies']; roleCreated: MiRole; roleDeleted: MiRole; roleUpdated: MiRole; userRoleAssigned: MiRoleAssignment; userRoleUnassigned: MiRoleAssignment; webhookCreated: MiWebhook; webhookDeleted: MiWebhook; webhookUpdated: MiWebhook; systemWebhookCreated: MiSystemWebhook; systemWebhookDeleted: MiSystemWebhook; systemWebhookUpdated: MiSystemWebhook; antennaCreated: MiAntenna; antennaDeleted: MiAntenna; antennaUpdated: MiAntenna; avatarDecorationCreated: MiAvatarDecoration; avatarDecorationDeleted: MiAvatarDecoration; avatarDecorationUpdated: MiAvatarDecoration; metaUpdated: { before?: MiMeta; after: MiMeta; }; followChannel: { userId: MiUser['id']; channelId: MiChannel['id']; }; unfollowChannel: { userId: MiUser['id']; channelId: MiChannel['id']; }; updateUserProfile: MiUserProfile; mute: { muterId: MiUser['id']; muteeId: MiUser['id']; }; unmute: { muterId: MiUser['id']; muteeId: MiUser['id']; }; userListMemberAdded: { userListId: MiUserList['id']; memberId: MiUser['id']; }; userListMemberRemoved: { userListId: MiUserList['id']; memberId: MiUser['id']; }; } type EventTypesToEventPayload = EventUnionFromDictionary>>; // name/messages(spec) pairs dictionary export type GlobalEvents = { internal: { name: 'internal'; payload: EventTypesToEventPayload; }; broadcast: { name: 'broadcast'; payload: EventTypesToEventPayload; }; main: { name: `mainStream:${MiUser['id']}`; payload: EventTypesToEventPayload; }; drive: { name: `driveStream:${MiUser['id']}`; payload: EventTypesToEventPayload; }; note: { name: `noteStream:${MiNote['id']}`; payload: EventTypesToEventPayload; }; userList: { name: `userListStream:${MiUserList['id']}`; payload: EventTypesToEventPayload; }; roleTimeline: { name: `roleTimelineStream:${MiRole['id']}`; payload: EventTypesToEventPayload; }; antenna: { name: `antennaStream:${MiAntenna['id']}`; payload: EventTypesToEventPayload; }; admin: { name: `adminStream:${MiUser['id']}`; payload: EventTypesToEventPayload; }; notes: { name: 'notesStream'; payload: Serialized>; }; reversi: { name: `reversiStream:${MiUser['id']}`; payload: EventTypesToEventPayload; }; reversiGame: { name: `reversiGameStream:${MiReversiGame['id']}`; payload: EventTypesToEventPayload; }; }; // API event definitions // ストリームごとのEmitterの辞書を用意 type EventEmitterDictionary = { [x in keyof GlobalEvents]: Emitter.default void }> }; // 共用体型を交差型にする型 https://stackoverflow.com/questions/54938141/typescript-convert-union-to-intersection type UnionToIntersection = (U extends any ? (k: U) => void : never) extends ((k: infer I) => void) ? I : never; // Emitter辞書から共用体型を作り、UnionToIntersectionで交差型にする export type StreamEventEmitter = UnionToIntersection; // { [y in name]: (e: spec) => void }をまとめてその交差型をEmitterにかけるとts(2590)にひっかかる // provide stream channels union export type StreamChannels = GlobalEvents[keyof GlobalEvents]['name']; @Injectable() export class GlobalEventService { constructor( @Inject(DI.config) private config: Config, @Inject(DI.redisForPub) private redisForPub: Redis.Redis, ) { } @bindThis private publish(channel: StreamChannels, type: string | null, value?: any): void { const message = type == null ? value : value == null ? { type: type, body: null } : { type: type, body: value }; this.redisForPub.publish(this.config.host, JSON.stringify({ channel: channel, message: message, })); } @bindThis public publishInternalEvent(type: K, value?: InternalEventTypes[K]): void { this.publish('internal', type, typeof value === 'undefined' ? null : value); } @bindThis public publishBroadcastStream(type: K, value?: BroadcastTypes[K]): void { this.publish('broadcast', type, typeof value === 'undefined' ? null : value); } @bindThis public publishMainStream(userId: MiUser['id'], type: K, value?: MainEventTypes[K]): void { this.publish(`mainStream:${userId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishDriveStream(userId: MiUser['id'], type: K, value?: DriveEventTypes[K]): void { this.publish(`driveStream:${userId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishNoteStream(noteId: MiNote['id'], type: K, value?: NoteEventTypes[K]): void { this.publish(`noteStream:${noteId}`, type, { id: noteId, body: value, }); } @bindThis public publishUserListStream(listId: MiUserList['id'], type: K, value?: UserListEventTypes[K]): void { this.publish(`userListStream:${listId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishAntennaStream(antennaId: MiAntenna['id'], type: K, value?: AntennaEventTypes[K]): void { this.publish(`antennaStream:${antennaId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishRoleTimelineStream(roleId: MiRole['id'], type: K, value?: RoleTimelineEventTypes[K]): void { this.publish(`roleTimelineStream:${roleId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishNotesStream(note: Packed<'Note'>): void { this.publish('notesStream', null, note); } @bindThis public publishAdminStream(userId: MiUser['id'], type: K, value?: AdminEventTypes[K]): void { this.publish(`adminStream:${userId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishReversiStream(userId: MiUser['id'], type: K, value?: ReversiEventTypes[K]): void { this.publish(`reversiStream:${userId}`, type, typeof value === 'undefined' ? null : value); } @bindThis public publishReversiGameStream(gameId: MiReversiGame['id'], type: K, value?: ReversiGameEventTypes[K]): void { this.publish(`reversiGameStream:${gameId}`, type, typeof value === 'undefined' ? null : value); } }