mirror of
https://github.com/immich-app/immich.git
synced 2026-01-24 10:24:39 -08:00
* feat: ProcessRepository#createSpawnDuplexStream
* test: write tests for ProcessRepository#createSpawnDuplexStream
* feat: StorageRepository#createGzip,createGunzip,createPlainReadStream
* feat: backups util (args, create, restore, progress)
* feat: wait on maintenance operation lock on boot
* chore: use backup util from backup.service.ts
test: update backup.service.ts tests with new util
* feat: list/delete backups (maintenance services)
* chore: open api
fix: missing action in cli.service.ts
* chore: add missing repositories to MaintenanceModule
* refactor: move logSecret into module init
* feat: initialise StorageCore in maintenance mode
* feat: authenticate websocket requests in maintenance mode
* test: add mock for new storage fns
* feat: add MaintenanceEphemeralStateRepository
refactor: cache the secret in memory
* test: update service worker tests
* feat: add external maintenance mode status
* feat: synchronised status, restore db action
* test: backup restore service tests
* refactor: DRY end maintenance
* feat: list and delete backup routes
* feat: start action on boot
* fix: should set status on restore end
* refactor: add maintenanceStore to hold writables
* feat: sync status to web app
* feat: web impl.
* test: various utils for testings
* test: web e2e tests
* test: e2e maintenance spec
* test: update cli spec
* chore: e2e lint
* chore: lint fixes
* chore: lint fixes
* feat: start restore flow route
* test: update e2e tests
* chore: remove neon lights on maintenance action pages
* fix: use 'startRestoreFlow' on onboarding page
* chore: ignore any library folder in `docker/`
* fix: load status on boot
* feat: upload backups
* refactor: permit any .sql(.gz) to be listed/restored
* feat: download backups from list
* fix: permit uploading just .sql files
* feat: restore just .sql files
* fix: don't show backups list if logged out
* feat: system integrity check in restore flow
* test: not providing failed backups in API anymore
* test: util should also not try to use failedBackups
* fix: actually assign inputStream
* test: correct test backup prep.
* fix: ensure task is defined to show error
* test: fix docker cp command
* test: update e2e web spec to select next button
* test: update e2e api tests
* test: refactor timeouts
* chore: remove `showDelete` from maint. settings
* chore: lint
* chore: lint
* fix: make sure backups are correctly sorted for clean up
* test: update service spec
* test: adjust e2e timeout
* test: increase web timeouts for ci
* chore: move gitignore changes
* chore: additional filename validation
* refactor: better typings for integrity API
* feat: higher accuracy progress tracking
* chore: delay lock retry
* refactor: remove old maintenance settings
* refactor: clean up tailwind classes
* refactor: use while loop rather than recursive calls
* test: update service specs
* chore: check canParse too
* chore: lint
* fix: logic error causing infinite loop
* refactor: use <ProgressBar /> from ui library
* fix: create or overwrite file
* chore: i18n pass, update progress bar
* fix: wrong translation string
* chore: update colour variables
* test: update web test for new maint. page
* chore: format, fix key
* test: update tests to be more linter complaint & use new routines
* chore: update onClick -> onAction, title -> breadcrumbs
* fix: use wrench icon in admin settings sidebar
* chore: add translation strings to accordion
* chore: lint
* refactor: move maintenance worker init into service
* refactor: `maintenanceStatus` -> `getMaintenanceStatus`
refactor: `integrityCheck` -> `detectPriorInstall`
chore: add `v2.4.0` version
refactor: `/backups/list` -> `/backups`
refactor: use sendFile in download route
refactor: use separate backups permissions
chore: correct descriptions
refactor: permit handler that doesn't return promise for sendfile
* refactor: move status impl into service
refactor: add active flag to maintenance status
* refactor: split into database backup controller
* test: split api e2e tests and passing
* fix: move end button into authed default maint page
* fix: also show in restore flow
* fix: import getMaintenanceStatus
* test: split web e2e tests
* refactor: ensure detect install is consistently named
* chore: ensure admin for detect install while out of maint.
* refactor: remove state repository
* test: update maint. worker service spec
* test: split backup service spec
* refactor: rename db backup routes
* refactor: instead of param, allow bulk backup deletion
* test: update sdk use in e2e test
* test: correct deleteBackup call
* fix: correct type for serverinstall response dto
* chore: validate filename for deletion
* test: wip
* test: backups no longer take path param
* refactor: scope util to database-backups instead of backups
* fix: update worker controller with new route
* chore: use new admin page actions
* chore: remove stray comment
* test: rename outdated test
* refactor: getter pattern for maintenance secret
* refactor: `createSpawnDuplexStream` -> `spawnDuplexStream`
* refactor: prefer `Object.assign`
* refactor: remove useless try {} block
* refactor: prefer `type Props`
refactor: prefer arrow function
* refactor: use luxon API for minutesAgo
* chore: remove change to gitignore
* refactor: prefer `type Props`
* refactor: remove async from onMount
* refactor: use luxon toRelative for relative time
* refactor: duplicate logic check
* chore: open api
* refactor: begin moving code into web//services
* refactor: don't use template string with $t
* test: use dialog role to match prompt
* refactor: split actions into flow/restore
* test: fix action value
* refactor: move more service calls into web//services
* chore: should void fn return
* chore: bump 2.4.0 to 2.5.0 in controller
* chore: bump 2.4.0 to 2.5.0 in controller
* refactor: use events for web//services
* chore: open api
* chore: open api
* refactor: don't await returned promise
* refactor: remove redundant check
* refactor: add `type: command` to actions
* refactor: split backup entries into own component
* refactor: split restore flow into separate components
* refactor(web): split BackupDelete event
* chore: stylings
* chore: stylings
* fix: don't log query failure on first boot
* feat: support pg_dumpall backups
* feat: display information about each backup
* chore: i18n
* feat: rollback to restore point on migrations failure
* feat: health check after restore
* chore: format
* refactor: split health check into separate function
* refactor: split health into repository
test: write tests covering rollbacks
* fix: omit 'health' requirement from createDbBackup
* test(e2e): rollback test
* fix: wrap text in backup entry
* fix: don't shrink context menu button
* fix: correct CREATE DB syntax for postgres
* test: rename backups generated by test
* feat: add filesize to backup response dto
* feat: restore list
* feat: ui work
* fix: e2e test
* fix: e2e test
* pr feedback
* pr feedback
---------
Co-authored-by: Alex <alex.tran1502@gmail.com>
Co-authored-by: Jason Rasmussen <jason@rasm.me>
194 lines
5.5 KiB
TypeScript
194 lines
5.5 KiB
TypeScript
import { Kysely, sql } from 'kysely';
|
|
import { CommandFactory } from 'nest-commander';
|
|
import { ChildProcess, fork } from 'node:child_process';
|
|
import { dirname, join } from 'node:path';
|
|
import { Worker } from 'node:worker_threads';
|
|
import { PostgresError } from 'postgres';
|
|
import { ImmichAdminModule } from 'src/app.module';
|
|
import { DatabaseLock, ExitCode, ImmichWorker, LogLevel, SystemMetadataKey } from 'src/enum';
|
|
import { ConfigRepository } from 'src/repositories/config.repository';
|
|
import { SystemMetadataRepository } from 'src/repositories/system-metadata.repository';
|
|
import { type DB } from 'src/schema';
|
|
import { getKyselyConfig } from 'src/utils/database';
|
|
|
|
/**
|
|
* Manages worker lifecycle
|
|
*/
|
|
class Workers {
|
|
/**
|
|
* Currently running workers
|
|
*/
|
|
workers: Partial<Record<ImmichWorker, { kill: (signal: NodeJS.Signals) => Promise<void> | void }>> = {};
|
|
|
|
/**
|
|
* Fail-safe in case anything dies during restart
|
|
*/
|
|
restarting = false;
|
|
|
|
/**
|
|
* Boot all enabled workers
|
|
*/
|
|
async bootstrap() {
|
|
const isMaintenanceMode = await this.isMaintenanceMode();
|
|
const { workers } = new ConfigRepository().getEnv();
|
|
|
|
if (isMaintenanceMode) {
|
|
this.startWorker(ImmichWorker.Maintenance);
|
|
} else {
|
|
await this.waitForFreeLock();
|
|
|
|
for (const worker of workers) {
|
|
this.startWorker(worker);
|
|
}
|
|
}
|
|
}
|
|
|
|
private async isMaintenanceMode(): Promise<boolean> {
|
|
const { database } = new ConfigRepository().getEnv();
|
|
const { log: _, ...kyselyConfig } = getKyselyConfig(database.config);
|
|
const kysely = new Kysely<DB>(kyselyConfig);
|
|
const systemMetadataRepository = new SystemMetadataRepository(kysely);
|
|
|
|
try {
|
|
const value = await systemMetadataRepository.get(SystemMetadataKey.MaintenanceMode);
|
|
return value?.isMaintenanceMode || false;
|
|
} catch (error) {
|
|
// Table doesn't exist (migrations haven't run yet)
|
|
if (error instanceof PostgresError && error.code === '42P01') {
|
|
return false;
|
|
}
|
|
|
|
throw error;
|
|
} finally {
|
|
await kysely.destroy();
|
|
}
|
|
}
|
|
|
|
private async waitForFreeLock() {
|
|
const { database } = new ConfigRepository().getEnv();
|
|
const kysely = new Kysely<DB>(getKyselyConfig(database.config));
|
|
|
|
let locked = false;
|
|
while (!locked) {
|
|
locked = await kysely.connection().execute(async (conn) => {
|
|
const { rows } = await sql<{
|
|
pg_try_advisory_lock: boolean;
|
|
}>`SELECT pg_try_advisory_lock(${DatabaseLock.MaintenanceOperation})`.execute(conn);
|
|
|
|
const isLocked = rows[0].pg_try_advisory_lock;
|
|
|
|
if (isLocked) {
|
|
await sql`SELECT pg_advisory_unlock(${DatabaseLock.MaintenanceOperation})`.execute(conn);
|
|
}
|
|
|
|
return isLocked;
|
|
});
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
}
|
|
|
|
await kysely.destroy();
|
|
}
|
|
|
|
/**
|
|
* Start an individual worker
|
|
* @param name Worker
|
|
*/
|
|
private startWorker(name: ImmichWorker) {
|
|
console.log(`Starting ${name} worker`);
|
|
|
|
// eslint-disable-next-line unicorn/prefer-module
|
|
const basePath = dirname(__filename);
|
|
const workerFile = join(basePath, 'workers', `${name}.js`);
|
|
|
|
let anyWorker: Worker | ChildProcess;
|
|
let kill: (signal?: NodeJS.Signals) => Promise<void> | void;
|
|
|
|
if (name === ImmichWorker.Api) {
|
|
const worker = fork(workerFile, [], {
|
|
execArgv: process.execArgv.map((arg) => (arg.startsWith('--inspect') ? '--inspect=0.0.0.0:9231' : arg)),
|
|
});
|
|
|
|
kill = (signal) => void worker.kill(signal);
|
|
anyWorker = worker;
|
|
} else {
|
|
const worker = new Worker(workerFile);
|
|
|
|
kill = async () => void (await worker.terminate());
|
|
anyWorker = worker;
|
|
}
|
|
|
|
anyWorker.on('error', (error) => this.onError(name, error));
|
|
anyWorker.on('exit', (exitCode) => this.onExit(name, exitCode));
|
|
|
|
this.workers[name] = { kill };
|
|
}
|
|
|
|
onError(name: ImmichWorker, error: Error) {
|
|
console.error(`${name} worker error: ${error}, stack: ${error.stack}`);
|
|
}
|
|
|
|
onExit(name: ImmichWorker, exitCode: number | null) {
|
|
// restart immich server
|
|
if (exitCode === ExitCode.AppRestart || this.restarting) {
|
|
this.restarting = true;
|
|
|
|
console.info(`${name} worker shutdown for restart`);
|
|
delete this.workers[name];
|
|
|
|
// once all workers shut down, bootstrap again
|
|
if (Object.keys(this.workers).length === 0) {
|
|
void this.bootstrap();
|
|
this.restarting = false;
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
// shutdown the entire process
|
|
delete this.workers[name];
|
|
|
|
if (exitCode !== 0) {
|
|
console.error(`${name} worker exited with code ${exitCode}`);
|
|
|
|
if (this.workers[ImmichWorker.Api] && name !== ImmichWorker.Api) {
|
|
console.error('Killing api process');
|
|
void this.workers[ImmichWorker.Api].kill('SIGTERM');
|
|
}
|
|
}
|
|
|
|
process.exit(exitCode);
|
|
}
|
|
}
|
|
|
|
function main() {
|
|
const immichApp = process.argv[2];
|
|
if (immichApp) {
|
|
process.argv.splice(2, 1);
|
|
}
|
|
|
|
if (immichApp === 'immich-admin') {
|
|
process.title = 'immich_admin_cli';
|
|
process.env.IMMICH_LOG_LEVEL = LogLevel.Warn;
|
|
|
|
return CommandFactory.run(ImmichAdminModule);
|
|
}
|
|
|
|
if (immichApp === 'immich' || immichApp === 'microservices') {
|
|
console.error(
|
|
`Using "start.sh ${immichApp}" has been deprecated. See https://github.com/immich-app/immich/releases/tag/v1.118.0 for more information.`,
|
|
);
|
|
process.exit(1);
|
|
}
|
|
|
|
if (immichApp) {
|
|
console.error(`Unknown command: "${immichApp}"`);
|
|
process.exit(1);
|
|
}
|
|
|
|
process.title = 'immich';
|
|
void new Workers().bootstrap();
|
|
}
|
|
|
|
void main();
|