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:
parent
20e0c03b39
commit
59bb727636
6
cli/src/api/open-api/api.ts
generated
6
cli/src/api/open-api/api.ts
generated
@ -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}
|
||||||
|
91
docs/docs/install/config-file.md
Normal file
91
docs/docs/install/config-file.md
Normal 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.
|
@ -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
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
---
|
---
|
||||||
sidebar_position: 100
|
sidebar_position: 80
|
||||||
---
|
---
|
||||||
|
|
||||||
import RegisterAdminUser from '../partials/_register-admin.md';
|
import RegisterAdminUser from '../partials/_register-admin.md';
|
||||||
|
BIN
mobile/openapi/doc/ServerFeaturesDto.md
generated
BIN
mobile/openapi/doc/ServerFeaturesDto.md
generated
Binary file not shown.
BIN
mobile/openapi/lib/model/server_features_dto.dart
generated
BIN
mobile/openapi/lib/model/server_features_dto.dart
generated
Binary file not shown.
BIN
mobile/openapi/test/server_features_dto_test.dart
generated
BIN
mobile/openapi/test/server_features_dto_test.dart
generated
Binary file not shown.
@ -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",
|
||||||
|
@ -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;
|
||||||
|
@ -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();
|
||||||
});
|
});
|
||||||
|
@ -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;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -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>;
|
||||||
}
|
}
|
||||||
|
@ -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', () => {
|
||||||
|
@ -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);
|
||||||
}
|
}
|
||||||
|
@ -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(),
|
||||||
};
|
};
|
||||||
|
6
web/src/api/open-api/api.ts
generated
6
web/src/api/open-api/api.ts
generated
@ -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}
|
||||||
|
@ -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;
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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"
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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>
|
||||||
|
@ -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 = () => {
|
||||||
|
@ -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 () => {
|
||||||
|
@ -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');
|
||||||
|
@ -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>
|
||||||
|
|
||||||
|
@ -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>
|
||||||
|
|
||||||
|
@ -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>
|
||||||
|
Loading…
Reference in New Issue
Block a user