feat: 通報を受けた際にメールまたはWebhookで通知を送出出来るようにする (#13758)
* feat: 通報を受けた際にメールまたはWebhookで通知を送出出来るようにする * モデログに対応&エンドポイントを単一オブジェクトでのサポートに変更(API経由で大量に作るシチュエーションもないと思うので) * fix spdx * fix migration * fix migration * fix models * add e2e webhook * tweak * fix modlog * fix bugs * add tests and fix bugs * add tests and fix bugs * add tests * fix path * regenerate locale * 混入除去 * 混入除去 * add abuseReportResolved * fix pnpm-lock.yaml * add abuseReportResolved test * fix bugs * fix ui * add tests * fix CHANGELOG.md * add tests * add RoleService.getModeratorIds tests * WebhookServiceをUserとSystemに分割 * fix CHANGELOG.md * fix test * insertOneを使う用に * fix * regenerate locales * revert version * separate webhook job queue * fix * 🎨 * Update QueueProcessorService.ts --------- Co-authored-by: osamu <46447427+sam-osamu@users.noreply.github.com> Co-authored-by: syuilo <4439005+syuilo@users.noreply.github.com>
This commit is contained in:
		
							
								
								
									
										233
									
								
								packages/backend/src/core/SystemWebhookService.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										233
									
								
								packages/backend/src/core/SystemWebhookService.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,233 @@ | ||||
| /* | ||||
|  * 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 type { MiUser, SystemWebhooksRepository } from '@/models/_.js'; | ||||
| import { DI } from '@/di-symbols.js'; | ||||
| import { bindThis } from '@/decorators.js'; | ||||
| import { GlobalEvents, GlobalEventService } from '@/core/GlobalEventService.js'; | ||||
| import { MiSystemWebhook, type SystemWebhookEventType } from '@/models/SystemWebhook.js'; | ||||
| import { IdService } from '@/core/IdService.js'; | ||||
| import { QueueService } from '@/core/QueueService.js'; | ||||
| import { ModerationLogService } from '@/core/ModerationLogService.js'; | ||||
| import { LoggerService } from '@/core/LoggerService.js'; | ||||
| import Logger from '@/logger.js'; | ||||
| import type { OnApplicationShutdown } from '@nestjs/common'; | ||||
|  | ||||
| @Injectable() | ||||
| export class SystemWebhookService implements OnApplicationShutdown { | ||||
| 	private logger: Logger; | ||||
| 	private activeSystemWebhooksFetched = false; | ||||
| 	private activeSystemWebhooks: MiSystemWebhook[] = []; | ||||
|  | ||||
| 	constructor( | ||||
| 		@Inject(DI.redisForSub) | ||||
| 		private redisForSub: Redis.Redis, | ||||
| 		@Inject(DI.systemWebhooksRepository) | ||||
| 		private systemWebhooksRepository: SystemWebhooksRepository, | ||||
| 		private idService: IdService, | ||||
| 		private queueService: QueueService, | ||||
| 		private moderationLogService: ModerationLogService, | ||||
| 		private loggerService: LoggerService, | ||||
| 		private globalEventService: GlobalEventService, | ||||
| 	) { | ||||
| 		this.redisForSub.on('message', this.onMessage); | ||||
| 		this.logger = this.loggerService.getLogger('webhook'); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public async fetchActiveSystemWebhooks() { | ||||
| 		if (!this.activeSystemWebhooksFetched) { | ||||
| 			this.activeSystemWebhooks = await this.systemWebhooksRepository.findBy({ | ||||
| 				isActive: true, | ||||
| 			}); | ||||
| 			this.activeSystemWebhooksFetched = true; | ||||
| 		} | ||||
|  | ||||
| 		return this.activeSystemWebhooks; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * SystemWebhook の一覧を取得する. | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async fetchSystemWebhooks(params?: { | ||||
| 		ids?: MiSystemWebhook['id'][]; | ||||
| 		isActive?: MiSystemWebhook['isActive']; | ||||
| 		on?: MiSystemWebhook['on']; | ||||
| 	}): Promise<MiSystemWebhook[]> { | ||||
| 		const query = this.systemWebhooksRepository.createQueryBuilder('systemWebhook'); | ||||
| 		if (params) { | ||||
| 			if (params.ids && params.ids.length > 0) { | ||||
| 				query.andWhere('systemWebhook.id IN (:...ids)', { ids: params.ids }); | ||||
| 			} | ||||
| 			if (params.isActive !== undefined) { | ||||
| 				query.andWhere('systemWebhook.isActive = :isActive', { isActive: params.isActive }); | ||||
| 			} | ||||
| 			if (params.on && params.on.length > 0) { | ||||
| 				query.andWhere(':on <@ systemWebhook.on', { on: params.on }); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return query.getMany(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * SystemWebhook を作成する. | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async createSystemWebhook( | ||||
| 		params: { | ||||
| 			isActive: MiSystemWebhook['isActive']; | ||||
| 			name: MiSystemWebhook['name']; | ||||
| 			on: MiSystemWebhook['on']; | ||||
| 			url: MiSystemWebhook['url']; | ||||
| 			secret: MiSystemWebhook['secret']; | ||||
| 		}, | ||||
| 		updater: MiUser, | ||||
| 	): Promise<MiSystemWebhook> { | ||||
| 		const id = this.idService.gen(); | ||||
| 		await this.systemWebhooksRepository.insert({ | ||||
| 			...params, | ||||
| 			id, | ||||
| 		}); | ||||
|  | ||||
| 		const webhook = await this.systemWebhooksRepository.findOneByOrFail({ id }); | ||||
| 		this.globalEventService.publishInternalEvent('systemWebhookCreated', webhook); | ||||
| 		this.moderationLogService | ||||
| 			.log(updater, 'createSystemWebhook', { | ||||
| 				systemWebhookId: webhook.id, | ||||
| 				webhook: webhook, | ||||
| 			}) | ||||
| 			.then(); | ||||
|  | ||||
| 		return webhook; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * SystemWebhook を更新する. | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async updateSystemWebhook( | ||||
| 		params: { | ||||
| 			id: MiSystemWebhook['id']; | ||||
| 			isActive: MiSystemWebhook['isActive']; | ||||
| 			name: MiSystemWebhook['name']; | ||||
| 			on: MiSystemWebhook['on']; | ||||
| 			url: MiSystemWebhook['url']; | ||||
| 			secret: MiSystemWebhook['secret']; | ||||
| 		}, | ||||
| 		updater: MiUser, | ||||
| 	): Promise<MiSystemWebhook> { | ||||
| 		const beforeEntity = await this.systemWebhooksRepository.findOneByOrFail({ id: params.id }); | ||||
| 		await this.systemWebhooksRepository.update(beforeEntity.id, { | ||||
| 			updatedAt: new Date(), | ||||
| 			isActive: params.isActive, | ||||
| 			name: params.name, | ||||
| 			on: params.on, | ||||
| 			url: params.url, | ||||
| 			secret: params.secret, | ||||
| 		}); | ||||
|  | ||||
| 		const afterEntity = await this.systemWebhooksRepository.findOneByOrFail({ id: beforeEntity.id }); | ||||
| 		this.globalEventService.publishInternalEvent('systemWebhookUpdated', afterEntity); | ||||
| 		this.moderationLogService | ||||
| 			.log(updater, 'updateSystemWebhook', { | ||||
| 				systemWebhookId: beforeEntity.id, | ||||
| 				before: beforeEntity, | ||||
| 				after: afterEntity, | ||||
| 			}) | ||||
| 			.then(); | ||||
|  | ||||
| 		return afterEntity; | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * SystemWebhook を削除する. | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async deleteSystemWebhook(id: MiSystemWebhook['id'], updater: MiUser) { | ||||
| 		const webhook = await this.systemWebhooksRepository.findOneByOrFail({ id }); | ||||
| 		await this.systemWebhooksRepository.delete(id); | ||||
|  | ||||
| 		this.globalEventService.publishInternalEvent('systemWebhookDeleted', webhook); | ||||
| 		this.moderationLogService | ||||
| 			.log(updater, 'deleteSystemWebhook', { | ||||
| 				systemWebhookId: webhook.id, | ||||
| 				webhook, | ||||
| 			}) | ||||
| 			.then(); | ||||
| 	} | ||||
|  | ||||
| 	/** | ||||
| 	 * SystemWebhook をWebhook配送キューに追加する | ||||
| 	 * @see QueueService.systemWebhookDeliver | ||||
| 	 */ | ||||
| 	@bindThis | ||||
| 	public async enqueueSystemWebhook(webhook: MiSystemWebhook | MiSystemWebhook['id'], type: SystemWebhookEventType, content: unknown) { | ||||
| 		const webhookEntity = typeof webhook === 'string' | ||||
| 			? (await this.fetchActiveSystemWebhooks()).find(a => a.id === webhook) | ||||
| 			: webhook; | ||||
| 		if (!webhookEntity || !webhookEntity.isActive) { | ||||
| 			this.logger.info(`Webhook is not active or not found : ${webhook}`); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		if (!webhookEntity.on.includes(type)) { | ||||
| 			this.logger.info(`Webhook ${webhookEntity.id} is not listening to ${type}`); | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		return this.queueService.systemWebhookDeliver(webhookEntity, type, content); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	private async onMessage(_: string, data: string): Promise<void> { | ||||
| 		const obj = JSON.parse(data); | ||||
| 		if (obj.channel !== 'internal') { | ||||
| 			return; | ||||
| 		} | ||||
|  | ||||
| 		const { type, body } = obj.message as GlobalEvents['internal']['payload']; | ||||
| 		switch (type) { | ||||
| 			case 'systemWebhookCreated': { | ||||
| 				if (body.isActive) { | ||||
| 					this.activeSystemWebhooks.push(MiSystemWebhook.deserialize(body)); | ||||
| 				} | ||||
| 				break; | ||||
| 			} | ||||
| 			case 'systemWebhookUpdated': { | ||||
| 				if (body.isActive) { | ||||
| 					const i = this.activeSystemWebhooks.findIndex(a => a.id === body.id); | ||||
| 					if (i > -1) { | ||||
| 						this.activeSystemWebhooks[i] = MiSystemWebhook.deserialize(body); | ||||
| 					} else { | ||||
| 						this.activeSystemWebhooks.push(MiSystemWebhook.deserialize(body)); | ||||
| 					} | ||||
| 				} else { | ||||
| 					this.activeSystemWebhooks = this.activeSystemWebhooks.filter(a => a.id !== body.id); | ||||
| 				} | ||||
| 				break; | ||||
| 			} | ||||
| 			case 'systemWebhookDeleted': { | ||||
| 				this.activeSystemWebhooks = this.activeSystemWebhooks.filter(a => a.id !== body.id); | ||||
| 				break; | ||||
| 			} | ||||
| 			default: | ||||
| 				break; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public dispose(): void { | ||||
| 		this.redisForSub.off('message', this.onMessage); | ||||
| 	} | ||||
|  | ||||
| 	@bindThis | ||||
| 	public onApplicationShutdown(signal?: string | undefined): void { | ||||
| 		this.dispose(); | ||||
| 	} | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	 おさむのひと
					おさむのひと