2023-06-01 12:32:51 +02:00
|
|
|
import {
|
2023-07-09 04:43:11 +02:00
|
|
|
AudioCodec,
|
2023-09-03 08:21:51 +02:00
|
|
|
Colorspace,
|
2023-10-24 17:05:42 +02:00
|
|
|
CQMode,
|
2023-12-14 17:55:40 +01:00
|
|
|
LogLevel,
|
2023-06-01 12:32:51 +02:00
|
|
|
SystemConfig,
|
|
|
|
SystemConfigEntity,
|
|
|
|
SystemConfigKey,
|
|
|
|
SystemConfigValue,
|
2023-08-07 22:35:25 +02:00
|
|
|
ToneMapping,
|
2023-08-02 03:56:10 +02:00
|
|
|
TranscodeHWAccel,
|
2023-07-09 04:43:11 +02:00
|
|
|
TranscodePolicy,
|
|
|
|
VideoCodec,
|
2023-06-01 12:32:51 +02:00
|
|
|
} from '@app/infra/entities';
|
2023-12-14 17:55:40 +01:00
|
|
|
import { ImmichLogger } from '@app/infra/logger';
|
|
|
|
import { BadRequestException, ForbiddenException, Injectable } from '@nestjs/common';
|
2023-10-31 21:19:12 +01:00
|
|
|
import { CronExpression } from '@nestjs/schedule';
|
2023-10-22 17:14:32 +02:00
|
|
|
import { plainToInstance } from 'class-transformer';
|
2023-08-25 19:44:52 +02:00
|
|
|
import { validate } from 'class-validator';
|
2022-12-09 21:51:42 +01:00
|
|
|
import * as _ from 'lodash';
|
2022-12-16 21:26:12 +01:00
|
|
|
import { Subject } from 'rxjs';
|
2023-06-01 12:32:51 +02:00
|
|
|
import { QueueName } from '../job/job.constants';
|
2023-10-09 16:25:03 +02:00
|
|
|
import { ISystemConfigRepository } from '../repositories';
|
2023-08-25 19:44:52 +02:00
|
|
|
import { SystemConfigDto } from './dto';
|
2022-12-09 21:51:42 +01:00
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
export type SystemConfigValidator = (config: SystemConfig, newConfig: SystemConfig) => void | Promise<void>;
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-07-15 06:03:56 +02:00
|
|
|
export const defaults = Object.freeze<SystemConfig>({
|
2022-12-09 21:51:42 +01:00
|
|
|
ffmpeg: {
|
2023-05-22 20:07:43 +02:00
|
|
|
crf: 23,
|
|
|
|
threads: 0,
|
2022-12-09 21:51:42 +01:00
|
|
|
preset: 'ultrafast',
|
2023-07-09 04:43:11 +02:00
|
|
|
targetVideoCodec: VideoCodec.H264,
|
|
|
|
targetAudioCodec: AudioCodec.AAC,
|
2023-04-04 03:42:53 +02:00
|
|
|
targetResolution: '720',
|
2023-05-22 20:07:43 +02:00
|
|
|
maxBitrate: '0',
|
2023-09-03 03:22:42 +02:00
|
|
|
bframes: -1,
|
|
|
|
refs: 0,
|
|
|
|
gopSize: 0,
|
|
|
|
npl: 0,
|
|
|
|
temporalAQ: false,
|
|
|
|
cqMode: CQMode.AUTO,
|
2023-05-22 20:07:43 +02:00
|
|
|
twoPass: false,
|
2023-07-09 04:43:11 +02:00
|
|
|
transcode: TranscodePolicy.REQUIRED,
|
2023-08-07 22:35:25 +02:00
|
|
|
tonemap: ToneMapping.HABLE,
|
2023-08-02 03:56:10 +02:00
|
|
|
accel: TranscodeHWAccel.DISABLED,
|
2022-11-15 05:39:32 +01:00
|
|
|
},
|
2023-06-01 12:32:51 +02:00
|
|
|
job: {
|
|
|
|
[QueueName.BACKGROUND_TASK]: { concurrency: 5 },
|
|
|
|
[QueueName.CLIP_ENCODING]: { concurrency: 2 },
|
|
|
|
[QueueName.METADATA_EXTRACTION]: { concurrency: 5 },
|
|
|
|
[QueueName.OBJECT_TAGGING]: { concurrency: 2 },
|
|
|
|
[QueueName.RECOGNIZE_FACES]: { concurrency: 2 },
|
|
|
|
[QueueName.SEARCH]: { concurrency: 5 },
|
|
|
|
[QueueName.SIDECAR]: { concurrency: 5 },
|
2023-10-11 00:59:13 +02:00
|
|
|
[QueueName.LIBRARY]: { concurrency: 5 },
|
2023-06-01 12:32:51 +02:00
|
|
|
[QueueName.STORAGE_TEMPLATE_MIGRATION]: { concurrency: 5 },
|
2023-09-25 17:07:21 +02:00
|
|
|
[QueueName.MIGRATION]: { concurrency: 5 },
|
2023-06-01 12:32:51 +02:00
|
|
|
[QueueName.THUMBNAIL_GENERATION]: { concurrency: 5 },
|
|
|
|
[QueueName.VIDEO_CONVERSION]: { concurrency: 1 },
|
|
|
|
},
|
2023-12-14 17:55:40 +01:00
|
|
|
logging: {
|
|
|
|
enabled: true,
|
|
|
|
level: LogLevel.LOG,
|
|
|
|
},
|
2023-08-25 06:15:03 +02:00
|
|
|
machineLearning: {
|
|
|
|
enabled: process.env.IMMICH_MACHINE_LEARNING_ENABLED !== 'false',
|
|
|
|
url: process.env.IMMICH_MACHINE_LEARNING_URL || 'http://immich-machine-learning:3003',
|
2023-08-29 15:58:00 +02:00
|
|
|
classification: {
|
|
|
|
enabled: true,
|
|
|
|
modelName: 'microsoft/resnet-50',
|
|
|
|
minScore: 0.9,
|
|
|
|
},
|
|
|
|
clip: {
|
|
|
|
enabled: true,
|
2023-10-31 11:02:04 +01:00
|
|
|
modelName: 'ViT-B-32__openai',
|
2023-08-29 15:58:00 +02:00
|
|
|
},
|
|
|
|
facialRecognition: {
|
|
|
|
enabled: true,
|
|
|
|
modelName: 'buffalo_l',
|
|
|
|
minScore: 0.7,
|
|
|
|
maxDistance: 0.6,
|
2023-09-18 06:05:35 +02:00
|
|
|
minFaces: 1,
|
2023-08-29 15:58:00 +02:00
|
|
|
},
|
2023-08-25 06:15:03 +02:00
|
|
|
},
|
2023-09-09 04:51:46 +02:00
|
|
|
map: {
|
|
|
|
enabled: true,
|
2023-11-09 17:10:56 +01:00
|
|
|
lightStyle: '',
|
|
|
|
darkStyle: '',
|
2023-09-09 04:51:46 +02:00
|
|
|
},
|
2023-09-26 09:03:57 +02:00
|
|
|
reverseGeocoding: {
|
|
|
|
enabled: true,
|
|
|
|
},
|
2022-12-09 21:51:42 +01:00
|
|
|
oauth: {
|
|
|
|
enabled: false,
|
|
|
|
issuerUrl: '',
|
|
|
|
clientId: '',
|
|
|
|
clientSecret: '',
|
2022-12-29 21:47:30 +01:00
|
|
|
mobileOverrideEnabled: false,
|
|
|
|
mobileRedirectUri: '',
|
2022-12-09 21:51:42 +01:00
|
|
|
scope: 'openid email profile',
|
2023-07-15 21:50:29 +02:00
|
|
|
storageLabelClaim: 'preferred_username',
|
2022-12-09 21:51:42 +01:00
|
|
|
buttonText: 'Login with OAuth',
|
|
|
|
autoRegister: true,
|
2023-01-09 22:32:58 +01:00
|
|
|
autoLaunch: false,
|
|
|
|
},
|
|
|
|
passwordLogin: {
|
|
|
|
enabled: true,
|
2022-11-15 05:39:32 +01:00
|
|
|
},
|
2022-12-16 21:26:12 +01:00
|
|
|
storageTemplate: {
|
|
|
|
template: '{{y}}/{{y}}-{{MM}}-{{dd}}/{{filename}}',
|
|
|
|
},
|
2023-08-08 16:39:51 +02:00
|
|
|
thumbnail: {
|
|
|
|
webpSize: 250,
|
|
|
|
jpegSize: 1440,
|
2023-09-03 08:21:51 +02:00
|
|
|
quality: 80,
|
|
|
|
colorspace: Colorspace.P3,
|
2023-08-08 16:39:51 +02:00
|
|
|
},
|
2023-10-24 17:05:42 +02:00
|
|
|
newVersionCheck: {
|
|
|
|
enabled: true,
|
|
|
|
},
|
2023-10-06 09:01:14 +02:00
|
|
|
trash: {
|
|
|
|
enabled: true,
|
|
|
|
days: 30,
|
|
|
|
},
|
2023-10-23 20:38:41 +02:00
|
|
|
theme: {
|
|
|
|
customCss: '',
|
|
|
|
},
|
2023-10-31 21:19:12 +01:00
|
|
|
library: {
|
|
|
|
scan: {
|
|
|
|
enabled: true,
|
|
|
|
cronExpression: CronExpression.EVERY_DAY_AT_MIDNIGHT,
|
|
|
|
},
|
|
|
|
},
|
2022-12-09 21:51:42 +01:00
|
|
|
});
|
2022-11-15 05:39:32 +01:00
|
|
|
|
2023-08-25 06:15:03 +02:00
|
|
|
export enum FeatureFlag {
|
|
|
|
CLIP_ENCODE = 'clipEncode',
|
|
|
|
FACIAL_RECOGNITION = 'facialRecognition',
|
|
|
|
TAG_IMAGE = 'tagImage',
|
2023-09-09 04:51:46 +02:00
|
|
|
MAP = 'map',
|
2023-09-26 09:03:57 +02:00
|
|
|
REVERSE_GEOCODING = 'reverseGeocoding',
|
2023-08-25 06:15:03 +02:00
|
|
|
SIDECAR = 'sidecar',
|
|
|
|
SEARCH = 'search',
|
|
|
|
OAUTH = 'oauth',
|
|
|
|
OAUTH_AUTO_LAUNCH = 'oauthAutoLaunch',
|
|
|
|
PASSWORD_LOGIN = 'passwordLogin',
|
2023-08-25 19:44:52 +02:00
|
|
|
CONFIG_FILE = 'configFile',
|
2023-10-06 09:01:14 +02:00
|
|
|
TRASH = 'trash',
|
2023-08-25 06:15:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export type FeatureFlags = Record<FeatureFlag, boolean>;
|
|
|
|
|
2023-10-09 02:51:03 +02:00
|
|
|
let instance: SystemConfigCore | null;
|
2023-01-24 05:13:42 +01:00
|
|
|
|
2022-11-15 05:39:32 +01:00
|
|
|
@Injectable()
|
2023-01-21 17:11:55 +01:00
|
|
|
export class SystemConfigCore {
|
2023-12-14 17:55:40 +01:00
|
|
|
private logger = new ImmichLogger(SystemConfigCore.name);
|
2022-12-16 21:26:12 +01:00
|
|
|
private validators: SystemConfigValidator[] = [];
|
2023-10-22 17:14:32 +02:00
|
|
|
private configCache: SystemConfigEntity<SystemConfigValue>[] | null = null;
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-10-09 02:51:03 +02:00
|
|
|
public config$ = new Subject<SystemConfig>();
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-10-09 02:51:03 +02:00
|
|
|
private constructor(private repository: ISystemConfigRepository) {}
|
|
|
|
|
|
|
|
static create(repository: ISystemConfigRepository) {
|
|
|
|
if (!instance) {
|
|
|
|
instance = new SystemConfigCore(repository);
|
|
|
|
}
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
|
|
|
static reset() {
|
|
|
|
instance = null;
|
|
|
|
}
|
2022-11-15 05:39:32 +01:00
|
|
|
|
2023-08-25 06:15:03 +02:00
|
|
|
async requireFeature(feature: FeatureFlag) {
|
|
|
|
const hasFeature = await this.hasFeature(feature);
|
|
|
|
if (!hasFeature) {
|
|
|
|
switch (feature) {
|
|
|
|
case FeatureFlag.CLIP_ENCODE:
|
|
|
|
throw new BadRequestException('Clip encoding is not enabled');
|
|
|
|
case FeatureFlag.FACIAL_RECOGNITION:
|
|
|
|
throw new BadRequestException('Facial recognition is not enabled');
|
|
|
|
case FeatureFlag.TAG_IMAGE:
|
|
|
|
throw new BadRequestException('Image tagging is not enabled');
|
|
|
|
case FeatureFlag.SIDECAR:
|
|
|
|
throw new BadRequestException('Sidecar is not enabled');
|
|
|
|
case FeatureFlag.SEARCH:
|
|
|
|
throw new BadRequestException('Search is not enabled');
|
|
|
|
case FeatureFlag.OAUTH:
|
|
|
|
throw new BadRequestException('OAuth is not enabled');
|
|
|
|
case FeatureFlag.PASSWORD_LOGIN:
|
|
|
|
throw new BadRequestException('Password login is not enabled');
|
2023-08-25 19:44:52 +02:00
|
|
|
case FeatureFlag.CONFIG_FILE:
|
|
|
|
throw new BadRequestException('Config file is not set');
|
2023-08-25 06:15:03 +02:00
|
|
|
default:
|
|
|
|
throw new ForbiddenException(`Missing required feature: ${feature}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async hasFeature(feature: FeatureFlag) {
|
|
|
|
const features = await this.getFeatures();
|
|
|
|
return features[feature] ?? false;
|
|
|
|
}
|
|
|
|
|
|
|
|
async getFeatures(): Promise<FeatureFlags> {
|
|
|
|
const config = await this.getConfig();
|
|
|
|
const mlEnabled = config.machineLearning.enabled;
|
|
|
|
|
|
|
|
return {
|
2023-08-29 15:58:00 +02:00
|
|
|
[FeatureFlag.CLIP_ENCODE]: mlEnabled && config.machineLearning.clip.enabled,
|
|
|
|
[FeatureFlag.FACIAL_RECOGNITION]: mlEnabled && config.machineLearning.facialRecognition.enabled,
|
|
|
|
[FeatureFlag.TAG_IMAGE]: mlEnabled && config.machineLearning.classification.enabled,
|
2023-09-09 04:51:46 +02:00
|
|
|
[FeatureFlag.MAP]: config.map.enabled,
|
2023-09-26 09:03:57 +02:00
|
|
|
[FeatureFlag.REVERSE_GEOCODING]: config.reverseGeocoding.enabled,
|
2023-08-25 06:15:03 +02:00
|
|
|
[FeatureFlag.SIDECAR]: true,
|
2023-12-08 17:15:46 +01:00
|
|
|
[FeatureFlag.SEARCH]: true,
|
2023-10-06 09:01:14 +02:00
|
|
|
[FeatureFlag.TRASH]: config.trash.enabled,
|
2023-08-25 06:15:03 +02:00
|
|
|
|
|
|
|
// TODO: use these instead of `POST oauth/config`
|
|
|
|
[FeatureFlag.OAUTH]: config.oauth.enabled,
|
|
|
|
[FeatureFlag.OAUTH_AUTO_LAUNCH]: config.oauth.autoLaunch,
|
|
|
|
[FeatureFlag.PASSWORD_LOGIN]: config.passwordLogin.enabled,
|
2023-08-25 19:44:52 +02:00
|
|
|
[FeatureFlag.CONFIG_FILE]: !!process.env.IMMICH_CONFIG_FILE,
|
2023-08-25 06:15:03 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-12-09 21:51:42 +01:00
|
|
|
public getDefaults(): SystemConfig {
|
|
|
|
return defaults;
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|
|
|
|
|
2022-12-16 21:26:12 +01:00
|
|
|
public addValidator(validator: SystemConfigValidator) {
|
|
|
|
this.validators.push(validator);
|
|
|
|
}
|
|
|
|
|
2023-10-22 17:14:32 +02:00
|
|
|
public async getConfig(force = false): Promise<SystemConfig> {
|
2023-08-25 19:44:52 +02:00
|
|
|
const configFilePath = process.env.IMMICH_CONFIG_FILE;
|
2023-10-22 17:14:32 +02:00
|
|
|
const config = _.cloneDeep(defaults);
|
|
|
|
const overrides = configFilePath ? await this.loadFromFile(configFilePath, force) : await this.repository.load();
|
|
|
|
|
|
|
|
for (const { key, value } of overrides) {
|
|
|
|
// set via dot notation
|
|
|
|
_.set(config, key, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
const errors = await validate(plainToInstance(SystemConfigDto, config), {
|
|
|
|
forbidNonWhitelisted: true,
|
|
|
|
forbidUnknownValues: true,
|
|
|
|
});
|
|
|
|
if (errors.length > 0) {
|
|
|
|
this.logger.error('Validation error', errors);
|
|
|
|
if (configFilePath) {
|
|
|
|
throw new Error(`Invalid value(s) in file: ${errors}`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return config;
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
public async updateConfig(newConfig: SystemConfig): Promise<SystemConfig> {
|
2023-08-25 19:44:52 +02:00
|
|
|
if (await this.hasFeature(FeatureFlag.CONFIG_FILE)) {
|
|
|
|
throw new BadRequestException('Cannot update configuration while IMMICH_CONFIG_FILE is in use');
|
|
|
|
}
|
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
const oldConfig = await this.getConfig();
|
|
|
|
|
2022-12-16 21:26:12 +01:00
|
|
|
try {
|
|
|
|
for (const validator of this.validators) {
|
2023-12-14 17:55:40 +01:00
|
|
|
await validator(newConfig, oldConfig);
|
2022-12-16 21:26:12 +01:00
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
this.logger.warn(`Unable to save system config due to a validation error: ${e}`);
|
|
|
|
throw new BadRequestException(e instanceof Error ? e.message : e);
|
|
|
|
}
|
|
|
|
|
2022-11-15 05:39:32 +01:00
|
|
|
const updates: SystemConfigEntity[] = [];
|
2022-12-09 21:51:42 +01:00
|
|
|
const deletes: SystemConfigEntity[] = [];
|
|
|
|
|
|
|
|
for (const key of Object.values(SystemConfigKey)) {
|
|
|
|
// get via dot notation
|
2023-12-14 17:55:40 +01:00
|
|
|
const item = { key, value: _.get(newConfig, key) as SystemConfigValue };
|
2022-12-09 21:51:42 +01:00
|
|
|
const defaultValue = _.get(defaults, key);
|
2023-12-14 17:55:40 +01:00
|
|
|
const isMissing = !_.has(newConfig, key);
|
2022-11-15 05:39:32 +01:00
|
|
|
|
2023-10-22 17:14:32 +02:00
|
|
|
if (
|
|
|
|
isMissing ||
|
|
|
|
item.value === null ||
|
|
|
|
item.value === '' ||
|
|
|
|
item.value === defaultValue ||
|
|
|
|
_.isEqual(item.value, defaultValue)
|
|
|
|
) {
|
2022-11-15 05:39:32 +01:00
|
|
|
deletes.push(item);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
updates.push(item);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (updates.length > 0) {
|
2023-01-21 17:11:55 +01:00
|
|
|
await this.repository.saveAll(updates);
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (deletes.length > 0) {
|
2023-01-21 17:11:55 +01:00
|
|
|
await this.repository.deleteKeys(deletes.map((item) => item.key));
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
const config = await this.getConfig();
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
this.config$.next(config);
|
2022-12-16 21:26:12 +01:00
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
return config;
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|
2022-12-19 19:13:10 +01:00
|
|
|
|
|
|
|
public async refreshConfig() {
|
2023-08-25 19:44:52 +02:00
|
|
|
const newConfig = await this.getConfig(true);
|
2022-12-19 19:13:10 +01:00
|
|
|
|
|
|
|
this.config$.next(newConfig);
|
|
|
|
}
|
2023-08-25 19:44:52 +02:00
|
|
|
|
|
|
|
private async loadFromFile(filepath: string, force = false) {
|
|
|
|
if (force || !this.configCache) {
|
|
|
|
try {
|
2023-10-22 17:14:32 +02:00
|
|
|
const file = JSON.parse((await this.repository.readFile(filepath)).toString());
|
|
|
|
const overrides: SystemConfigEntity<SystemConfigValue>[] = [];
|
|
|
|
|
|
|
|
for (const key of Object.values(SystemConfigKey)) {
|
|
|
|
const value = _.get(file, key);
|
|
|
|
this.unsetDeep(file, key);
|
|
|
|
if (value !== undefined) {
|
|
|
|
overrides.push({ key, value });
|
|
|
|
}
|
2023-08-25 19:44:52 +02:00
|
|
|
}
|
|
|
|
|
2023-10-22 17:14:32 +02:00
|
|
|
if (!_.isEmpty(file)) {
|
2023-11-14 14:13:42 +01:00
|
|
|
throw new Error(`Unknown keys found: ${JSON.stringify(file)}`);
|
2023-10-22 17:14:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
this.configCache = overrides;
|
2023-08-25 19:44:52 +02:00
|
|
|
} catch (error: Error | any) {
|
|
|
|
this.logger.error(`Unable to load configuration file: ${filepath} due to ${error}`, error?.stack);
|
|
|
|
throw new Error('Invalid configuration file');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.configCache;
|
|
|
|
}
|
2023-10-22 17:14:32 +02:00
|
|
|
|
|
|
|
private unsetDeep(object: object, key: string) {
|
|
|
|
_.unset(object, key);
|
|
|
|
const path = key.split('.');
|
|
|
|
while (path.pop()) {
|
|
|
|
if (!_.isEmpty(_.get(object, path))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_.unset(object, path);
|
|
|
|
}
|
|
|
|
}
|
2022-11-15 05:39:32 +01:00
|
|
|
}
|