2023-12-14 17:55:40 +01:00
|
|
|
import { LogLevel, SystemConfig } from '@app/infra/entities';
|
|
|
|
import { ImmichLogger } from '@app/infra/logger';
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
|
|
import { instanceToPlain } from 'class-transformer';
|
|
|
|
import _ from 'lodash';
|
2023-12-08 17:15:46 +01:00
|
|
|
import {
|
2023-12-13 18:23:51 +01:00
|
|
|
ClientEvent,
|
2023-12-08 17:15:46 +01:00
|
|
|
ICommunicationRepository,
|
2024-02-13 02:50:47 +01:00
|
|
|
ISearchRepository,
|
2023-12-08 17:15:46 +01:00
|
|
|
ISystemConfigRepository,
|
2023-12-13 18:23:51 +01:00
|
|
|
ServerEvent,
|
2023-12-08 17:15:46 +01:00
|
|
|
} from '../repositories';
|
2023-09-04 21:45:59 +02:00
|
|
|
import { SystemConfigDto, mapConfig } from './dto/system-config.dto';
|
2023-06-16 21:54:17 +02:00
|
|
|
import { SystemConfigTemplateStorageOptionDto } from './response-dto/system-config-template-storage-option.dto';
|
2023-01-21 17:11:55 +01:00
|
|
|
import {
|
|
|
|
supportedDayTokens,
|
|
|
|
supportedHourTokens,
|
|
|
|
supportedMinuteTokens,
|
|
|
|
supportedMonthTokens,
|
|
|
|
supportedPresetTokens,
|
|
|
|
supportedSecondTokens,
|
2023-09-28 19:47:31 +02:00
|
|
|
supportedWeekTokens,
|
2023-01-21 17:11:55 +01:00
|
|
|
supportedYearTokens,
|
2023-01-24 05:13:42 +01:00
|
|
|
} from './system-config.constants';
|
2023-01-21 17:11:55 +01:00
|
|
|
import { SystemConfigCore, SystemConfigValidator } from './system-config.core';
|
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class SystemConfigService {
|
2023-12-14 17:55:40 +01:00
|
|
|
private logger = new ImmichLogger(SystemConfigService.name);
|
2023-01-21 17:11:55 +01:00
|
|
|
private core: SystemConfigCore;
|
2023-12-13 18:23:51 +01:00
|
|
|
|
2023-01-21 17:11:55 +01:00
|
|
|
constructor(
|
2023-11-09 17:10:56 +01:00
|
|
|
@Inject(ISystemConfigRepository) private repository: ISystemConfigRepository,
|
2023-10-06 09:01:14 +02:00
|
|
|
@Inject(ICommunicationRepository) private communicationRepository: ICommunicationRepository,
|
2024-02-13 02:50:47 +01:00
|
|
|
@Inject(ISearchRepository) private smartInfoRepository: ISearchRepository,
|
2023-01-21 17:11:55 +01:00
|
|
|
) {
|
2023-10-09 02:51:03 +02:00
|
|
|
this.core = SystemConfigCore.create(repository);
|
2023-12-13 18:23:51 +01:00
|
|
|
this.communicationRepository.on(ServerEvent.CONFIG_UPDATE, () => this.handleConfigUpdate());
|
2023-12-14 17:55:40 +01:00
|
|
|
this.core.config$.subscribe((config) => this.setLogLevel(config));
|
|
|
|
this.core.addValidator((newConfig, oldConfig) => this.validateConfig(newConfig, oldConfig));
|
|
|
|
}
|
|
|
|
|
|
|
|
async init() {
|
|
|
|
const config = await this.core.getConfig();
|
2024-01-01 19:16:44 +01:00
|
|
|
this.config$.next(config);
|
2023-01-21 17:11:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
get config$() {
|
|
|
|
return this.core.config$;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getConfig(): Promise<SystemConfigDto> {
|
|
|
|
const config = await this.core.getConfig();
|
|
|
|
return mapConfig(config);
|
|
|
|
}
|
|
|
|
|
|
|
|
getDefaults(): SystemConfigDto {
|
|
|
|
const config = this.core.getDefaults();
|
|
|
|
return mapConfig(config);
|
|
|
|
}
|
|
|
|
|
|
|
|
async updateConfig(dto: SystemConfigDto): Promise<SystemConfigDto> {
|
2023-12-08 17:15:46 +01:00
|
|
|
const oldConfig = await this.core.getConfig();
|
|
|
|
const newConfig = await this.core.updateConfig(dto);
|
2023-12-13 18:23:51 +01:00
|
|
|
|
|
|
|
this.communicationRepository.broadcast(ClientEvent.CONFIG_UPDATE, {});
|
|
|
|
this.communicationRepository.sendServerEvent(ServerEvent.CONFIG_UPDATE);
|
|
|
|
|
2023-12-08 17:15:46 +01:00
|
|
|
if (oldConfig.machineLearning.clip.modelName !== newConfig.machineLearning.clip.modelName) {
|
|
|
|
await this.smartInfoRepository.init(newConfig.machineLearning.clip.modelName);
|
|
|
|
}
|
|
|
|
return mapConfig(newConfig);
|
2023-01-21 17:11:55 +01:00
|
|
|
}
|
|
|
|
|
2023-12-13 18:23:51 +01:00
|
|
|
// this is only used by the cli on config change, and it's not actually needed anymore
|
2023-01-21 17:11:55 +01:00
|
|
|
async refreshConfig() {
|
2023-12-13 18:23:51 +01:00
|
|
|
this.communicationRepository.sendServerEvent(ServerEvent.CONFIG_UPDATE);
|
2023-01-21 17:11:55 +01:00
|
|
|
await this.core.refreshConfig();
|
2023-05-26 21:43:24 +02:00
|
|
|
return true;
|
2023-01-21 17:11:55 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
addValidator(validator: SystemConfigValidator) {
|
|
|
|
this.core.addValidator(validator);
|
|
|
|
}
|
|
|
|
|
|
|
|
getStorageTemplateOptions(): SystemConfigTemplateStorageOptionDto {
|
|
|
|
const options = new SystemConfigTemplateStorageOptionDto();
|
|
|
|
|
|
|
|
options.dayOptions = supportedDayTokens;
|
2023-09-28 19:47:31 +02:00
|
|
|
options.weekOptions = supportedWeekTokens;
|
2023-01-21 17:11:55 +01:00
|
|
|
options.monthOptions = supportedMonthTokens;
|
|
|
|
options.yearOptions = supportedYearTokens;
|
|
|
|
options.hourOptions = supportedHourTokens;
|
|
|
|
options.secondOptions = supportedSecondTokens;
|
|
|
|
options.minuteOptions = supportedMinuteTokens;
|
|
|
|
options.presetOptions = supportedPresetTokens;
|
|
|
|
|
|
|
|
return options;
|
|
|
|
}
|
2023-11-09 17:10:56 +01:00
|
|
|
|
|
|
|
async getMapStyle(theme: 'light' | 'dark') {
|
|
|
|
const { map } = await this.getConfig();
|
|
|
|
const styleUrl = theme === 'dark' ? map.darkStyle : map.lightStyle;
|
|
|
|
|
|
|
|
if (styleUrl) {
|
|
|
|
return this.repository.fetchStyle(styleUrl);
|
|
|
|
}
|
|
|
|
|
2023-11-25 19:53:30 +01:00
|
|
|
return JSON.parse(await this.repository.readFile(`./resources/style-${theme}.json`));
|
2023-11-09 17:10:56 +01:00
|
|
|
}
|
2023-11-18 05:13:36 +01:00
|
|
|
|
|
|
|
async getCustomCss(): Promise<string> {
|
|
|
|
const { theme } = await this.core.getConfig();
|
|
|
|
return theme.customCss;
|
|
|
|
}
|
2023-12-13 18:23:51 +01:00
|
|
|
|
|
|
|
private async handleConfigUpdate() {
|
|
|
|
await this.core.refreshConfig();
|
|
|
|
}
|
2023-12-14 17:55:40 +01:00
|
|
|
|
|
|
|
private async setLogLevel({ logging }: SystemConfig) {
|
|
|
|
const envLevel = this.getEnvLogLevel();
|
|
|
|
const configLevel = logging.enabled ? logging.level : false;
|
2024-02-02 04:18:00 +01:00
|
|
|
const level = envLevel ?? configLevel;
|
2023-12-14 17:55:40 +01:00
|
|
|
ImmichLogger.setLogLevel(level);
|
|
|
|
this.logger.log(`LogLevel=${level} ${envLevel ? '(set via LOG_LEVEL)' : '(set via system config)'}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
private getEnvLogLevel() {
|
|
|
|
return process.env.LOG_LEVEL as LogLevel;
|
|
|
|
}
|
|
|
|
|
|
|
|
private async validateConfig(newConfig: SystemConfig, oldConfig: SystemConfig) {
|
|
|
|
if (!_.isEqual(instanceToPlain(newConfig.logging), oldConfig.logging) && this.getEnvLogLevel()) {
|
|
|
|
throw new Error('Logging cannot be changed while the environment variable LOG_LEVEL is set.');
|
|
|
|
}
|
|
|
|
}
|
2023-01-21 17:11:55 +01:00
|
|
|
}
|