mirror of
https://github.com/immich-app/immich
synced 2025-06-11 00:48:28 +00:00

* refactor: store latest immich version available on the server * don't store admins acknowledgement * merge main * fix: api * feat: custom interval * pr feedback * remove unused code * update environment-variables * pr feedback * ci: fix server tests * fix: dart number * pr feedback * remove proxy * pr feedback * feat: make stringToVersion more flexible * feat(web): disable check * feat: working version * remove env * fix: check if interval exists when updating the interval * feat: show last check * fix: tests * fix: remove availableVersion when updated * fix merge * fix: web * fix e2e tests * merge main * merge main * pr feedback * pr feedback * fix: tests * pr feedback * pr feedback * pr feedback * pr feedback * pr feedback * fix: migration * regenerate api * fix: typo * fix: compare versions * pr feedback * fix * pr feedback * fix: checkIntervalTime on startup * refactor: websockets and interval logic * chore: open api * chore: remove unused code * fix: use interval instead of cron * mobile: handle WS event data as json object --------- Co-authored-by: Jason Rasmussen <jrasm91@gmail.com> Co-authored-by: shalong-tanwen <139912620+shalong-tanwen@users.noreply.github.com>
235 lines
6.5 KiB
TypeScript
235 lines
6.5 KiB
TypeScript
import { QueueName } from '@app/domain';
|
|
import { Column, Entity, PrimaryColumn } from 'typeorm';
|
|
|
|
@Entity('system_config')
|
|
export class SystemConfigEntity<T = SystemConfigValue> {
|
|
@PrimaryColumn()
|
|
key!: SystemConfigKey;
|
|
|
|
@Column({ type: 'varchar', nullable: true, transformer: { to: JSON.stringify, from: JSON.parse } })
|
|
value!: T;
|
|
}
|
|
|
|
export type SystemConfigValue = string | number | boolean;
|
|
|
|
// dot notation matches path in `SystemConfig`
|
|
export enum SystemConfigKey {
|
|
FFMPEG_CRF = 'ffmpeg.crf',
|
|
FFMPEG_THREADS = 'ffmpeg.threads',
|
|
FFMPEG_PRESET = 'ffmpeg.preset',
|
|
FFMPEG_TARGET_VIDEO_CODEC = 'ffmpeg.targetVideoCodec',
|
|
FFMPEG_TARGET_AUDIO_CODEC = 'ffmpeg.targetAudioCodec',
|
|
FFMPEG_TARGET_RESOLUTION = 'ffmpeg.targetResolution',
|
|
FFMPEG_MAX_BITRATE = 'ffmpeg.maxBitrate',
|
|
FFMPEG_BFRAMES = 'ffmpeg.bframes',
|
|
FFMPEG_REFS = 'ffmpeg.refs',
|
|
FFMPEG_GOP_SIZE = 'ffmpeg.gopSize',
|
|
FFMPEG_NPL = 'ffmpeg.npl',
|
|
FFMPEG_TEMPORAL_AQ = 'ffmpeg.temporalAQ',
|
|
FFMPEG_CQ_MODE = 'ffmpeg.cqMode',
|
|
FFMPEG_TWO_PASS = 'ffmpeg.twoPass',
|
|
FFMPEG_TRANSCODE = 'ffmpeg.transcode',
|
|
FFMPEG_ACCEL = 'ffmpeg.accel',
|
|
FFMPEG_TONEMAP = 'ffmpeg.tonemap',
|
|
|
|
JOB_THUMBNAIL_GENERATION_CONCURRENCY = 'job.thumbnailGeneration.concurrency',
|
|
JOB_METADATA_EXTRACTION_CONCURRENCY = 'job.metadataExtraction.concurrency',
|
|
JOB_VIDEO_CONVERSION_CONCURRENCY = 'job.videoConversion.concurrency',
|
|
JOB_OBJECT_TAGGING_CONCURRENCY = 'job.objectTagging.concurrency',
|
|
JOB_RECOGNIZE_FACES_CONCURRENCY = 'job.recognizeFaces.concurrency',
|
|
JOB_CLIP_ENCODING_CONCURRENCY = 'job.clipEncoding.concurrency',
|
|
JOB_BACKGROUND_TASK_CONCURRENCY = 'job.backgroundTask.concurrency',
|
|
JOB_STORAGE_TEMPLATE_MIGRATION_CONCURRENCY = 'job.storageTemplateMigration.concurrency',
|
|
JOB_SEARCH_CONCURRENCY = 'job.search.concurrency',
|
|
JOB_SIDECAR_CONCURRENCY = 'job.sidecar.concurrency',
|
|
JOB_LIBRARY_CONCURRENCY = 'job.library.concurrency',
|
|
JOB_MIGRATION_CONCURRENCY = 'job.migration.concurrency',
|
|
|
|
MACHINE_LEARNING_ENABLED = 'machineLearning.enabled',
|
|
MACHINE_LEARNING_URL = 'machineLearning.url',
|
|
|
|
MACHINE_LEARNING_CLASSIFICATION_ENABLED = 'machineLearning.classification.enabled',
|
|
MACHINE_LEARNING_CLASSIFICATION_MODEL_NAME = 'machineLearning.classification.modelName',
|
|
MACHINE_LEARNING_CLASSIFICATION_MIN_SCORE = 'machineLearning.classification.minScore',
|
|
|
|
MACHINE_LEARNING_CLIP_ENABLED = 'machineLearning.clip.enabled',
|
|
MACHINE_LEARNING_CLIP_MODEL_NAME = 'machineLearning.clip.modelName',
|
|
|
|
MACHINE_LEARNING_FACIAL_RECOGNITION_ENABLED = 'machineLearning.facialRecognition.enabled',
|
|
MACHINE_LEARNING_FACIAL_RECOGNITION_MODEL_NAME = 'machineLearning.facialRecognition.modelName',
|
|
MACHINE_LEARNING_FACIAL_RECOGNITION_MIN_SCORE = 'machineLearning.facialRecognition.minScore',
|
|
MACHINE_LEARNING_FACIAL_RECOGNITION_MAX_DISTANCE = 'machineLearning.facialRecognition.maxDistance',
|
|
MACHINE_LEARNING_FACIAL_RECOGNITION_MIN_FACES = 'machineLearning.facialRecognition.minFaces',
|
|
|
|
MAP_ENABLED = 'map.enabled',
|
|
MAP_TILE_URL = 'map.tileUrl',
|
|
|
|
REVERSE_GEOCODING_ENABLED = 'reverseGeocoding.enabled',
|
|
REVERSE_GEOCODING_CITIES_FILE_OVERRIDE = 'reverseGeocoding.citiesFileOverride',
|
|
|
|
NEW_VERSION_CHECK_ENABLED = 'newVersionCheck.enabled',
|
|
|
|
OAUTH_ENABLED = 'oauth.enabled',
|
|
OAUTH_ISSUER_URL = 'oauth.issuerUrl',
|
|
OAUTH_CLIENT_ID = 'oauth.clientId',
|
|
OAUTH_CLIENT_SECRET = 'oauth.clientSecret',
|
|
OAUTH_SCOPE = 'oauth.scope',
|
|
OAUTH_STORAGE_LABEL_CLAIM = 'oauth.storageLabelClaim',
|
|
OAUTH_AUTO_LAUNCH = 'oauth.autoLaunch',
|
|
OAUTH_BUTTON_TEXT = 'oauth.buttonText',
|
|
OAUTH_AUTO_REGISTER = 'oauth.autoRegister',
|
|
OAUTH_MOBILE_OVERRIDE_ENABLED = 'oauth.mobileOverrideEnabled',
|
|
OAUTH_MOBILE_REDIRECT_URI = 'oauth.mobileRedirectUri',
|
|
|
|
PASSWORD_LOGIN_ENABLED = 'passwordLogin.enabled',
|
|
|
|
STORAGE_TEMPLATE = 'storageTemplate.template',
|
|
|
|
THUMBNAIL_WEBP_SIZE = 'thumbnail.webpSize',
|
|
THUMBNAIL_JPEG_SIZE = 'thumbnail.jpegSize',
|
|
THUMBNAIL_QUALITY = 'thumbnail.quality',
|
|
THUMBNAIL_COLORSPACE = 'thumbnail.colorspace',
|
|
|
|
TRASH_ENABLED = 'trash.enabled',
|
|
TRASH_DAYS = 'trash.days',
|
|
|
|
THEME_CUSTOM_CSS = 'theme.customCss',
|
|
}
|
|
|
|
export enum TranscodePolicy {
|
|
ALL = 'all',
|
|
OPTIMAL = 'optimal',
|
|
REQUIRED = 'required',
|
|
DISABLED = 'disabled',
|
|
}
|
|
|
|
export enum VideoCodec {
|
|
H264 = 'h264',
|
|
HEVC = 'hevc',
|
|
VP9 = 'vp9',
|
|
}
|
|
|
|
export enum AudioCodec {
|
|
MP3 = 'mp3',
|
|
AAC = 'aac',
|
|
LIBOPUS = 'libopus',
|
|
}
|
|
|
|
export enum TranscodeHWAccel {
|
|
NVENC = 'nvenc',
|
|
QSV = 'qsv',
|
|
VAAPI = 'vaapi',
|
|
DISABLED = 'disabled',
|
|
}
|
|
|
|
export enum ToneMapping {
|
|
HABLE = 'hable',
|
|
MOBIUS = 'mobius',
|
|
REINHARD = 'reinhard',
|
|
DISABLED = 'disabled',
|
|
}
|
|
|
|
export enum CQMode {
|
|
AUTO = 'auto',
|
|
CQP = 'cqp',
|
|
ICQ = 'icq',
|
|
}
|
|
|
|
export enum Colorspace {
|
|
SRGB = 'srgb',
|
|
P3 = 'p3',
|
|
}
|
|
|
|
export enum CitiesFile {
|
|
CITIES_15000 = 'cities15000',
|
|
CITIES_5000 = 'cities5000',
|
|
CITIES_1000 = 'cities1000',
|
|
CITIES_500 = 'cities500',
|
|
}
|
|
|
|
export interface SystemConfig {
|
|
ffmpeg: {
|
|
crf: number;
|
|
threads: number;
|
|
preset: string;
|
|
targetVideoCodec: VideoCodec;
|
|
targetAudioCodec: AudioCodec;
|
|
targetResolution: string;
|
|
maxBitrate: string;
|
|
bframes: number;
|
|
refs: number;
|
|
gopSize: number;
|
|
npl: number;
|
|
temporalAQ: boolean;
|
|
cqMode: CQMode;
|
|
twoPass: boolean;
|
|
transcode: TranscodePolicy;
|
|
accel: TranscodeHWAccel;
|
|
tonemap: ToneMapping;
|
|
};
|
|
job: Record<QueueName, { concurrency: number }>;
|
|
machineLearning: {
|
|
enabled: boolean;
|
|
url: string;
|
|
classification: {
|
|
enabled: boolean;
|
|
modelName: string;
|
|
minScore: number;
|
|
};
|
|
clip: {
|
|
enabled: boolean;
|
|
modelName: string;
|
|
};
|
|
facialRecognition: {
|
|
enabled: boolean;
|
|
modelName: string;
|
|
minScore: number;
|
|
minFaces: number;
|
|
maxDistance: number;
|
|
};
|
|
};
|
|
map: {
|
|
enabled: boolean;
|
|
tileUrl: string;
|
|
};
|
|
reverseGeocoding: {
|
|
enabled: boolean;
|
|
citiesFileOverride: CitiesFile;
|
|
};
|
|
oauth: {
|
|
enabled: boolean;
|
|
issuerUrl: string;
|
|
clientId: string;
|
|
clientSecret: string;
|
|
scope: string;
|
|
storageLabelClaim: string;
|
|
buttonText: string;
|
|
autoRegister: boolean;
|
|
autoLaunch: boolean;
|
|
mobileOverrideEnabled: boolean;
|
|
mobileRedirectUri: string;
|
|
};
|
|
passwordLogin: {
|
|
enabled: boolean;
|
|
};
|
|
storageTemplate: {
|
|
template: string;
|
|
};
|
|
thumbnail: {
|
|
webpSize: number;
|
|
jpegSize: number;
|
|
quality: number;
|
|
colorspace: Colorspace;
|
|
};
|
|
newVersionCheck: {
|
|
enabled: boolean;
|
|
};
|
|
trash: {
|
|
enabled: boolean;
|
|
days: number;
|
|
};
|
|
theme: {
|
|
customCss: string;
|
|
};
|
|
}
|