2023-07-09 04:43:11 +02:00
|
|
|
import {
|
|
|
|
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-07-09 04:43:11 +02:00
|
|
|
SystemConfig,
|
|
|
|
SystemConfigEntity,
|
|
|
|
SystemConfigKey,
|
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,
|
|
|
|
} from '@app/infra/entities';
|
2023-12-21 02:47:56 +01:00
|
|
|
import { ImmichLogger } from '@app/infra/logger';
|
2023-01-21 17:11:55 +01:00
|
|
|
import { BadRequestException } from '@nestjs/common';
|
2023-12-13 18:23:51 +01:00
|
|
|
import { newCommunicationRepositoryMock, newSystemConfigRepositoryMock } from '@test';
|
|
|
|
import { QueueName } from '../job';
|
|
|
|
import { ICommunicationRepository, ISmartInfoRepository, ISystemConfigRepository, ServerEvent } from '../repositories';
|
2023-10-24 17:05:42 +02:00
|
|
|
import { defaults, SystemConfigValidator } from './system-config.core';
|
2023-01-21 17:11:55 +01:00
|
|
|
import { SystemConfigService } from './system-config.service';
|
|
|
|
|
|
|
|
const updates: SystemConfigEntity[] = [
|
2023-05-22 20:07:43 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_CRF, value: 30 },
|
2023-01-21 17:11:55 +01:00
|
|
|
{ key: SystemConfigKey.OAUTH_AUTO_LAUNCH, value: true },
|
2023-10-06 09:01:14 +02:00
|
|
|
{ key: SystemConfigKey.TRASH_DAYS, value: 10 },
|
2023-01-21 17:11:55 +01:00
|
|
|
];
|
|
|
|
|
2023-06-01 12:32:51 +02:00
|
|
|
const updatedConfig = Object.freeze<SystemConfig>({
|
|
|
|
job: {
|
|
|
|
[QueueName.BACKGROUND_TASK]: { concurrency: 5 },
|
2023-12-16 17:50:46 +01:00
|
|
|
[QueueName.SMART_SEARCH]: { concurrency: 2 },
|
2023-06-01 12:32:51 +02:00
|
|
|
[QueueName.METADATA_EXTRACTION]: { concurrency: 5 },
|
|
|
|
[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-01-21 17:11:55 +01:00
|
|
|
ffmpeg: {
|
2023-05-22 20:07:43 +02:00
|
|
|
crf: 30,
|
|
|
|
threads: 0,
|
2023-01-21 17:11:55 +01:00
|
|
|
preset: 'ultrafast',
|
2023-07-09 04:43:11 +02:00
|
|
|
targetAudioCodec: AudioCodec.AAC,
|
2023-04-04 03:42:53 +02:00
|
|
|
targetResolution: '720',
|
2023-07-09 04:43:11 +02:00
|
|
|
targetVideoCodec: VideoCodec.H264,
|
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-02 03:56:10 +02:00
|
|
|
accel: TranscodeHWAccel.DISABLED,
|
2023-08-07 22:35:25 +02:00
|
|
|
tonemap: ToneMapping.HABLE,
|
2023-01-21 17:11:55 +01:00
|
|
|
},
|
2023-12-14 17:55:40 +01:00
|
|
|
logging: {
|
|
|
|
enabled: true,
|
|
|
|
level: LogLevel.LOG,
|
|
|
|
},
|
2023-08-25 06:15:03 +02:00
|
|
|
machineLearning: {
|
|
|
|
enabled: true,
|
|
|
|
url: 'http://immich-machine-learning:3003',
|
2023-08-29 15:58:00 +02:00
|
|
|
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,
|
|
|
|
},
|
2023-01-21 17:11:55 +01:00
|
|
|
oauth: {
|
|
|
|
autoLaunch: true,
|
|
|
|
autoRegister: true,
|
|
|
|
buttonText: 'Login with OAuth',
|
|
|
|
clientId: '',
|
|
|
|
clientSecret: '',
|
|
|
|
enabled: false,
|
|
|
|
issuerUrl: '',
|
|
|
|
mobileOverrideEnabled: false,
|
|
|
|
mobileRedirectUri: '',
|
|
|
|
scope: 'openid email profile',
|
2023-07-15 21:50:29 +02:00
|
|
|
storageLabelClaim: 'preferred_username',
|
2023-01-21 17:11:55 +01:00
|
|
|
},
|
|
|
|
passwordLogin: {
|
|
|
|
enabled: true,
|
|
|
|
},
|
|
|
|
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: 10,
|
|
|
|
},
|
2023-10-23 20:38:41 +02:00
|
|
|
theme: {
|
|
|
|
customCss: '',
|
|
|
|
},
|
2023-10-31 21:19:12 +01:00
|
|
|
library: {
|
|
|
|
scan: {
|
|
|
|
enabled: true,
|
|
|
|
cronExpression: '0 0 * * *',
|
|
|
|
},
|
|
|
|
},
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe(SystemConfigService.name, () => {
|
|
|
|
let sut: SystemConfigService;
|
|
|
|
let configMock: jest.Mocked<ISystemConfigRepository>;
|
2023-10-06 09:01:14 +02:00
|
|
|
let communicationMock: jest.Mocked<ICommunicationRepository>;
|
2023-12-08 17:15:46 +01:00
|
|
|
let smartInfoMock: jest.Mocked<ISmartInfoRepository>;
|
2023-01-21 17:11:55 +01:00
|
|
|
|
|
|
|
beforeEach(async () => {
|
2023-08-25 19:44:52 +02:00
|
|
|
delete process.env.IMMICH_CONFIG_FILE;
|
2023-01-21 17:11:55 +01:00
|
|
|
configMock = newSystemConfigRepositoryMock();
|
2023-10-06 09:01:14 +02:00
|
|
|
communicationMock = newCommunicationRepositoryMock();
|
2023-12-13 18:23:51 +01:00
|
|
|
sut = new SystemConfigService(configMock, communicationMock, smartInfoMock);
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should work', () => {
|
|
|
|
expect(sut).toBeDefined();
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getDefaults', () => {
|
|
|
|
it('should return the default config', () => {
|
|
|
|
configMock.load.mockResolvedValue(updates);
|
|
|
|
|
2023-07-15 06:03:56 +02:00
|
|
|
expect(sut.getDefaults()).toEqual(defaults);
|
2023-01-21 17:11:55 +01:00
|
|
|
expect(configMock.load).not.toHaveBeenCalled();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('addValidator', () => {
|
|
|
|
it('should call the validator on config changes', async () => {
|
|
|
|
const validator: SystemConfigValidator = jest.fn();
|
|
|
|
sut.addValidator(validator);
|
2023-07-15 06:03:56 +02:00
|
|
|
await sut.updateConfig(defaults);
|
2023-12-14 17:55:40 +01:00
|
|
|
expect(validator).toHaveBeenCalledWith(defaults, defaults);
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getConfig', () => {
|
2023-12-21 02:47:56 +01:00
|
|
|
let warnLog: jest.SpyInstance;
|
|
|
|
|
|
|
|
beforeEach(() => {
|
|
|
|
warnLog = jest.spyOn(ImmichLogger.prototype, 'warn');
|
|
|
|
});
|
|
|
|
|
|
|
|
afterEach(() => {
|
|
|
|
warnLog.mockRestore();
|
|
|
|
});
|
|
|
|
|
2023-01-21 17:11:55 +01:00
|
|
|
it('should return the default config', async () => {
|
|
|
|
configMock.load.mockResolvedValue([]);
|
|
|
|
|
2023-07-15 06:03:56 +02:00
|
|
|
await expect(sut.getConfig()).resolves.toEqual(defaults);
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should merge the overrides', async () => {
|
|
|
|
configMock.load.mockResolvedValue([
|
2023-05-22 20:07:43 +02:00
|
|
|
{ key: SystemConfigKey.FFMPEG_CRF, value: 30 },
|
2023-01-21 17:11:55 +01:00
|
|
|
{ key: SystemConfigKey.OAUTH_AUTO_LAUNCH, value: true },
|
2023-10-06 09:01:14 +02:00
|
|
|
{ key: SystemConfigKey.TRASH_DAYS, value: 10 },
|
2023-01-21 17:11:55 +01:00
|
|
|
]);
|
|
|
|
|
|
|
|
await expect(sut.getConfig()).resolves.toEqual(updatedConfig);
|
|
|
|
});
|
2023-08-25 19:44:52 +02:00
|
|
|
|
|
|
|
it('should load the config from a file', async () => {
|
|
|
|
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
|
2023-10-06 09:01:14 +02:00
|
|
|
const partialConfig = { ffmpeg: { crf: 30 }, oauth: { autoLaunch: true }, trash: { days: 10 } };
|
2023-11-09 17:10:56 +01:00
|
|
|
configMock.readFile.mockResolvedValue(JSON.stringify(partialConfig));
|
2023-08-25 19:44:52 +02:00
|
|
|
|
|
|
|
await expect(sut.getConfig()).resolves.toEqual(updatedConfig);
|
|
|
|
|
|
|
|
expect(configMock.readFile).toHaveBeenCalledWith('immich-config.json');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should accept an empty configuration file', async () => {
|
|
|
|
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
|
2023-11-09 17:10:56 +01:00
|
|
|
configMock.readFile.mockResolvedValue(JSON.stringify({}));
|
2023-08-25 19:44:52 +02:00
|
|
|
|
|
|
|
await expect(sut.getConfig()).resolves.toEqual(defaults);
|
|
|
|
|
|
|
|
expect(configMock.readFile).toHaveBeenCalledWith('immich-config.json');
|
|
|
|
});
|
|
|
|
|
2023-10-17 23:34:16 +02:00
|
|
|
it('should allow underscores in the machine learning url', async () => {
|
|
|
|
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
|
|
|
|
const partialConfig = { machineLearning: { url: 'immich_machine_learning' } };
|
2023-11-09 17:10:56 +01:00
|
|
|
configMock.readFile.mockResolvedValue(JSON.stringify(partialConfig));
|
2023-10-17 23:34:16 +02:00
|
|
|
|
|
|
|
const config = await sut.getConfig();
|
|
|
|
expect(config.machineLearning.url).toEqual('immich_machine_learning');
|
|
|
|
});
|
|
|
|
|
2023-08-25 19:44:52 +02:00
|
|
|
const tests = [
|
|
|
|
{ should: 'validate numbers', config: { ffmpeg: { crf: 'not-a-number' } } },
|
|
|
|
{ should: 'validate booleans', config: { oauth: { enabled: 'invalid' } } },
|
|
|
|
{ should: 'validate enums', config: { ffmpeg: { transcode: 'unknown' } } },
|
|
|
|
{ should: 'validate required oauth fields', config: { oauth: { enabled: true } } },
|
2023-12-21 02:47:56 +01:00
|
|
|
{ should: 'warn for top level unknown options', warn: true, config: { unknownOption: true } },
|
|
|
|
{ should: 'warn for nested unknown options', warn: true, config: { ffmpeg: { unknownOption: true } } },
|
2023-08-25 19:44:52 +02:00
|
|
|
];
|
|
|
|
|
|
|
|
for (const test of tests) {
|
|
|
|
it(`should ${test.should}`, async () => {
|
|
|
|
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
|
2023-11-09 17:10:56 +01:00
|
|
|
configMock.readFile.mockResolvedValue(JSON.stringify(test.config));
|
2023-08-25 19:44:52 +02:00
|
|
|
|
2023-12-21 02:47:56 +01:00
|
|
|
if (test.warn) {
|
|
|
|
await sut.getConfig();
|
|
|
|
expect(warnLog).toHaveBeenCalled();
|
|
|
|
} else {
|
|
|
|
await expect(sut.getConfig()).rejects.toBeInstanceOf(Error);
|
|
|
|
}
|
2023-08-25 19:44:52 +02:00
|
|
|
});
|
|
|
|
}
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('getStorageTemplateOptions', () => {
|
|
|
|
it('should send back the datetime variables', () => {
|
|
|
|
expect(sut.getStorageTemplateOptions()).toEqual({
|
|
|
|
dayOptions: ['d', 'dd'],
|
|
|
|
hourOptions: ['h', 'hh', 'H', 'HH'],
|
|
|
|
minuteOptions: ['m', 'mm'],
|
|
|
|
monthOptions: ['M', 'MM', 'MMM', 'MMMM'],
|
|
|
|
presetOptions: [
|
|
|
|
'{{y}}/{{y}}-{{MM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}/{{MM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}/{{MMMM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}/{{MM}}/{{filename}}',
|
|
|
|
'{{y}}/{{MMM}}/{{filename}}',
|
|
|
|
'{{y}}/{{MMMM}}/{{filename}}',
|
|
|
|
'{{y}}/{{MM}}/{{dd}}/{{filename}}',
|
|
|
|
'{{y}}/{{MMMM}}/{{dd}}/{{filename}}',
|
|
|
|
'{{y}}/{{y}}-{{MM}}/{{y}}-{{MM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}-{{MM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}-{{MMM}}-{{dd}}/{{filename}}',
|
|
|
|
'{{y}}-{{MMMM}}-{{dd}}/{{filename}}',
|
2023-03-01 14:10:01 +01:00
|
|
|
'{{y}}/{{y}}-{{MM}}/{{filename}}',
|
2023-09-28 19:47:31 +02:00
|
|
|
'{{y}}/{{y}}-{{WW}}/{{filename}}',
|
2023-10-20 23:17:17 +02:00
|
|
|
'{{y}}/{{y}}-{{MM}}-{{dd}}/{{assetId}}',
|
|
|
|
'{{y}}/{{y}}-{{MM}}/{{assetId}}',
|
|
|
|
'{{y}}/{{y}}-{{WW}}/{{assetId}}',
|
2023-10-23 20:00:31 +02:00
|
|
|
'{{album}}/{{filename}}',
|
2023-01-21 17:11:55 +01:00
|
|
|
],
|
|
|
|
secondOptions: ['s', 'ss'],
|
2023-09-28 19:47:31 +02:00
|
|
|
weekOptions: ['W', 'WW'],
|
2023-01-21 17:11:55 +01:00
|
|
|
yearOptions: ['y', 'yy'],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('updateConfig', () => {
|
2023-12-13 18:23:51 +01:00
|
|
|
it('should update the config and emit client and server events', async () => {
|
2023-01-21 17:11:55 +01:00
|
|
|
configMock.load.mockResolvedValue(updates);
|
|
|
|
|
|
|
|
await expect(sut.updateConfig(updatedConfig)).resolves.toEqual(updatedConfig);
|
|
|
|
|
2023-12-13 18:23:51 +01:00
|
|
|
expect(communicationMock.broadcast).toHaveBeenCalled();
|
|
|
|
expect(communicationMock.sendServerEvent).toHaveBeenCalledWith(ServerEvent.CONFIG_UPDATE);
|
2023-01-21 17:11:55 +01:00
|
|
|
expect(configMock.saveAll).toHaveBeenCalledWith(updates);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw an error if the config is not valid', async () => {
|
|
|
|
const validator = jest.fn().mockRejectedValue('invalid config');
|
|
|
|
|
|
|
|
sut.addValidator(validator);
|
|
|
|
|
|
|
|
await expect(sut.updateConfig(updatedConfig)).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
2023-12-14 17:55:40 +01:00
|
|
|
expect(validator).toHaveBeenCalledWith(updatedConfig, defaults);
|
2023-01-21 17:11:55 +01:00
|
|
|
expect(configMock.saveAll).not.toHaveBeenCalled();
|
|
|
|
});
|
2023-08-25 19:44:52 +02:00
|
|
|
|
|
|
|
it('should throw an error if a config file is in use', async () => {
|
|
|
|
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
|
2023-11-09 17:10:56 +01:00
|
|
|
configMock.readFile.mockResolvedValue(JSON.stringify({}));
|
2023-08-25 19:44:52 +02:00
|
|
|
await expect(sut.updateConfig(defaults)).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
expect(configMock.saveAll).not.toHaveBeenCalled();
|
|
|
|
});
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
describe('refreshConfig', () => {
|
|
|
|
it('should notify the subscribers', async () => {
|
|
|
|
const changeMock = jest.fn();
|
|
|
|
const subscription = sut.config$.subscribe(changeMock);
|
|
|
|
|
|
|
|
await sut.refreshConfig();
|
|
|
|
|
2023-07-15 06:03:56 +02:00
|
|
|
expect(changeMock).toHaveBeenCalledWith(defaults);
|
2023-01-21 17:11:55 +01:00
|
|
|
|
|
|
|
subscription.unsubscribe();
|
|
|
|
});
|
|
|
|
});
|
2023-10-26 00:13:05 +02:00
|
|
|
|
2023-11-18 05:13:36 +01:00
|
|
|
describe('getCustomCss', () => {
|
2023-10-26 00:13:05 +02:00
|
|
|
it('should return the default theme', async () => {
|
2023-11-18 05:13:36 +01:00
|
|
|
await expect(sut.getCustomCss()).resolves.toEqual(defaults.theme.customCss);
|
2023-10-26 00:13:05 +02:00
|
|
|
});
|
|
|
|
});
|
2023-01-21 17:11:55 +01:00
|
|
|
});
|