import { AllJobStatusResponseDto, AssetFileUploadResponseDto, AssetResponseDto, CreateAlbumDto, CreateAssetDto, CreateLibraryDto, CreateUserDto, MetadataSearchDto, PersonCreateDto, SharedLinkCreateDto, ValidateLibraryDto, createAlbum, createApiKey, createLibrary, createPerson, createSharedLink, createUser, defaults, deleteAssets, getAllAssets, getAllJobsStatus, getAssetInfo, login, searchMetadata, setAdminOnboarding, signUpAdmin, 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 CliResponse = { stdout: string; stderr: string; exitCode: number | null }; type EventType = 'assetUpload' | 'assetUpdate' | 'assetDelete' | 'userDelete'; type WaitOptions = { event: EventType; id?: string; total?: number; timeout?: number }; type AdminSetupOptions = { onboarding?: boolean }; type AssetData = { bytes?: Buffer; filename: string }; const dbUrl = 'postgres://postgres:postgres@127.0.0.1:5433/immich'; const baseUrl = 'http://127.0.0.1:2283'; export const app = `${baseUrl}/api`; // TODO move test assets into e2e/assets export const testAssetDir = path.resolve(`./../server/test/assets/`); export const testAssetDirInternal = '/data/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 = async (args: string[]) => { let _resolve: (value: CliResponse) => void; const deferred = new Promise((resolve) => (_resolve = resolve)); const _args = ['node_modules/.bin/immich', '-d', `/${tempDir}/immich/`, ...args]; const child = spawn('node', _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 deferred; }; let client: pg.Client | null = null; const events: Record> = { 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', 'user_token', 'users', 'system_metadata', ]; const sql: string[] = []; if (tables.includes('asset_stack')) { sql.push('UPDATE "assets" SET "stackId" = NULL;'); } for (const table of tables) { 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_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, }; } }); }, setApiEndpoint: () => { defaults.baseUrl = 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 setAdminOnboarding({ headers: asBearerAuth(response.accessToken) }); } return response; }, userSetup: async (accessToken: string, dto: CreateUserDto) => { await createUser({ createUserDto: 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) }), createAsset: async ( accessToken: string, dto?: Partial> & { assetData?: AssetData }, ) => { 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(`/asset/upload`) .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 AssetFileUploadResponseDto; }, createImageFile: (path: string) => { if (!existsSync(dirname(path))) { mkdirSync(dirname(path), { recursive: true }); } if (!existsSync(path)) { writeFileSync(path, makeRandomImage()); } }, removeImageFile: (path: string) => { if (!existsSync(path)) { return; } rmSync(path); }, getAssetInfo: (accessToken: string, id: string) => getAssetInfo({ id }, { headers: asBearerAuth(accessToken) }), getAllAssets: (accessToken: string) => getAllAssets({}, { 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; } const vector = Array.from({ length: 512 }, Math.random); const embedding = `[${vector.join(',')}]`; await client.query('INSERT INTO asset_faces ("assetId", "personId", "embedding") VALUES ($1, $2, $3)', [ assetId, personId, embedding, ]); }, 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) }), setAuthCookies: async (context: BrowserContext, accessToken: string) => await context.addCookies([ { name: 'immich_access_token', value: accessToken, domain: '127.0.0.1', path: '/', expires: 1_742_402_728, httpOnly: true, secure: false, sameSite: 'Lax', }, { name: 'immich_auth_type', value: 'password', domain: '127.0.0.1', path: '/', expires: 1_742_402_728, httpOnly: true, secure: false, sameSite: 'Lax', }, { name: 'immich_is_authenticated', value: 'true', domain: '127.0.0.1', path: '/', expires: 1_742_402_728, httpOnly: false, secure: false, sameSite: 'Lax', }, ]), deleteTempFolder: () => { rmSync(`${testAssetDir}/temp`, { recursive: true, force: true }); }, 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.setApiEndpoint(); if (!existsSync(`${testAssetDir}/albums`)) { throw new Error( `Test assets not found. Please checkout https://github.com/immich-app/test-assets into ${testAssetDir} before testing`, ); }