import { AllJobStatusResponseDto, AssetMediaCreateDto, AssetMediaResponseDto, AssetResponseDto, CheckExistingAssetsDto, CreateAlbumDto, CreateLibraryDto, MetadataSearchDto, PersonCreateDto, SharedLinkCreateDto, UserAdminCreateDto, ValidateLibraryDto, checkExistingAssets, createAlbum, createApiKey, createLibrary, createPartner, createPerson, createSharedLink, createUserAdmin, deleteAssets, getAllJobsStatus, getAssetInfo, getConfigDefaults, login, searchMetadata, setBaseUrl, signUpAdmin, updateAdminOnboarding, updateAlbumUser, updateConfig, validate, } from '@immich/sdk'; import { BrowserContext } from '@playwright/test'; import { exec, spawn } from 'node:child_process'; import { createHash } from 'node:crypto'; import { existsSync, mkdirSync, rmSync, writeFileSync } from 'node:fs'; import { tmpdir } from 'node:os'; import path, { dirname } from 'node:path'; import { setTimeout as setAsyncTimeout } from 'node:timers/promises'; import { promisify } from 'node:util'; import pg from 'pg'; import { io, type Socket } from 'socket.io-client'; import { loginDto, signupDto } from 'src/fixtures'; import { makeRandomImage } from 'src/generators'; import request from 'supertest'; type CommandResponse = { stdout: string; stderr: string; exitCode: number | null }; type EventType = 'assetUpload' | 'assetUpdate' | 'assetDelete' | 'userDelete' | 'assetHidden'; type WaitOptions = { event: EventType; id?: string; total?: number; timeout?: number }; type AdminSetupOptions = { onboarding?: boolean }; type FileData = { bytes?: Buffer; filename: string }; const dbUrl = 'postgres://postgres:postgres@127.0.0.1:5433/immich'; export const baseUrl = 'http://127.0.0.1:2283'; export const shareUrl = `${baseUrl}/share`; export const app = `${baseUrl}/api`; // TODO move test assets into e2e/assets export const testAssetDir = path.resolve('./test-assets'); export const testAssetDirInternal = '/test-assets'; export const tempDir = tmpdir(); export const asBearerAuth = (accessToken: string) => ({ Authorization: `Bearer ${accessToken}` }); export const asKeyAuth = (key: string) => ({ 'x-api-key': key }); export const immichCli = (args: string[]) => executeCommand('node', ['node_modules/.bin/immich', '-d', `/${tempDir}/immich/`, ...args]).promise; export const immichAdmin = (args: string[]) => executeCommand('docker', ['exec', '-i', 'immich-e2e-server', '/bin/bash', '-c', `immich-admin ${args.join(' ')}`]); const executeCommand = (command: string, args: string[]) => { let _resolve: (value: CommandResponse) => void; const promise = new Promise((resolve) => (_resolve = resolve)); const child = spawn(command, args, { stdio: 'pipe' }); let stdout = ''; let stderr = ''; child.stdout.on('data', (data) => (stdout += data.toString())); child.stderr.on('data', (data) => (stderr += data.toString())); child.on('exit', (exitCode) => { _resolve({ stdout: stdout.trim(), stderr: stderr.trim(), exitCode, }); }); return { promise, child }; }; let client: pg.Client | null = null; const events: Record> = { assetHidden: new Set(), assetUpload: new Set(), assetUpdate: new Set(), assetDelete: new Set(), userDelete: new Set(), }; const idCallbacks: Record void> = {}; const countCallbacks: Record void }> = {}; const execPromise = promisify(exec); const onEvent = ({ event, id }: { event: EventType; id: string }) => { // console.log(`Received event: ${event} [id=${id}]`); const set = events[event]; set.add(id); const idCallback = idCallbacks[id]; if (idCallback) { idCallback(); delete idCallbacks[id]; } const item = countCallbacks[event]; if (item) { const { count, callback: countCallback } = item; if (set.size >= count) { countCallback(); delete countCallbacks[event]; } } }; export const utils = { resetDatabase: async (tables?: string[]) => { try { if (!client) { client = new pg.Client(dbUrl); await client.connect(); } tables = tables || [ // TODO e2e test for deleting a stack, since it is quite complex 'asset_stack', 'libraries', 'shared_links', 'person', 'albums', 'assets', 'asset_faces', 'activity', 'api_keys', 'sessions', 'users', 'system_metadata', ]; const sql: string[] = []; for (const table of tables) { if (table === 'system_metadata') { // prevent reverse geocoder from being re-initialized sql.push(`DELETE FROM "system_metadata" where "key" != 'reverse-geocoding-state';`); } else { sql.push(`DELETE FROM ${table} CASCADE;`); } } await client.query(sql.join('\n')); } catch (error) { console.error('Failed to reset database', error); throw error; } }, resetFilesystem: async () => { const mediaInternal = '/usr/src/app/upload'; const dirs = [ `"${mediaInternal}/thumbs"`, `"${mediaInternal}/upload"`, `"${mediaInternal}/library"`, `"${mediaInternal}/encoded-video"`, ].join(' '); await execPromise(`docker exec -i "immich-e2e-server" /bin/bash -c "rm -rf ${dirs} && mkdir ${dirs}"`); }, unzip: async (input: string, output: string) => { await execPromise(`unzip -o -d "${output}" "${input}"`); }, sha1: (bytes: Buffer) => createHash('sha1').update(bytes).digest('base64'), connectWebsocket: async (accessToken: string) => { const websocket = io(baseUrl, { path: '/api/socket.io', transports: ['websocket'], extraHeaders: { Authorization: `Bearer ${accessToken}` }, autoConnect: true, forceNew: true, }); return new Promise((resolve) => { websocket .on('connect', () => resolve(websocket)) .on('on_upload_success', (data: AssetResponseDto) => onEvent({ event: 'assetUpload', id: data.id })) .on('on_asset_update', (data: AssetResponseDto) => onEvent({ event: 'assetUpdate', id: data.id })) .on('on_asset_hidden', (assetId: string) => onEvent({ event: 'assetHidden', id: assetId })) .on('on_asset_delete', (assetId: string) => onEvent({ event: 'assetDelete', id: assetId })) .on('on_user_delete', (userId: string) => onEvent({ event: 'userDelete', id: userId })) .connect(); }); }, disconnectWebsocket: (ws: Socket) => { if (ws?.connected) { ws.disconnect(); } for (const set of Object.values(events)) { set.clear(); } }, resetEvents: () => { for (const set of Object.values(events)) { set.clear(); } }, waitForWebsocketEvent: ({ event, id, total: count, timeout: ms }: WaitOptions): Promise => { return new Promise((resolve, reject) => { if (!id && !count) { reject(new Error('id or count must be provided for waitForWebsocketEvent')); } const timeout = setTimeout(() => reject(new Error(`Timed out waiting for ${event} event`)), ms || 10_000); const type = id ? `id=${id}` : `count=${count}`; console.log(`Waiting for ${event} [${type}]`); const set = events[event]; const onId = () => { clearTimeout(timeout); resolve(); }; if ((id && set.has(id)) || (count && set.size >= count)) { onId(); return; } if (id) { idCallbacks[id] = onId; } if (count) { countCallbacks[event] = { count, callback: onId, }; } }); }, initSdk: () => { setBaseUrl(app); }, adminSetup: async (options?: AdminSetupOptions) => { options = options || { onboarding: true }; await signUpAdmin({ signUpDto: signupDto.admin }); const response = await login({ loginCredentialDto: loginDto.admin }); if (options.onboarding) { await updateAdminOnboarding( { adminOnboardingUpdateDto: { isOnboarded: true } }, { headers: asBearerAuth(response.accessToken) }, ); } return response; }, userSetup: async (accessToken: string, dto: UserAdminCreateDto) => { await createUserAdmin({ userAdminCreateDto: dto }, { headers: asBearerAuth(accessToken) }); return login({ loginCredentialDto: { email: dto.email, password: dto.password }, }); }, createApiKey: (accessToken: string) => { return createApiKey({ apiKeyCreateDto: { name: 'e2e' } }, { headers: asBearerAuth(accessToken) }); }, createAlbum: (accessToken: string, dto: CreateAlbumDto) => createAlbum({ createAlbumDto: dto }, { headers: asBearerAuth(accessToken) }), updateAlbumUser: (accessToken: string, args: Parameters[0]) => updateAlbumUser(args, { headers: asBearerAuth(accessToken) }), createAsset: async ( accessToken: string, dto?: Partial> & { assetData?: FileData; sidecarData?: FileData; }, ) => { const _dto = { deviceAssetId: 'test-1', deviceId: 'test', fileCreatedAt: new Date().toISOString(), fileModifiedAt: new Date().toISOString(), ...dto, }; const assetData = dto?.assetData?.bytes || makeRandomImage(); const filename = dto?.assetData?.filename || 'example.png'; if (dto?.assetData?.bytes) { console.log(`Uploading ${filename}`); } const builder = request(app) .post(`/assets`) .attach('assetData', assetData, filename) .set('Authorization', `Bearer ${accessToken}`); if (dto?.sidecarData?.bytes) { void builder.attach('sidecarData', dto.sidecarData.bytes, dto.sidecarData.filename); } for (const [key, value] of Object.entries(_dto)) { void builder.field(key, String(value)); } const { body } = await builder; return body as AssetMediaResponseDto; }, replaceAsset: async ( accessToken: string, assetId: string, dto?: Partial> & { assetData?: FileData }, ) => { const _dto = { deviceAssetId: 'test-1', deviceId: 'test', fileCreatedAt: new Date().toISOString(), fileModifiedAt: new Date().toISOString(), ...dto, }; const assetData = dto?.assetData?.bytes || makeRandomImage(); const filename = dto?.assetData?.filename || 'example.png'; if (dto?.assetData?.bytes) { console.log(`Uploading ${filename}`); } const builder = request(app) .put(`/assets/${assetId}/original`) .attach('assetData', assetData, filename) .set('Authorization', `Bearer ${accessToken}`); for (const [key, value] of Object.entries(_dto)) { void builder.field(key, String(value)); } const { body } = await builder; return body as AssetMediaResponseDto; }, createImageFile: (path: string) => { if (!existsSync(dirname(path))) { mkdirSync(dirname(path), { recursive: true }); } writeFileSync(path, makeRandomImage()); }, removeImageFile: (path: string) => { if (!existsSync(path)) { return; } rmSync(path); }, getAssetInfo: (accessToken: string, id: string) => getAssetInfo({ id }, { headers: asBearerAuth(accessToken) }), checkExistingAssets: (accessToken: string, checkExistingAssetsDto: CheckExistingAssetsDto) => checkExistingAssets({ checkExistingAssetsDto }, { headers: asBearerAuth(accessToken) }), metadataSearch: async (accessToken: string, dto: MetadataSearchDto) => { return searchMetadata({ metadataSearchDto: dto }, { headers: asBearerAuth(accessToken) }); }, deleteAssets: (accessToken: string, ids: string[]) => deleteAssets({ assetBulkDeleteDto: { ids } }, { headers: asBearerAuth(accessToken) }), createPerson: async (accessToken: string, dto?: PersonCreateDto) => { const person = await createPerson({ personCreateDto: dto || {} }, { headers: asBearerAuth(accessToken) }); await utils.setPersonThumbnail(person.id); return person; }, createFace: async ({ assetId, personId }: { assetId: string; personId: string }) => { if (!client) { return; } await client.query('INSERT INTO asset_faces ("assetId", "personId") VALUES ($1, $2)', [assetId, personId]); }, setPersonThumbnail: async (personId: string) => { if (!client) { return; } await client.query(`UPDATE "person" set "thumbnailPath" = '/my/awesome/thumbnail.jpg' where "id" = $1`, [personId]); }, createSharedLink: (accessToken: string, dto: SharedLinkCreateDto) => createSharedLink({ sharedLinkCreateDto: dto }, { headers: asBearerAuth(accessToken) }), createLibrary: (accessToken: string, dto: CreateLibraryDto) => createLibrary({ createLibraryDto: dto }, { headers: asBearerAuth(accessToken) }), validateLibrary: (accessToken: string, id: string, dto: ValidateLibraryDto) => validate({ id, validateLibraryDto: dto }, { headers: asBearerAuth(accessToken) }), createPartner: (accessToken: string, id: string) => createPartner({ id }, { headers: asBearerAuth(accessToken) }), setAuthCookies: async (context: BrowserContext, accessToken: string, domain = '127.0.0.1') => await context.addCookies([ { name: 'immich_access_token', value: accessToken, domain, path: '/', expires: 1_742_402_728, httpOnly: true, secure: false, sameSite: 'Lax', }, { name: 'immich_auth_type', value: 'password', domain, path: '/', expires: 1_742_402_728, httpOnly: true, secure: false, sameSite: 'Lax', }, { name: 'immich_is_authenticated', value: 'true', domain, path: '/', expires: 1_742_402_728, httpOnly: false, secure: false, sameSite: 'Lax', }, ]), resetTempFolder: () => { rmSync(`${testAssetDir}/temp`, { recursive: true, force: true }); mkdirSync(`${testAssetDir}/temp`, { recursive: true }); }, resetAdminConfig: async (accessToken: string) => { const defaultConfig = await getConfigDefaults({ headers: asBearerAuth(accessToken) }); await updateConfig({ systemConfigDto: defaultConfig }, { headers: asBearerAuth(accessToken) }); }, isQueueEmpty: async (accessToken: string, queue: keyof AllJobStatusResponseDto) => { const queues = await getAllJobsStatus({ headers: asBearerAuth(accessToken) }); const jobCounts = queues[queue].jobCounts; return !jobCounts.active && !jobCounts.waiting; }, waitForQueueFinish: (accessToken: string, queue: keyof AllJobStatusResponseDto, ms?: number) => { return new Promise(async (resolve, reject) => { const timeout = setTimeout(() => reject(new Error('Timed out waiting for queue to empty')), ms || 10_000); while (true) { const done = await utils.isQueueEmpty(accessToken, queue); if (done) { break; } await setAsyncTimeout(200); } clearTimeout(timeout); resolve(); }); }, cliLogin: async (accessToken: string) => { const key = await utils.createApiKey(accessToken); await immichCli(['login', app, `${key.secret}`]); return key.secret; }, }; utils.initSdk(); if (!existsSync(`${testAssetDir}/albums`)) { throw new Error( `Test assets not found. Please checkout https://github.com/immich-app/test-assets into ${testAssetDir} before testing`, ); }