diff --git a/api/src/channel/channel.service.ts b/api/src/channel/channel.service.ts index 00c987c00..9e414d7b1 100644 --- a/api/src/channel/channel.service.ts +++ b/api/src/channel/channel.service.ts @@ -13,7 +13,9 @@ import { SubscriberService } from '@/chat/services/subscriber.service'; import { CONSOLE_CHANNEL_NAME } from '@/extensions/channels/console/settings'; import { WEB_CHANNEL_NAME } from '@/extensions/channels/web/settings'; import { LoggerService } from '@/logger/logger.service'; +import { SettingService } from '@/setting/services/setting.service'; import { getSessionStore } from '@/utils/constants/session-store'; +import { ExtensionService } from '@/utils/generics/extension-service'; import { SocketGet, SocketPost, @@ -27,13 +29,27 @@ import ChannelHandler from './lib/Handler'; import { ChannelName } from './types'; @Injectable() -export class ChannelService { +export class ChannelService extends ExtensionService< + ChannelHandler +> { private registry: Map> = new Map(); constructor( - private readonly logger: LoggerService, + protected readonly logger: LoggerService, + protected readonly settingService: SettingService, private readonly subscriberService: SubscriberService, - ) {} + ) { + super(settingService, logger); + } + + /** + * Retrieves the type of extension this service manages. + * + * @returns The type of extension this service manages. + */ + public getExtensionType(): 'channel' { + return 'channel'; + } /** * Registers a channel with a specific handler. diff --git a/api/src/helper/helper.service.ts b/api/src/helper/helper.service.ts index db0a88008..e4a6fd432 100644 --- a/api/src/helper/helper.service.ts +++ b/api/src/helper/helper.service.ts @@ -10,24 +10,36 @@ import { Injectable, InternalServerErrorException } from '@nestjs/common'; import { LoggerService } from '@/logger/logger.service'; import { SettingService } from '@/setting/services/setting.service'; +import { ExtensionService } from '@/utils/generics/extension-service'; import BaseHelper from './lib/base-helper'; import { HelperName, HelperRegistry, HelperType, TypeOfHelper } from './types'; @Injectable() -export class HelperService { +export class HelperService extends ExtensionService { private registry: HelperRegistry = new Map(); constructor( - private readonly settingService: SettingService, - private readonly logger: LoggerService, + protected readonly settingService: SettingService, + protected readonly logger: LoggerService, ) { + super(settingService, logger); + // Init empty registry Object.values(HelperType).forEach((type: HelperType) => { this.registry.set(type, new Map()); }); } + /** + * Retrieves the type of extension this service manages. + * + * @returns The type of extension this service manages. + */ + protected getExtensionType(): 'helper' { + return 'helper'; + } + /** * Registers a helper. * diff --git a/api/src/plugins/plugins.service.spec.ts b/api/src/plugins/plugins.service.spec.ts index bb9f42e4f..ba34d0c1d 100644 --- a/api/src/plugins/plugins.service.spec.ts +++ b/api/src/plugins/plugins.service.spec.ts @@ -9,17 +9,27 @@ import { Test } from '@nestjs/testing'; import { LoggerModule } from '@/logger/logger.module'; +import { SettingService } from '@/setting/services/setting.service'; import { DummyPlugin } from '@/utils/test/dummy/dummy.plugin'; import { BaseBlockPlugin } from './base-block-plugin'; import { PluginService } from './plugins.service'; import { PluginType } from './types'; +// Mock services +const mockSettingService = { + get: jest.fn(), +} as unknown as SettingService; + describe('PluginsService', () => { let pluginsService: PluginService; beforeAll(async () => { const module = await Test.createTestingModule({ - providers: [PluginService, DummyPlugin], + providers: [ + PluginService, + DummyPlugin, + { provide: SettingService, useValue: mockSettingService }, + ], imports: [LoggerModule], }).compile(); pluginsService = module.get(PluginService); diff --git a/api/src/plugins/plugins.service.ts b/api/src/plugins/plugins.service.ts index 8a2667c94..c3a700a44 100644 --- a/api/src/plugins/plugins.service.ts +++ b/api/src/plugins/plugins.service.ts @@ -1,5 +1,5 @@ /* - * Copyright © 2024 Hexastack. All rights reserved. + * Copyright © 2025 Hexastack. All rights reserved. * * Licensed under the GNU Affero General Public License v3.0 (AGPLv3) with the following additional terms: * 1. The name "Hexabot" is a trademark of Hexastack. You may not use this name in derivative works without express written permission. @@ -8,6 +8,10 @@ import { Injectable, InternalServerErrorException } from '@nestjs/common'; +import { LoggerService } from '@/logger/logger.service'; +import { SettingService } from '@/setting/services/setting.service'; +import { ExtensionService } from '@/utils/generics/extension-service'; + import { BasePlugin } from './base-plugin.service'; import { PluginInstance } from './map-types'; import { PluginName, PluginType } from './types'; @@ -26,7 +30,9 @@ import { PluginName, PluginType } from './types'; * @typeparam T - The plugin type, which extends from `BasePlugin`. By default, it uses `BaseBlockPlugin`. */ @Injectable() -export class PluginService { +export class PluginService< + T extends BasePlugin = BasePlugin, +> extends ExtensionService { /** * The registry of plugins, stored as a map where the first key is the type of plugin, * the second key is the name of the plugin and the value is a plugin of type `T`. @@ -35,7 +41,21 @@ export class PluginService { Object.keys(PluginType).map((t) => [t as PluginType, new Map()]), ); - constructor() {} + constructor( + protected readonly settingService: SettingService, + protected readonly logger: LoggerService, + ) { + super(settingService, logger); + } + + /** + * Retrieves the type of extension this service manages. + * + * @returns The type of extension this service manages. + */ + public getExtensionType(): 'plugin' { + return 'plugin'; + } /** * Registers a plugin with a given name. diff --git a/api/src/setting/services/setting.service.ts b/api/src/setting/services/setting.service.ts index 56fffba3a..1c37f08d1 100644 --- a/api/src/setting/services/setting.service.ts +++ b/api/src/setting/services/setting.service.ts @@ -20,6 +20,7 @@ import { } from '@/utils/constants/cache'; import { Cacheable } from '@/utils/decorators/cacheable.decorator'; import { BaseService } from '@/utils/generics/base-service'; +import { ExtensionName } from '@/utils/types/extension'; import { SettingCreateDto } from '../dto/setting.dto'; import { SettingRepository } from '../repositories/setting.repository'; @@ -51,6 +52,15 @@ export class SettingService extends BaseService { } } + /** + * Removes all settings that belong to the provided group. + * + * @param group - The group of settings to remove. + */ + async deleteGroup(group: string) { + await this.repository.deleteMany({ group }); + } + /** * Loads all settings and returns them grouped in ascending order by weight. * @@ -100,6 +110,29 @@ export class SettingService extends BaseService { ); } + /** + * + * Retrieves a list of all setting groups for a specific extension type. + * + * @param extensionType - The extension type to filter by. + * + * @returns A promise that resolves to a list of setting groups. + */ + async getExtensionSettings( + extensionType: 'channel' | 'plugin' | 'helper', + ): Promise[]> { + const settings = await this.find({ + group: { $regex: `_${extensionType}$` }, + }); + const groups = settings.reduce((acc, setting) => { + if (!acc.includes(setting.group)) { + acc.push(setting.group); + } + return acc; + }, []) as HyphenToUnderscore[]; + return groups; + } + /** * Retrieves the application configuration object. * diff --git a/api/src/utils/generics/extension-service.ts b/api/src/utils/generics/extension-service.ts new file mode 100644 index 000000000..6d9b537f2 --- /dev/null +++ b/api/src/utils/generics/extension-service.ts @@ -0,0 +1,62 @@ +/* + * Copyright © 2025 Hexastack. All rights reserved. + * + * Licensed under the GNU Affero General Public License v3.0 (AGPLv3) with the following additional terms: + * 1. The name "Hexabot" is a trademark of Hexastack. You may not use this name in derivative works without express written permission. + * 2. All derivative works must include clear attribution to the original creator and software, Hexastack and Hexabot, in a prominent location (e.g., in the software's "About" section, documentation, and README file). + */ + +import { OnApplicationBootstrap } from '@nestjs/common'; + +import { LoggerService } from '@/logger/logger.service'; +import { SettingService } from '@/setting/services/setting.service'; + +export abstract class ExtensionService< + T extends { getNamespace(): string; getName(): string }, +> implements OnApplicationBootstrap +{ + constructor( + protected readonly settingService: SettingService, + protected readonly logger: LoggerService, + ) {} + + async onApplicationBootstrap(): Promise { + await this.cleanup(this.getExtensionType()); + } + + /** + * Cleanups the unregistered extensions from settings. + * + * @param extensionType - The type of extension (e.g., 'plugin', 'helper', 'channel'). + */ + async cleanup(extensionType: 'channel' | 'plugin' | 'helper'): Promise { + const activeExtensions = this.getAll().map((handler) => + handler.getNamespace(), + ); + + const orphanSettings = ( + await this.settingService.getExtensionSettings(extensionType) + ).filter((group) => !activeExtensions.includes(group)); + + await Promise.all( + orphanSettings.map(async (group) => { + this.logger.log(`Deleting orphaned settings for ${group}...`); + return this.settingService.deleteGroup(group); + }), + ); + } + + /** + * Retrieves all registered extensions as an array. + * + * @returns An array containing all the registered extensions. + */ + public abstract getAll(): T[]; + + /** + * Abstract method to get the type of extension this service manages. + * + * @returns The type of extension (e.g., 'plugin', 'helper', 'channel'). + */ + protected abstract getExtensionType(): 'channel' | 'plugin' | 'helper'; +}