1
0
mirror of https://github.com/immich-app/immich.git synced 2025-01-12 15:32:36 +02:00

feat(web, server): Ability to use config file instead of admin UI (#3836)

* implement method to read config file

* getConfig returns config file if present

* return isConfigFile for http requests

* disable elements if config file is used, show message if config file is set, copy existing config to clipboard

* fix allowing partial configuration files

* add new env variable to docs

* fix tests

* minor refactoring, address review

* adapt config type in frontend

* remove unnecessary imports

* move config file reading to system-config repo

* add documentation

* fix code formatting in system settings page

* add validator for config file

* fix formatting in docs

* update generated files

* throw error when trying to update config. e.g. via cli or api

* switch to feature flags for isConfigFile

* refactoring

* refactor: config file

* chore: open api

* feat: always show copy/export buttons

* fix: default flags

* refactor: copy to clipboard

---------

Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
This commit is contained in:
Daniel Dietzler 2023-08-25 19:44:52 +02:00 committed by GitHub
parent 20e0c03b39
commit 59bb727636
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
33 changed files with 344 additions and 83 deletions

View File

@ -2173,6 +2173,12 @@ export interface ServerFeaturesDto {
* @memberof ServerFeaturesDto * @memberof ServerFeaturesDto
*/ */
'clipEncode': boolean; 'clipEncode': boolean;
/**
*
* @type {boolean}
* @memberof ServerFeaturesDto
*/
'configFile': boolean;
/** /**
* *
* @type {boolean} * @type {boolean}

View File

@ -0,0 +1,91 @@
# Config File
A config file can be provided as an alternative to the UI configuration.
### Step 1 - Create a new config file
In JSON format, create a new config file (e.g. `immich.config`) and put it in a location that can be accessed by Immich.
The default configuration looks like this:
```json
{
"ffmpeg": {
"crf": 23,
"threads": 0,
"preset": "ultrafast",
"targetVideoCodec": "h264",
"targetAudioCodec": "aac",
"targetResolution": "720",
"maxBitrate": "0",
"twoPass": false,
"transcode": "required",
"tonemap": "hable",
"accel": "disabled"
},
"job": {
"backgroundTask": {
"concurrency": 5
},
"clipEncoding": {
"concurrency": 2
},
"metadataExtraction": {
"concurrency": 5
},
"objectTagging": {
"concurrency": 2
},
"recognizeFaces": {
"concurrency": 2
},
"search": {
"concurrency": 5
},
"sidecar": {
"concurrency": 5
},
"storageTemplateMigration": {
"concurrency": 5
},
"thumbnailGeneration": {
"concurrency": 5
},
"videoConversion": {
"concurrency": 1
}
},
"oauth": {
"enabled": false,
"issuerUrl": "",
"clientId": "",
"clientSecret": "",
"mobileOverrideEnabled": false,
"mobileRedirectUri": "",
"scope": "openid email profile",
"storageLabelClaim": "preferred_username",
"buttonText": "Login with OAuth",
"autoRegister": true,
"autoLaunch": false
},
"passwordLogin": {
"enabled": true
},
"storageTemplate": {
"template": "{{y}}/{{y}}-{{MM}}-{{dd}}/{{filename}}"
},
"thumbnail": {
"webpSize": 250,
"jpegSize": 1440
}
}
```
:::tip
In Administration > Settings is a button to copy the current configuration to your clipboard.
So you can just grab it from there, paste it into a file and you're pretty much good to go.
:::
### Step 2 - Specify the file location
In your `.env` file, set the variable `IMMICH_CONFIG_FILE` to the path of your config.
For more information, refer to the [Environment Variables](https://docs.immich.app/docs/install/environment-variables) section.

View File

@ -1,3 +1,7 @@
---
sidebar_position: 90
---
# Environment Variables # Environment Variables
## Docker Compose ## Docker Compose
@ -22,6 +26,7 @@ These environment variables are used by the `docker-compose.yml` file and do **N
| `LOG_LEVEL` | Log Level (verbose, debug, log, warn, error) | `log` | server, microservices | | `LOG_LEVEL` | Log Level (verbose, debug, log, warn, error) | `log` | server, microservices |
| `IMMICH_MEDIA_LOCATION` | Media Location | `./upload` | server, microservices | | `IMMICH_MEDIA_LOCATION` | Media Location | `./upload` | server, microservices |
| `PUBLIC_LOGIN_PAGE_MESSAGE` | Public Login Page Message | | web | | `PUBLIC_LOGIN_PAGE_MESSAGE` | Public Login Page Message | | web |
| `IMMICH_CONFIG_FILE` | Path to config file | | server |
:::tip :::tip

View File

@ -1,5 +1,5 @@
--- ---
sidebar_position: 100 sidebar_position: 80
--- ---
import RegisterAdminUser from '../partials/_register-admin.md'; import RegisterAdminUser from '../partials/_register-admin.md';

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@ -6478,6 +6478,9 @@
"clipEncode": { "clipEncode": {
"type": "boolean" "type": "boolean"
}, },
"configFile": {
"type": "boolean"
},
"facialRecognition": { "facialRecognition": {
"type": "boolean" "type": "boolean"
}, },
@ -6501,6 +6504,7 @@
} }
}, },
"required": [ "required": [
"configFile",
"clipEncode", "clipEncode",
"facialRecognition", "facialRecognition",
"sidecar", "sidecar",

View File

@ -80,6 +80,7 @@ export class ServerMediaTypesResponseDto {
} }
export class ServerFeaturesDto implements FeatureFlags { export class ServerFeaturesDto implements FeatureFlags {
configFile!: boolean;
clipEncode!: boolean; clipEncode!: boolean;
facialRecognition!: boolean; facialRecognition!: boolean;
sidecar!: boolean; sidecar!: boolean;

View File

@ -155,6 +155,7 @@ describe(ServerInfoService.name, () => {
search: true, search: true,
sidecar: true, sidecar: true,
tagImage: true, tagImage: true,
configFile: false,
}); });
expect(configMock.load).toHaveBeenCalled(); expect(configMock.load).toHaveBeenCalled();
}); });

View File

@ -10,10 +10,13 @@ import {
VideoCodec, VideoCodec,
} from '@app/infra/entities'; } from '@app/infra/entities';
import { BadRequestException, ForbiddenException, Injectable, Logger } from '@nestjs/common'; import { BadRequestException, ForbiddenException, Injectable, Logger } from '@nestjs/common';
import { plainToClass } from 'class-transformer';
import { validate } from 'class-validator';
import * as _ from 'lodash'; import * as _ from 'lodash';
import { Subject } from 'rxjs'; import { Subject } from 'rxjs';
import { DeepPartial } from 'typeorm'; import { DeepPartial } from 'typeorm';
import { QueueName } from '../job/job.constants'; import { QueueName } from '../job/job.constants';
import { SystemConfigDto } from './dto';
import { ISystemConfigRepository } from './system-config.repository'; import { ISystemConfigRepository } from './system-config.repository';
export type SystemConfigValidator = (config: SystemConfig) => void | Promise<void>; export type SystemConfigValidator = (config: SystemConfig) => void | Promise<void>;
@ -87,6 +90,7 @@ export enum FeatureFlag {
OAUTH = 'oauth', OAUTH = 'oauth',
OAUTH_AUTO_LAUNCH = 'oauthAutoLaunch', OAUTH_AUTO_LAUNCH = 'oauthAutoLaunch',
PASSWORD_LOGIN = 'passwordLogin', PASSWORD_LOGIN = 'passwordLogin',
CONFIG_FILE = 'configFile',
} }
export type FeatureFlags = Record<FeatureFlag, boolean>; export type FeatureFlags = Record<FeatureFlag, boolean>;
@ -97,6 +101,7 @@ const singleton = new Subject<SystemConfig>();
export class SystemConfigCore { export class SystemConfigCore {
private logger = new Logger(SystemConfigCore.name); private logger = new Logger(SystemConfigCore.name);
private validators: SystemConfigValidator[] = []; private validators: SystemConfigValidator[] = [];
private configCache: SystemConfig | null = null;
public config$ = singleton; public config$ = singleton;
@ -120,6 +125,8 @@ export class SystemConfigCore {
throw new BadRequestException('OAuth is not enabled'); throw new BadRequestException('OAuth is not enabled');
case FeatureFlag.PASSWORD_LOGIN: case FeatureFlag.PASSWORD_LOGIN:
throw new BadRequestException('Password login is not enabled'); throw new BadRequestException('Password login is not enabled');
case FeatureFlag.CONFIG_FILE:
throw new BadRequestException('Config file is not set');
default: default:
throw new ForbiddenException(`Missing required feature: ${feature}`); throw new ForbiddenException(`Missing required feature: ${feature}`);
} }
@ -146,6 +153,7 @@ export class SystemConfigCore {
[FeatureFlag.OAUTH]: config.oauth.enabled, [FeatureFlag.OAUTH]: config.oauth.enabled,
[FeatureFlag.OAUTH_AUTO_LAUNCH]: config.oauth.autoLaunch, [FeatureFlag.OAUTH_AUTO_LAUNCH]: config.oauth.autoLaunch,
[FeatureFlag.PASSWORD_LOGIN]: config.passwordLogin.enabled, [FeatureFlag.PASSWORD_LOGIN]: config.passwordLogin.enabled,
[FeatureFlag.CONFIG_FILE]: !!process.env.IMMICH_CONFIG_FILE,
}; };
} }
@ -157,18 +165,16 @@ export class SystemConfigCore {
this.validators.push(validator); this.validators.push(validator);
} }
public async getConfig() { public getConfig(force = false): Promise<SystemConfig> {
const overrides = await this.repository.load(); const configFilePath = process.env.IMMICH_CONFIG_FILE;
const config: DeepPartial<SystemConfig> = {}; return configFilePath ? this.loadFromFile(configFilePath, force) : this.loadFromDatabase();
for (const { key, value } of overrides) {
// set via dot notation
_.set(config, key, value);
}
return _.defaultsDeep(config, defaults) as SystemConfig;
} }
public async updateConfig(config: SystemConfig): Promise<SystemConfig> { public async updateConfig(config: SystemConfig): Promise<SystemConfig> {
if (await this.hasFeature(FeatureFlag.CONFIG_FILE)) {
throw new BadRequestException('Cannot update configuration while IMMICH_CONFIG_FILE is in use');
}
try { try {
for (const validator of this.validators) { for (const validator of this.validators) {
await validator(config); await validator(config);
@ -211,8 +217,45 @@ export class SystemConfigCore {
} }
public async refreshConfig() { public async refreshConfig() {
const newConfig = await this.getConfig(); const newConfig = await this.getConfig(true);
this.config$.next(newConfig); this.config$.next(newConfig);
} }
private async loadFromDatabase() {
const config: DeepPartial<SystemConfig> = {};
const overrides = await this.repository.load();
for (const { key, value } of overrides) {
// set via dot notation
_.set(config, key, value);
}
return _.defaultsDeep(config, defaults) as SystemConfig;
}
private async loadFromFile(filepath: string, force = false) {
if (force || !this.configCache) {
try {
const overrides = JSON.parse((await this.repository.readFile(filepath)).toString());
const config = plainToClass(SystemConfigDto, _.defaultsDeep(overrides, defaults));
const errors = await validate(config, {
whitelist: true,
forbidNonWhitelisted: true,
forbidUnknownValues: true,
});
if (errors.length > 0) {
this.logger.error('Validation error', errors);
throw new Error(`Invalid value(s) in file: ${errors}`);
}
this.configCache = config;
} 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;
}
} }

View File

@ -4,6 +4,7 @@ export const ISystemConfigRepository = 'ISystemConfigRepository';
export interface ISystemConfigRepository { export interface ISystemConfigRepository {
load(): Promise<SystemConfigEntity[]>; load(): Promise<SystemConfigEntity[]>;
readFile(filename: string): Promise<Buffer>;
saveAll(items: SystemConfigEntity[]): Promise<SystemConfigEntity[]>; saveAll(items: SystemConfigEntity[]): Promise<SystemConfigEntity[]>;
deleteKeys(keys: string[]): Promise<void>; deleteKeys(keys: string[]): Promise<void>;
} }

View File

@ -84,6 +84,7 @@ describe(SystemConfigService.name, () => {
let jobMock: jest.Mocked<IJobRepository>; let jobMock: jest.Mocked<IJobRepository>;
beforeEach(async () => { beforeEach(async () => {
delete process.env.IMMICH_CONFIG_FILE;
configMock = newSystemConfigRepositoryMock(); configMock = newSystemConfigRepositoryMock();
jobMock = newJobRepositoryMock(); jobMock = newJobRepositoryMock();
sut = new SystemConfigService(configMock, jobMock); sut = new SystemConfigService(configMock, jobMock);
@ -126,6 +127,43 @@ describe(SystemConfigService.name, () => {
await expect(sut.getConfig()).resolves.toEqual(updatedConfig); await expect(sut.getConfig()).resolves.toEqual(updatedConfig);
}); });
it('should load the config from a file', async () => {
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
const partialConfig = { ffmpeg: { crf: 30 }, oauth: { autoLaunch: true } };
configMock.readFile.mockResolvedValue(Buffer.from(JSON.stringify(partialConfig)));
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';
configMock.readFile.mockResolvedValue(Buffer.from(JSON.stringify({})));
await expect(sut.getConfig()).resolves.toEqual(defaults);
expect(configMock.readFile).toHaveBeenCalledWith('immich-config.json');
});
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 top level unknown options', config: { unknownOption: true } },
{ should: 'validate nested unknown options', config: { ffmpeg: { unknownOption: true } } },
{ should: 'validate required oauth fields', config: { oauth: { enabled: true } } },
];
for (const test of tests) {
it(`should ${test.should}`, async () => {
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
configMock.readFile.mockResolvedValue(Buffer.from(JSON.stringify(test.config)));
await expect(sut.getConfig()).rejects.toBeInstanceOf(Error);
});
}
}); });
describe('getStorageTemplateOptions', () => { describe('getStorageTemplateOptions', () => {
@ -176,6 +214,13 @@ describe(SystemConfigService.name, () => {
expect(validator).toHaveBeenCalledWith(updatedConfig); expect(validator).toHaveBeenCalledWith(updatedConfig);
expect(configMock.saveAll).not.toHaveBeenCalled(); expect(configMock.saveAll).not.toHaveBeenCalled();
}); });
it('should throw an error if a config file is in use', async () => {
process.env.IMMICH_CONFIG_FILE = 'immich-config.json';
configMock.readFile.mockResolvedValue(Buffer.from(JSON.stringify({})));
await expect(sut.updateConfig(defaults)).rejects.toBeInstanceOf(BadRequestException);
expect(configMock.saveAll).not.toHaveBeenCalled();
});
}); });
describe('refreshConfig', () => { describe('refreshConfig', () => {

View File

@ -1,5 +1,6 @@
import { ISystemConfigRepository } from '@app/domain'; import { ISystemConfigRepository } from '@app/domain';
import { InjectRepository } from '@nestjs/typeorm'; import { InjectRepository } from '@nestjs/typeorm';
import { readFile } from 'fs/promises';
import { In, Repository } from 'typeorm'; import { In, Repository } from 'typeorm';
import { SystemConfigEntity } from '../entities'; import { SystemConfigEntity } from '../entities';
@ -13,6 +14,8 @@ export class SystemConfigRepository implements ISystemConfigRepository {
return this.repository.find(); return this.repository.find();
} }
readFile = readFile;
saveAll(items: SystemConfigEntity[]): Promise<SystemConfigEntity[]> { saveAll(items: SystemConfigEntity[]): Promise<SystemConfigEntity[]> {
return this.repository.save(items); return this.repository.save(items);
} }

View File

@ -3,6 +3,7 @@ import { ISystemConfigRepository } from '@app/domain';
export const newSystemConfigRepositoryMock = (): jest.Mocked<ISystemConfigRepository> => { export const newSystemConfigRepositoryMock = (): jest.Mocked<ISystemConfigRepository> => {
return { return {
load: jest.fn().mockResolvedValue([]), load: jest.fn().mockResolvedValue([]),
readFile: jest.fn(),
saveAll: jest.fn().mockResolvedValue([]), saveAll: jest.fn().mockResolvedValue([]),
deleteKeys: jest.fn(), deleteKeys: jest.fn(),
}; };

View File

@ -2173,6 +2173,12 @@ export interface ServerFeaturesDto {
* @memberof ServerFeaturesDto * @memberof ServerFeaturesDto
*/ */
'clipEncode': boolean; 'clipEncode': boolean;
/**
*
* @type {boolean}
* @memberof ServerFeaturesDto
*/
'configFile': boolean;
/** /**
* *
* @type {boolean} * @type {boolean}

View File

@ -1,9 +1,23 @@
import type { AxiosError, AxiosPromise } from 'axios'; import type { AxiosError, AxiosPromise } from 'axios';
import {
notificationController,
NotificationType,
} from '../lib/components/shared-components/notification/notification';
import { handleError } from '../lib/utils/handle-error';
import { api } from './api'; import { api } from './api';
import type { UserResponseDto } from './open-api'; import type { UserResponseDto } from './open-api';
export type ApiError = AxiosError<{ message: string }>; export type ApiError = AxiosError<{ message: string }>;
export const copyToClipboard = async (secret: string) => {
try {
await navigator.clipboard.writeText(secret);
notificationController.show({ message: 'Copied to clipboard!', type: NotificationType.Info });
} catch (error) {
handleError(error, 'Cannot copy to clipboard, make sure you are accessing the page through https');
}
};
export const oauth = { export const oauth = {
isCallback: (location: Location) => { isCallback: (location: Location) => {
const search = location.search; const search = location.search;

View File

@ -20,6 +20,7 @@
import { fade } from 'svelte/transition'; import { fade } from 'svelte/transition';
export let ffmpegConfig: SystemConfigFFmpegDto; // this is the config that is being edited export let ffmpegConfig: SystemConfigFFmpegDto; // this is the config that is being edited
export let disabled = false;
let savedConfig: SystemConfigFFmpegDto; let savedConfig: SystemConfigFFmpegDto;
let defaultConfig: SystemConfigFFmpegDto; let defaultConfig: SystemConfigFFmpegDto;
@ -90,6 +91,7 @@
<div class="ml-4 mt-4 flex flex-col gap-4"> <div class="ml-4 mt-4 flex flex-col gap-4">
<SettingInputField <SettingInputField
inputType={SettingInputFieldType.NUMBER} inputType={SettingInputFieldType.NUMBER}
{disabled}
label="CONSTANT RATE FACTOR (-crf)" label="CONSTANT RATE FACTOR (-crf)"
desc="Video quality level. Typical values are 23 for H.264, 28 for HEVC, and 31 for VP9. Lower is better, but takes longer to encode and produces larger files." desc="Video quality level. Typical values are 23 for H.264, 28 for HEVC, and 31 for VP9. Lower is better, but takes longer to encode and produces larger files."
bind:value={ffmpegConfig.crf} bind:value={ffmpegConfig.crf}
@ -99,6 +101,7 @@
<SettingSelect <SettingSelect
label="PRESET (-preset)" label="PRESET (-preset)"
{disabled}
desc="Compression speed. Slower presets produce smaller files, and increase quality when targeting a certain bitrate. VP9 ignores speeds above `faster`." desc="Compression speed. Slower presets produce smaller files, and increase quality when targeting a certain bitrate. VP9 ignores speeds above `faster`."
bind:value={ffmpegConfig.preset} bind:value={ffmpegConfig.preset}
name="preset" name="preset"
@ -118,6 +121,7 @@
<SettingSelect <SettingSelect
label="AUDIO CODEC" label="AUDIO CODEC"
{disabled}
desc="Opus is the highest quality option, but has lower compatibility with old devices or software." desc="Opus is the highest quality option, but has lower compatibility with old devices or software."
bind:value={ffmpegConfig.targetAudioCodec} bind:value={ffmpegConfig.targetAudioCodec}
options={[ options={[
@ -131,6 +135,7 @@
<SettingSelect <SettingSelect
label="VIDEO CODEC" label="VIDEO CODEC"
{disabled}
desc="VP9 has high efficiency and web compatibility, but takes longer to transcode. HEVC performs similarly, but has lower web compatibility. H.264 is widely compatible and quick to transcode, but produces much larger files." desc="VP9 has high efficiency and web compatibility, but takes longer to transcode. HEVC performs similarly, but has lower web compatibility. H.264 is widely compatible and quick to transcode, but produces much larger files."
bind:value={ffmpegConfig.targetVideoCodec} bind:value={ffmpegConfig.targetVideoCodec}
options={[ options={[
@ -144,6 +149,7 @@
<SettingSelect <SettingSelect
label="TARGET RESOLUTION" label="TARGET RESOLUTION"
{disabled}
desc="Higher resolutions can preserve more detail but take longer to encode, have larger file sizes, and can reduce app responsiveness." desc="Higher resolutions can preserve more detail but take longer to encode, have larger file sizes, and can reduce app responsiveness."
bind:value={ffmpegConfig.targetResolution} bind:value={ffmpegConfig.targetResolution}
options={[ options={[
@ -160,6 +166,7 @@
<SettingInputField <SettingInputField
inputType={SettingInputFieldType.TEXT} inputType={SettingInputFieldType.TEXT}
{disabled}
label="MAX BITRATE" label="MAX BITRATE"
desc="Setting a max bitrate can make file sizes more predictable at a minor cost to quality. At 720p, typical values are 2600k for VP9 or HEVC, or 4500k for H.264. Disabled if set to 0." desc="Setting a max bitrate can make file sizes more predictable at a minor cost to quality. At 720p, typical values are 2600k for VP9 or HEVC, or 4500k for H.264. Disabled if set to 0."
bind:value={ffmpegConfig.maxBitrate} bind:value={ffmpegConfig.maxBitrate}
@ -168,6 +175,7 @@
<SettingInputField <SettingInputField
inputType={SettingInputFieldType.NUMBER} inputType={SettingInputFieldType.NUMBER}
{disabled}
label="THREADS" label="THREADS"
desc="Higher values lead to faster encoding, but leave less room for the server to process other tasks while active. This value should not be more than the number of CPU cores. Maximizes utilization if set to 0." desc="Higher values lead to faster encoding, but leave less room for the server to process other tasks while active. This value should not be more than the number of CPU cores. Maximizes utilization if set to 0."
bind:value={ffmpegConfig.threads} bind:value={ffmpegConfig.threads}
@ -176,6 +184,7 @@
<SettingSelect <SettingSelect
label="TRANSCODE POLICY" label="TRANSCODE POLICY"
{disabled}
desc="Policy for when a video should be transcoded." desc="Policy for when a video should be transcoded."
bind:value={ffmpegConfig.transcode} bind:value={ffmpegConfig.transcode}
name="transcode" name="transcode"
@ -199,6 +208,7 @@
<SettingSelect <SettingSelect
label="HARDWARE ACCELERATION" label="HARDWARE ACCELERATION"
{disabled}
desc="Experimental. Much faster, but will have lower quality at the same bitrate. This setting is 'best effort': it will fallback to software transcoding on failure. VP9 may or may not work depending on your hardware." desc="Experimental. Much faster, but will have lower quality at the same bitrate. This setting is 'best effort': it will fallback to software transcoding on failure. VP9 may or may not work depending on your hardware."
bind:value={ffmpegConfig.accel} bind:value={ffmpegConfig.accel}
name="accel" name="accel"
@ -222,6 +232,7 @@
<SettingSelect <SettingSelect
label="TONE-MAPPING" label="TONE-MAPPING"
{disabled}
desc="Attempts to preserve the appearance of HDR videos when converted to SDR. Each algorithm makes different tradeoffs for color, detail and brightness. Hable preserves detail, Mobius preserves color, and Reinhard preserves brightness." desc="Attempts to preserve the appearance of HDR videos when converted to SDR. Each algorithm makes different tradeoffs for color, detail and brightness. Hable preserves detail, Mobius preserves color, and Reinhard preserves brightness."
bind:value={ffmpegConfig.tonemap} bind:value={ffmpegConfig.tonemap}
name="tonemap" name="tonemap"
@ -248,6 +259,7 @@
<SettingSwitch <SettingSwitch
title="TWO-PASS ENCODING" title="TWO-PASS ENCODING"
{disabled}
subtitle="Transcode in two passes to produce better encoded videos. When max bitrate is enabled (required for it to work with H.264 and HEVC), this mode uses a bitrate range based on the max bitrate and ignores CRF. For VP9, CRF can be used if max bitrate is disabled." subtitle="Transcode in two passes to produce better encoded videos. When max bitrate is enabled (required for it to work with H.264 and HEVC), this mode uses a bitrate range based on the max bitrate and ignores CRF. For VP9, CRF can be used if max bitrate is disabled."
bind:checked={ffmpegConfig.twoPass} bind:checked={ffmpegConfig.twoPass}
isEdited={!(ffmpegConfig.twoPass === savedConfig.twoPass)} isEdited={!(ffmpegConfig.twoPass === savedConfig.twoPass)}
@ -260,6 +272,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</div> </div>
</form> </form>

View File

@ -11,6 +11,7 @@
import SettingInputField, { SettingInputFieldType } from '../setting-input-field.svelte'; import SettingInputField, { SettingInputFieldType } from '../setting-input-field.svelte';
export let jobConfig: SystemConfigJobDto; // this is the config that is being edited export let jobConfig: SystemConfigJobDto; // this is the config that is being edited
export let disabled = false;
let savedConfig: SystemConfigJobDto; let savedConfig: SystemConfigJobDto;
let defaultConfig: SystemConfigJobDto; let defaultConfig: SystemConfigJobDto;
@ -78,6 +79,7 @@
<div class="ml-4 mt-4 flex flex-col gap-4"> <div class="ml-4 mt-4 flex flex-col gap-4">
<SettingInputField <SettingInputField
inputType={SettingInputFieldType.NUMBER} inputType={SettingInputFieldType.NUMBER}
{disabled}
label="{api.getJobName(jobName)} Concurrency" label="{api.getJobName(jobName)} Concurrency"
desc="" desc=""
bind:value={jobConfig[jobName].concurrency} bind:value={jobConfig[jobName].concurrency}
@ -93,6 +95,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</div> </div>
</form> </form>

View File

@ -11,6 +11,8 @@
import SettingInputField, { SettingInputFieldType } from '../setting-input-field.svelte'; import SettingInputField, { SettingInputFieldType } from '../setting-input-field.svelte';
import SettingSwitch from '../setting-switch.svelte'; import SettingSwitch from '../setting-switch.svelte';
export let disabled = false;
let config: SystemConfigDto; let config: SystemConfigDto;
let defaultConfig: SystemConfigDto; let defaultConfig: SystemConfigDto;
@ -56,6 +58,7 @@
<SettingSwitch <SettingSwitch
title="Enabled" title="Enabled"
subtitle="Use machine learning features" subtitle="Use machine learning features"
{disabled}
bind:checked={config.machineLearning.enabled} bind:checked={config.machineLearning.enabled}
/> />
@ -67,7 +70,7 @@
desc="URL of machine learning server" desc="URL of machine learning server"
bind:value={config.machineLearning.url} bind:value={config.machineLearning.url}
required={true} required={true}
disabled={!config.machineLearning.enabled} disabled={disabled || !config.machineLearning.enabled}
isEdited={!(config.machineLearning.url === config.machineLearning.url)} isEdited={!(config.machineLearning.url === config.machineLearning.url)}
/> />
@ -75,20 +78,20 @@
title="SMART SEARCH" title="SMART SEARCH"
subtitle="Extract CLIP embeddings for smart search" subtitle="Extract CLIP embeddings for smart search"
bind:checked={config.machineLearning.clipEncodeEnabled} bind:checked={config.machineLearning.clipEncodeEnabled}
disabled={!config.machineLearning.enabled} disabled={disabled || !config.machineLearning.enabled}
/> />
<SettingSwitch <SettingSwitch
title="FACIAL RECOGNITION" title="FACIAL RECOGNITION"
subtitle="Recognize and group faces in photos" subtitle="Recognize and group faces in photos"
disabled={!config.machineLearning.enabled} disabled={disabled || !config.machineLearning.enabled}
bind:checked={config.machineLearning.facialRecognitionEnabled} bind:checked={config.machineLearning.facialRecognitionEnabled}
/> />
<SettingSwitch <SettingSwitch
title="IMAGE TAGGING" title="IMAGE TAGGING"
subtitle="Tag and classify images" subtitle="Tag and classify images"
disabled={!config.machineLearning.enabled} disabled={disabled || !config.machineLearning.enabled}
bind:checked={config.machineLearning.tagImageEnabled} bind:checked={config.machineLearning.tagImageEnabled}
/> />
@ -97,6 +100,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(config, defaultConfig)} showResetToDefault={!isEqual(config, defaultConfig)}
{disabled}
/> />
</form> </form>
</div> </div>

View File

@ -13,6 +13,7 @@
import SettingSwitch from '../setting-switch.svelte'; import SettingSwitch from '../setting-switch.svelte';
export let oauthConfig: SystemConfigOAuthDto; export let oauthConfig: SystemConfigOAuthDto;
export let disabled = false;
let savedConfig: SystemConfigOAuthDto; let savedConfig: SystemConfigOAuthDto;
let defaultConfig: SystemConfigOAuthDto; let defaultConfig: SystemConfigOAuthDto;
@ -117,14 +118,14 @@
>. >.
</p> </p>
<SettingSwitch title="ENABLE" bind:checked={oauthConfig.enabled} /> <SettingSwitch {disabled} title="ENABLE" bind:checked={oauthConfig.enabled} />
<hr /> <hr />
<SettingInputField <SettingInputField
inputType={SettingInputFieldType.TEXT} inputType={SettingInputFieldType.TEXT}
label="ISSUER URL" label="ISSUER URL"
bind:value={oauthConfig.issuerUrl} bind:value={oauthConfig.issuerUrl}
required={true} required={true}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.issuerUrl == savedConfig.issuerUrl)} isEdited={!(oauthConfig.issuerUrl == savedConfig.issuerUrl)}
/> />
@ -133,7 +134,7 @@
label="CLIENT ID" label="CLIENT ID"
bind:value={oauthConfig.clientId} bind:value={oauthConfig.clientId}
required={true} required={true}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.clientId == savedConfig.clientId)} isEdited={!(oauthConfig.clientId == savedConfig.clientId)}
/> />
@ -142,7 +143,7 @@
label="CLIENT SECRET" label="CLIENT SECRET"
bind:value={oauthConfig.clientSecret} bind:value={oauthConfig.clientSecret}
required={true} required={true}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.clientSecret == savedConfig.clientSecret)} isEdited={!(oauthConfig.clientSecret == savedConfig.clientSecret)}
/> />
@ -151,7 +152,7 @@
label="SCOPE" label="SCOPE"
bind:value={oauthConfig.scope} bind:value={oauthConfig.scope}
required={true} required={true}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.scope == savedConfig.scope)} isEdited={!(oauthConfig.scope == savedConfig.scope)}
/> />
@ -161,7 +162,7 @@
desc="Automatically set the user's storage label to the value of this claim." desc="Automatically set the user's storage label to the value of this claim."
bind:value={oauthConfig.storageLabelClaim} bind:value={oauthConfig.storageLabelClaim}
required={true} required={true}
disabled={!oauthConfig.storageLabelClaim} disabled={disabled || !oauthConfig.storageLabelClaim}
isEdited={!(oauthConfig.storageLabelClaim == savedConfig.storageLabelClaim)} isEdited={!(oauthConfig.storageLabelClaim == savedConfig.storageLabelClaim)}
/> />
@ -170,7 +171,7 @@
label="BUTTON TEXT" label="BUTTON TEXT"
bind:value={oauthConfig.buttonText} bind:value={oauthConfig.buttonText}
required={false} required={false}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.buttonText == savedConfig.buttonText)} isEdited={!(oauthConfig.buttonText == savedConfig.buttonText)}
/> />
@ -178,20 +179,20 @@
title="AUTO REGISTER" title="AUTO REGISTER"
subtitle="Automatically register new users after signing in with OAuth" subtitle="Automatically register new users after signing in with OAuth"
bind:checked={oauthConfig.autoRegister} bind:checked={oauthConfig.autoRegister}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
/> />
<SettingSwitch <SettingSwitch
title="AUTO LAUNCH" title="AUTO LAUNCH"
subtitle="Start the OAuth login flow automatically upon navigating to the login page" subtitle="Start the OAuth login flow automatically upon navigating to the login page"
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
bind:checked={oauthConfig.autoLaunch} bind:checked={oauthConfig.autoLaunch}
/> />
<SettingSwitch <SettingSwitch
title="MOBILE REDIRECT URI OVERRIDE" title="MOBILE REDIRECT URI OVERRIDE"
subtitle="Enable when `app.immich:/` is an invalid redirect URI." subtitle="Enable when `app.immich:/` is an invalid redirect URI."
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
on:click={() => handleToggleOverride()} on:click={() => handleToggleOverride()}
bind:checked={oauthConfig.mobileOverrideEnabled} bind:checked={oauthConfig.mobileOverrideEnabled}
/> />
@ -202,7 +203,7 @@
label="MOBILE REDIRECT URI" label="MOBILE REDIRECT URI"
bind:value={oauthConfig.mobileRedirectUri} bind:value={oauthConfig.mobileRedirectUri}
required={true} required={true}
disabled={!oauthConfig.enabled} disabled={disabled || !oauthConfig.enabled}
isEdited={!(oauthConfig.mobileRedirectUri == savedConfig.mobileRedirectUri)} isEdited={!(oauthConfig.mobileRedirectUri == savedConfig.mobileRedirectUri)}
/> />
{/if} {/if}
@ -212,6 +213,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</form> </form>
</div> </div>

View File

@ -12,6 +12,7 @@
import SettingSwitch from '../setting-switch.svelte'; import SettingSwitch from '../setting-switch.svelte';
export let passwordLoginConfig: SystemConfigPasswordLoginDto; // this is the config that is being edited export let passwordLoginConfig: SystemConfigPasswordLoginDto; // this is the config that is being edited
export let disabled = false;
let savedConfig: SystemConfigPasswordLoginDto; let savedConfig: SystemConfigPasswordLoginDto;
let defaultConfig: SystemConfigPasswordLoginDto; let defaultConfig: SystemConfigPasswordLoginDto;
@ -100,6 +101,7 @@
<div class="ml-4"> <div class="ml-4">
<SettingSwitch <SettingSwitch
title="ENABLED" title="ENABLED"
{disabled}
subtitle="Login with email and password" subtitle="Login with email and password"
bind:checked={passwordLoginConfig.enabled} bind:checked={passwordLoginConfig.enabled}
/> />
@ -109,6 +111,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</div> </div>
</div> </div>

View File

@ -5,6 +5,7 @@
const dispatch = createEventDispatcher(); const dispatch = createEventDispatcher();
export let showResetToDefault = true; export let showResetToDefault = true;
export let disabled = false;
</script> </script>
<div class="mt-8 flex justify-between gap-2"> <div class="mt-8 flex justify-between gap-2">
@ -20,7 +21,7 @@
</div> </div>
<div class="right"> <div class="right">
<Button size="sm" color="gray" on:click={() => dispatch('reset')}>Reset</Button> <Button {disabled} size="sm" color="gray" on:click={() => dispatch('reset')}>Reset</Button>
<Button size="sm" on:click={() => dispatch('save')}>Save</Button> <Button {disabled} size="sm" on:click={() => dispatch('save')}>Save</Button>
</div> </div>
</div> </div>

View File

@ -9,6 +9,7 @@
export let name = ''; export let name = '';
export let isEdited = false; export let isEdited = false;
export let number = false; export let number = false;
export let disabled = false;
const handleChange = (e: Event) => { const handleChange = (e: Event) => {
value = (e.target as HTMLInputElement).value; value = (e.target as HTMLInputElement).value;
@ -40,6 +41,7 @@
<select <select
class="immich-form-input w-full pb-2" class="immich-form-input w-full pb-2"
{disabled}
aria-describedby={desc ? `${name}-desc` : undefined} aria-describedby={desc ? `${name}-desc` : undefined}
{name} {name}
id="{name}-select" id="{name}-select"

View File

@ -16,6 +16,7 @@
export let storageConfig: SystemConfigStorageTemplateDto; export let storageConfig: SystemConfigStorageTemplateDto;
export let user: UserResponseDto; export let user: UserResponseDto;
export let disabled = false;
let savedConfig: SystemConfigStorageTemplateDto; let savedConfig: SystemConfigStorageTemplateDto;
let defaultConfig: SystemConfigStorageTemplateDto; let defaultConfig: SystemConfigStorageTemplateDto;
@ -178,6 +179,7 @@
<label class="text-xs" for="preset-select">PRESET</label> <label class="text-xs" for="preset-select">PRESET</label>
<select <select
class="mt-2 rounded-lg bg-slate-200 p-2 text-sm hover:cursor-pointer dark:bg-gray-600" class="mt-2 rounded-lg bg-slate-200 p-2 text-sm hover:cursor-pointer dark:bg-gray-600"
{disabled}
name="presets" name="presets"
id="preset-select" id="preset-select"
bind:value={selectedPreset} bind:value={selectedPreset}
@ -191,6 +193,7 @@
<div class="flex gap-2 align-bottom"> <div class="flex gap-2 align-bottom">
<SettingInputField <SettingInputField
label="TEMPLATE" label="TEMPLATE"
{disabled}
required required
inputType={SettingInputFieldType.TEXT} inputType={SettingInputFieldType.TEXT}
bind:value={storageConfig.template} bind:value={storageConfig.template}
@ -216,6 +219,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</form> </form>
</div> </div>

View File

@ -10,6 +10,7 @@
} from '$lib/components/shared-components/notification/notification'; } from '$lib/components/shared-components/notification/notification';
export let thumbnailConfig: SystemConfigThumbnailDto; // this is the config that is being edited export let thumbnailConfig: SystemConfigThumbnailDto; // this is the config that is being edited
export let disabled = false;
let savedConfig: SystemConfigThumbnailDto; let savedConfig: SystemConfigThumbnailDto;
let defaultConfig: SystemConfigThumbnailDto; let defaultConfig: SystemConfigThumbnailDto;
@ -91,6 +92,7 @@
]} ]}
name="resolution" name="resolution"
isEdited={!(thumbnailConfig.webpSize === savedConfig.webpSize)} isEdited={!(thumbnailConfig.webpSize === savedConfig.webpSize)}
{disabled}
/> />
<SettingSelect <SettingSelect
@ -104,6 +106,7 @@
]} ]}
name="resolution" name="resolution"
isEdited={!(thumbnailConfig.jpegSize === savedConfig.jpegSize)} isEdited={!(thumbnailConfig.jpegSize === savedConfig.jpegSize)}
{disabled}
/> />
</div> </div>
@ -113,6 +116,7 @@
on:save={saveSetting} on:save={saveSetting}
on:reset-to-default={resetToDefault} on:reset-to-default={resetToDefault}
showResetToDefault={!isEqual(savedConfig, defaultConfig)} showResetToDefault={!isEqual(savedConfig, defaultConfig)}
{disabled}
/> />
</div> </div>
</form> </form>

View File

@ -1,10 +1,9 @@
<script lang="ts"> <script lang="ts">
import { createEventDispatcher, onMount } from 'svelte'; import { createEventDispatcher, onMount } from 'svelte';
import KeyVariant from 'svelte-material-icons/KeyVariant.svelte'; import KeyVariant from 'svelte-material-icons/KeyVariant.svelte';
import { handleError } from '../../utils/handle-error'; import { copyToClipboard } from '@api';
import FullScreenModal from '../shared-components/full-screen-modal.svelte';
import { notificationController, NotificationType } from '../shared-components/notification/notification';
import Button from '../elements/buttons/button.svelte'; import Button from '../elements/buttons/button.svelte';
import FullScreenModal from '../shared-components/full-screen-modal.svelte';
export let secret = ''; export let secret = '';
@ -16,17 +15,6 @@
const module = await import('copy-image-clipboard'); const module = await import('copy-image-clipboard');
canCopyImagesToClipboard = module.canCopyImagesToClipboard(); canCopyImagesToClipboard = module.canCopyImagesToClipboard();
}); });
const handleCopy = async () => {
try {
await navigator.clipboard.writeText(secret);
notificationController.show({
message: 'Copied to clipboard!',
type: NotificationType.Info,
});
} catch (error) {
handleError(error, 'Unable to copy to clipboard');
}
};
</script> </script>
<FullScreenModal> <FullScreenModal>
@ -51,7 +39,7 @@
<div class="mt-8 flex w-full gap-4 px-4"> <div class="mt-8 flex w-full gap-4 px-4">
{#if canCopyImagesToClipboard} {#if canCopyImagesToClipboard}
<Button on:click={() => handleCopy()} fullwidth>Copy to Clipboard</Button> <Button on:click={() => copyToClipboard(secret)} fullwidth>Copy to Clipboard</Button>
{/if} {/if}
<Button on:click={() => handleDone()} fullwidth>Done</Button> <Button on:click={() => handleDone()} fullwidth>Done</Button>
</div> </div>

View File

@ -5,7 +5,7 @@
import SettingSwitch from '$lib/components/admin-page/settings/setting-switch.svelte'; import SettingSwitch from '$lib/components/admin-page/settings/setting-switch.svelte';
import Button from '$lib/components/elements/buttons/button.svelte'; import Button from '$lib/components/elements/buttons/button.svelte';
import { handleError } from '$lib/utils/handle-error'; import { handleError } from '$lib/utils/handle-error';
import { api, SharedLinkResponseDto, SharedLinkType } from '@api'; import { api, copyToClipboard, SharedLinkResponseDto, SharedLinkType } from '@api';
import { createEventDispatcher, onMount } from 'svelte'; import { createEventDispatcher, onMount } from 'svelte';
import Link from 'svelte-material-icons/Link.svelte'; import Link from 'svelte-material-icons/Link.svelte';
import BaseModal from '../base-modal.svelte'; import BaseModal from '../base-modal.svelte';
@ -80,12 +80,7 @@
return; return;
} }
try { await copyToClipboard(sharedLink);
await navigator.clipboard.writeText(sharedLink);
notificationController.show({ message: 'Copied to clipboard!', type: NotificationType.Info });
} catch (e) {
handleError(e, 'Cannot copy to clipboard, make sure you are accessing the page through https');
}
}; };
const getExpirationTimeInMillisecond = () => { const getExpirationTimeInMillisecond = () => {

View File

@ -12,6 +12,7 @@ export const featureFlags = writable<FeatureFlags>({
oauth: true, oauth: true,
oauthAutoLaunch: true, oauthAutoLaunch: true,
passwordLogin: true, passwordLogin: true,
configFile: false,
}); });
export const loadFeatureFlags = async () => { export const loadFeatureFlags = async () => {

View File

@ -20,7 +20,7 @@ export const addAssetsToAlbum = async (albumId: string, assetIds: Array<string>)
return results; return results;
}); });
const downloadBlob = (data: Blob, filename: string) => { export const downloadBlob = (data: Blob, filename: string) => {
const url = URL.createObjectURL(data); const url = URL.createObjectURL(data);
const anchor = document.createElement('a'); const anchor = document.createElement('a');

View File

@ -1,7 +1,7 @@
<script lang="ts"> <script lang="ts">
import ControlAppBar from '$lib/components/shared-components/control-app-bar.svelte'; import ControlAppBar from '$lib/components/shared-components/control-app-bar.svelte';
import ArrowLeft from 'svelte-material-icons/ArrowLeft.svelte'; import ArrowLeft from 'svelte-material-icons/ArrowLeft.svelte';
import { api, SharedLinkResponseDto } from '@api'; import { api, copyToClipboard, SharedLinkResponseDto } from '@api';
import { goto } from '$app/navigation'; import { goto } from '$app/navigation';
import SharedLinkCard from '$lib/components/sharedlinks-page/shared-link-card.svelte'; import SharedLinkCard from '$lib/components/sharedlinks-page/shared-link-card.svelte';
import { import {
@ -49,12 +49,7 @@
}; };
const handleCopyLink = async (key: string) => { const handleCopyLink = async (key: string) => {
const link = `${window.location.origin}/share/${key}`; await copyToClipboard(`${window.location.origin}/share/${key}`);
await navigator.clipboard.writeText(link);
notificationController.show({
message: 'Link copied to clipboard',
type: NotificationType.Info,
});
}; };
</script> </script>

View File

@ -1,15 +1,11 @@
<script> <script>
import { page } from '$app/stores'; import { page } from '$app/stores';
import Message from 'svelte-material-icons/Message.svelte'; import ImmichLogo from '$lib/components/shared-components/immich-logo.svelte';
import PartyPopper from 'svelte-material-icons/PartyPopper.svelte';
import CodeTags from 'svelte-material-icons/CodeTags.svelte'; import CodeTags from 'svelte-material-icons/CodeTags.svelte';
import ContentCopy from 'svelte-material-icons/ContentCopy.svelte'; import ContentCopy from 'svelte-material-icons/ContentCopy.svelte';
import { import Message from 'svelte-material-icons/Message.svelte';
notificationController, import PartyPopper from 'svelte-material-icons/PartyPopper.svelte';
NotificationType, import { copyToClipboard } from '../api/utils';
} from '$lib/components/shared-components/notification/notification';
import { handleError } from '$lib/utils/handle-error';
import ImmichLogo from '$lib/components/shared-components/immich-logo.svelte';
const handleCopy = async () => { const handleCopy = async () => {
// //
@ -18,15 +14,7 @@
return; return;
} }
try { await copyToClipboard(`${error.message} - ${error.code}\n${error.stack}`);
await navigator.clipboard.writeText(`${error.message} - ${error.code}\n${error.stack}`);
notificationController.show({
type: NotificationType.Info,
message: 'Copied error to clipboard',
});
} catch (error) {
handleError(error, 'Unable to copy to clipboard');
}
}; };
</script> </script>

View File

@ -8,8 +8,15 @@
import SettingAccordion from '$lib/components/admin-page/settings/setting-accordion.svelte'; import SettingAccordion from '$lib/components/admin-page/settings/setting-accordion.svelte';
import StorageTemplateSettings from '$lib/components/admin-page/settings/storage-template/storage-template-settings.svelte'; import StorageTemplateSettings from '$lib/components/admin-page/settings/storage-template/storage-template-settings.svelte';
import ThumbnailSettings from '$lib/components/admin-page/settings/thumbnail/thumbnail-settings.svelte'; import ThumbnailSettings from '$lib/components/admin-page/settings/thumbnail/thumbnail-settings.svelte';
import Button from '$lib/components/elements/buttons/button.svelte';
import LoadingSpinner from '$lib/components/shared-components/loading-spinner.svelte'; import LoadingSpinner from '$lib/components/shared-components/loading-spinner.svelte';
import { api } from '@api'; import { downloadManager } from '$lib/stores/download';
import { featureFlags } from '$lib/stores/feature-flags.store';
import { downloadBlob } from '$lib/utils/asset-utils';
import { SystemConfigDto, api, copyToClipboard } from '@api';
import Alert from 'svelte-material-icons/Alert.svelte';
import ContentCopy from 'svelte-material-icons/ContentCopy.svelte';
import Download from 'svelte-material-icons/Download.svelte';
import type { PageData } from './$types'; import type { PageData } from './$types';
export let data: PageData; export let data: PageData;
@ -18,21 +25,47 @@
const { data } = await api.systemConfigApi.getConfig(); const { data } = await api.systemConfigApi.getConfig();
return data; return data;
}; };
const downloadConfig = (configs: SystemConfigDto) => {
const blob = new Blob([JSON.stringify(configs, null, 2)], { type: 'application/json' });
const downloadKey = 'immich-config.json';
downloadManager.add(downloadKey, blob.size);
downloadManager.update(downloadKey, blob.size);
downloadBlob(blob, downloadKey);
setTimeout(() => downloadManager.clear(downloadKey), 5_000);
};
</script> </script>
{#if $featureFlags.configFile}
<div class="mb-8 flex flex-row items-center gap-2 rounded-md bg-gray-100 p-3 dark:bg-gray-800">
<Alert class="text-yellow-400" size={18} />
<h2 class="text-md text-immich-primary dark:text-immich-dark-primary">Config is currently set by a config file</h2>
</div>
{/if}
<section class=""> <section class="">
{#await getConfig()} {#await getConfig()}
<LoadingSpinner /> <LoadingSpinner />
{:then configs} {:then configs}
<div class="flex justify-end gap-2">
<Button size="sm" on:click={() => copyToClipboard(JSON.stringify(configs, null, 2))}>
<ContentCopy size="18" />
<span class="pl-2">Copy to Clipboard</span>
</Button>
<Button size="sm" on:click={() => downloadConfig(configs)}>
<Download size="18" />
<span class="pl-2">Export as JSON</span>
</Button>
</div>
<SettingAccordion title="Thumbnail Settings" subtitle="Manage the resolution of thumbnail sizes"> <SettingAccordion title="Thumbnail Settings" subtitle="Manage the resolution of thumbnail sizes">
<ThumbnailSettings thumbnailConfig={configs.thumbnail} /> <ThumbnailSettings disabled={$featureFlags.configFile} thumbnailConfig={configs.thumbnail} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion <SettingAccordion
title="FFmpeg Settings" title="FFmpeg Settings"
subtitle="Manage the resolution and encoding information of the video files" subtitle="Manage the resolution and encoding information of the video files"
> >
<FFmpegSettings ffmpegConfig={configs.ffmpeg} /> <FFmpegSettings disabled={$featureFlags.configFile} ffmpegConfig={configs.ffmpeg} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion <SettingAccordion
@ -40,19 +73,19 @@
subtitle="Manage job concurrency" subtitle="Manage job concurrency"
isOpen={$page.url.searchParams.get('open') === 'job-settings'} isOpen={$page.url.searchParams.get('open') === 'job-settings'}
> >
<JobSettings jobConfig={configs.job} /> <JobSettings disabled={$featureFlags.configFile} jobConfig={configs.job} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion title="Password Authentication" subtitle="Manage login with password settings"> <SettingAccordion title="Password Authentication" subtitle="Manage login with password settings">
<PasswordLoginSettings passwordLoginConfig={configs.passwordLogin} /> <PasswordLoginSettings disabled={$featureFlags.configFile} passwordLoginConfig={configs.passwordLogin} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion title="OAuth Authentication" subtitle="Manage the login with OAuth settings"> <SettingAccordion title="OAuth Authentication" subtitle="Manage the login with OAuth settings">
<OAuthSettings oauthConfig={configs.oauth} /> <OAuthSettings disabled={$featureFlags.configFile} oauthConfig={configs.oauth} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion title="Machine Learning" subtitle="Manage machine learning settings"> <SettingAccordion title="Machine Learning" subtitle="Manage machine learning settings">
<MachineLearningSettings /> <MachineLearningSettings disabled={$featureFlags.configFile} />
</SettingAccordion> </SettingAccordion>
<SettingAccordion <SettingAccordion
@ -60,7 +93,11 @@
subtitle="Manage the folder structure and file name of the upload asset" subtitle="Manage the folder structure and file name of the upload asset"
isOpen={$page.url.searchParams.get('open') === 'storage-template'} isOpen={$page.url.searchParams.get('open') === 'storage-template'}
> >
<StorageTemplateSettings storageConfig={configs.storageTemplate} user={data.user} /> <StorageTemplateSettings
disabled={$featureFlags.configFile}
storageConfig={configs.storageTemplate}
user={data.user}
/>
</SettingAccordion> </SettingAccordion>
{/await} {/await}
</section> </section>