import { BullModule } from '@nestjs/bullmq'; import { Inject, Module, OnModuleDestroy, OnModuleInit, ValidationPipe } from '@nestjs/common'; import { APP_FILTER, APP_GUARD, APP_INTERCEPTOR, APP_PIPE } from '@nestjs/core'; import { ScheduleModule, SchedulerRegistry } from '@nestjs/schedule'; import { ClsModule } from 'nestjs-cls'; import { KyselyModule } from 'nestjs-kysely'; import { OpenTelemetryModule } from 'nestjs-otel'; import { commandsAndQuestions } from 'src/commands'; import { IWorker } from 'src/constants'; import { controllers } from 'src/controllers'; import { StorageCore } from 'src/cores/storage.core'; import { ImmichWorker, SystemMetadataKey } from 'src/enum'; import { MaintenanceAuthGuard } from 'src/maintenance/maintenance-auth.guard'; import { MaintenanceEphemeralStateRepository } from 'src/maintenance/maintenance-ephemeral-state.repository'; import { MaintenanceWebsocketRepository } from 'src/maintenance/maintenance-websocket.repository'; import { MaintenanceWorkerController } from 'src/maintenance/maintenance-worker.controller'; import { MaintenanceWorkerService } from 'src/maintenance/maintenance-worker.service'; import { AuthGuard } from 'src/middleware/auth.guard'; import { ErrorInterceptor } from 'src/middleware/error.interceptor'; import { FileUploadInterceptor } from 'src/middleware/file-upload.interceptor'; import { GlobalExceptionFilter } from 'src/middleware/global-exception.filter'; import { LoggingInterceptor } from 'src/middleware/logging.interceptor'; import { repositories } from 'src/repositories'; import { AppRepository } from 'src/repositories/app.repository'; import { ConfigRepository } from 'src/repositories/config.repository'; import { DatabaseRepository } from 'src/repositories/database.repository'; import { EventRepository } from 'src/repositories/event.repository'; import { LoggingRepository } from 'src/repositories/logging.repository'; import { ProcessRepository } from 'src/repositories/process.repository'; import { StorageRepository } from 'src/repositories/storage.repository'; import { SystemMetadataRepository } from 'src/repositories/system-metadata.repository'; import { teardownTelemetry, TelemetryRepository } from 'src/repositories/telemetry.repository'; import { WebsocketRepository } from 'src/repositories/websocket.repository'; import { services } from 'src/services'; import { AuthService } from 'src/services/auth.service'; import { CliService } from 'src/services/cli.service'; import { QueueService } from 'src/services/queue.service'; import { MaintenanceModeState } from 'src/types'; import { getKyselyConfig } from 'src/utils/database'; const common = [...repositories, ...services, GlobalExceptionFilter]; const commonMiddleware = [ { provide: APP_FILTER, useClass: GlobalExceptionFilter }, { provide: APP_PIPE, useValue: new ValidationPipe({ transform: true, whitelist: true }) }, { provide: APP_INTERCEPTOR, useClass: LoggingInterceptor }, { provide: APP_INTERCEPTOR, useClass: ErrorInterceptor }, ]; const apiMiddleware = [FileUploadInterceptor, ...commonMiddleware, { provide: APP_GUARD, useClass: AuthGuard }]; const configRepository = new ConfigRepository(); const { bull, cls, database, otel } = configRepository.getEnv(); const commonImports = [ ClsModule.forRoot(cls.config), KyselyModule.forRoot(getKyselyConfig(database.config)), OpenTelemetryModule.forRoot(otel), ]; const bullImports = [BullModule.forRoot(bull.config), BullModule.registerQueue(...bull.queues)]; export class BaseModule implements OnModuleInit, OnModuleDestroy { constructor( @Inject(IWorker) private worker: ImmichWorker, logger: LoggingRepository, private authService: AuthService, private eventRepository: EventRepository, private queueService: QueueService, private telemetryRepository: TelemetryRepository, private websocketRepository: WebsocketRepository, ) { logger.setAppName(this.worker); } async onModuleInit() { this.telemetryRepository.setup({ repositories }); this.queueService.setServices(services); this.websocketRepository.setAuthFn(async (client) => this.authService.authenticate({ headers: client.request.headers, queryParams: {}, metadata: { adminRoute: false, sharedLinkRoute: false, uri: '/api/socket.io' }, }), ); this.eventRepository.setup({ services }); await this.eventRepository.emit('AppBootstrap'); } async onModuleDestroy() { await this.eventRepository.emit('AppShutdown'); await teardownTelemetry(); } } @Module({ imports: [...bullImports, ...commonImports, ScheduleModule.forRoot()], controllers: [...controllers], providers: [...common, ...apiMiddleware, { provide: IWorker, useValue: ImmichWorker.Api }], }) export class ApiModule extends BaseModule {} @Module({ imports: [...commonImports], controllers: [MaintenanceWorkerController], providers: [ ConfigRepository, LoggingRepository, StorageRepository, ProcessRepository, DatabaseRepository, SystemMetadataRepository, AppRepository, MaintenanceWebsocketRepository, MaintenanceEphemeralStateRepository, MaintenanceWorkerService, ...commonMiddleware, { provide: APP_GUARD, useClass: MaintenanceAuthGuard }, { provide: IWorker, useValue: ImmichWorker.Maintenance }, ], }) export class MaintenanceModule { constructor( @Inject(IWorker) private worker: ImmichWorker, logger: LoggingRepository, private systemMetadataRepository: SystemMetadataRepository, private maintenanceWorkerService: MaintenanceWorkerService, private maintenanceWebsocketRepository: MaintenanceWebsocketRepository, private maintenanceEphemeralStateRepository: MaintenanceEphemeralStateRepository, ) { logger.setAppName(this.worker); } async onModuleInit() { const state = (await this.systemMetadataRepository.get( SystemMetadataKey.MaintenanceMode, )) as MaintenanceModeState & { isMaintenanceMode: true }; this.maintenanceEphemeralStateRepository.setSecret(state.secret); StorageCore.setMediaLocation(this.maintenanceWorkerService.detectMediaLocation()); this.maintenanceWebsocketRepository.setAuthFn(async (client) => this.maintenanceWorkerService.authenticate(client.request.headers), ); this.maintenanceWebsocketRepository.setStatusUpdateFn((status) => this.maintenanceEphemeralStateRepository.setStatus(status), ); await this.maintenanceWorkerService.logSecret(); } } @Module({ imports: [...bullImports, ...commonImports], providers: [...common, { provide: IWorker, useValue: ImmichWorker.Microservices }, SchedulerRegistry], }) export class MicroservicesModule extends BaseModule {} @Module({ imports: [...bullImports, ...commonImports], providers: [...common, ...commandsAndQuestions, SchedulerRegistry], }) export class ImmichAdminModule implements OnModuleDestroy { constructor(private service: CliService) {} async onModuleDestroy() { await this.service.cleanup(); } }