2024-01-29 16:11:02 +01:00
|
|
|
import { IJobRepository, IMediaRepository, JobItem, JobItemHandler, QueueName } from '@app/domain';
|
2023-10-06 23:32:28 +02:00
|
|
|
import { AppModule } from '@app/immich';
|
2024-01-09 23:07:01 +01:00
|
|
|
import { InfraModule, InfraTestModule, dataSource } from '@app/infra';
|
2024-01-29 16:11:02 +01:00
|
|
|
import { MediaRepository } from '@app/infra/repositories';
|
2023-10-19 00:02:42 +02:00
|
|
|
import { INestApplication } from '@nestjs/common';
|
|
|
|
import { Test } from '@nestjs/testing';
|
2023-12-08 17:15:46 +01:00
|
|
|
import { DateTime } from 'luxon';
|
2024-01-27 18:27:55 +01:00
|
|
|
import * as fs from 'node:fs';
|
|
|
|
import path from 'node:path';
|
|
|
|
import { Server } from 'node:tls';
|
2024-01-31 09:15:54 +01:00
|
|
|
import { EventEmitter } from 'stream';
|
2023-11-14 23:47:15 +01:00
|
|
|
import { EntityTarget, ObjectLiteral } from 'typeorm';
|
2024-01-31 09:15:54 +01:00
|
|
|
import { AppService } from '../immich/app.service';
|
|
|
|
import { AppService as MicroAppService } from '../microservices/app.service';
|
2023-10-06 23:32:28 +02:00
|
|
|
|
2024-01-24 23:24:53 +01:00
|
|
|
export const IMMICH_TEST_ASSET_PATH = process.env.IMMICH_TEST_ASSET_PATH as string;
|
2023-10-06 23:32:28 +02:00
|
|
|
export const IMMICH_TEST_ASSET_TEMP_PATH = path.normalize(`${IMMICH_TEST_ASSET_PATH}/temp/`);
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2023-12-08 17:15:46 +01:00
|
|
|
export const today = DateTime.fromObject({ year: 2023, month: 11, day: 3 });
|
|
|
|
export const yesterday = today.minus({ days: 1 });
|
|
|
|
|
2023-11-14 23:47:15 +01:00
|
|
|
export interface ResetOptions {
|
|
|
|
entities?: EntityTarget<ObjectLiteral>[];
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
export const db = {
|
2023-11-14 23:47:15 +01:00
|
|
|
reset: async (options?: ResetOptions) => {
|
2023-12-21 17:06:26 +01:00
|
|
|
if (!dataSource.isInitialized) {
|
|
|
|
await dataSource.initialize();
|
|
|
|
}
|
2023-09-20 13:16:33 +02:00
|
|
|
await dataSource.transaction(async (em) => {
|
2023-11-14 23:47:15 +01:00
|
|
|
const entities = options?.entities || [];
|
|
|
|
const tableNames =
|
|
|
|
entities.length > 0
|
|
|
|
? entities.map((entity) => em.getRepository(entity).metadata.tableName)
|
2023-11-25 19:53:30 +01:00
|
|
|
: dataSource.entityMetadatas
|
|
|
|
.map((entity) => entity.tableName)
|
|
|
|
.filter((tableName) => !tableName.startsWith('geodata'));
|
2023-11-14 23:47:15 +01:00
|
|
|
|
|
|
|
let deleteUsers = false;
|
|
|
|
for (const tableName of tableNames) {
|
|
|
|
if (tableName === 'users') {
|
|
|
|
deleteUsers = true;
|
2023-09-20 13:16:33 +02:00
|
|
|
continue;
|
|
|
|
}
|
2023-11-14 23:47:15 +01:00
|
|
|
await em.query(`DELETE FROM ${tableName} CASCADE;`);
|
|
|
|
}
|
|
|
|
if (deleteUsers) {
|
|
|
|
await em.query(`DELETE FROM "users" CASCADE;`);
|
2023-09-20 13:16:33 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
},
|
|
|
|
disconnect: async () => {
|
|
|
|
if (dataSource.isInitialized) {
|
|
|
|
await dataSource.destroy();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2024-01-29 16:11:02 +01:00
|
|
|
class JobMock implements IJobRepository {
|
|
|
|
private _handler: JobItemHandler = () => Promise.resolve();
|
|
|
|
addHandler(_queueName: QueueName, _concurrency: number, handler: JobItemHandler) {
|
|
|
|
this._handler = handler;
|
|
|
|
}
|
|
|
|
addCronJob() {}
|
|
|
|
updateCronJob() {}
|
|
|
|
deleteCronJob() {}
|
|
|
|
validateCronExpression() {}
|
|
|
|
queue(item: JobItem) {
|
|
|
|
return this._handler(item);
|
|
|
|
}
|
|
|
|
queueAll(items: JobItem[]) {
|
|
|
|
return Promise.all(items.map(this._handler)).then(() => Promise.resolve());
|
|
|
|
}
|
|
|
|
async resume() {}
|
|
|
|
async empty() {}
|
|
|
|
async setConcurrency() {}
|
|
|
|
async getQueueStatus() {
|
|
|
|
return null as any;
|
|
|
|
}
|
|
|
|
async getJobCounts() {
|
|
|
|
return null as any;
|
|
|
|
}
|
|
|
|
async pause() {}
|
|
|
|
async clear() {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
async waitForQueueCompletion() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
class MediaMockRepository extends MediaRepository {
|
|
|
|
async generateThumbhash() {
|
|
|
|
return Buffer.from('mock-thumbhash');
|
|
|
|
}
|
|
|
|
}
|
2023-10-06 23:32:28 +02:00
|
|
|
|
2023-10-19 00:02:42 +02:00
|
|
|
let app: INestApplication;
|
2023-10-06 23:32:28 +02:00
|
|
|
|
2023-10-19 00:02:42 +02:00
|
|
|
export const testApp = {
|
2024-01-24 23:24:53 +01:00
|
|
|
create: async (): Promise<INestApplication> => {
|
2024-01-31 09:15:54 +01:00
|
|
|
const moduleFixture = await Test.createTestingModule({
|
|
|
|
imports: [AppModule],
|
|
|
|
providers: [AppService, MicroAppService],
|
|
|
|
})
|
2024-01-09 23:07:01 +01:00
|
|
|
.overrideModule(InfraModule)
|
|
|
|
.useModule(InfraTestModule)
|
2023-10-19 00:02:42 +02:00
|
|
|
.overrideProvider(IJobRepository)
|
2024-01-29 16:11:02 +01:00
|
|
|
.useClass(JobMock)
|
|
|
|
.overrideProvider(IMediaRepository)
|
|
|
|
.useClass(MediaMockRepository)
|
2023-10-19 00:02:42 +02:00
|
|
|
.compile();
|
2023-10-06 23:32:28 +02:00
|
|
|
|
2023-10-19 00:02:42 +02:00
|
|
|
app = await moduleFixture.createNestApplication().init();
|
2023-12-19 03:29:26 +01:00
|
|
|
await app.listen(0);
|
2024-01-31 09:15:54 +01:00
|
|
|
await db.reset();
|
2023-12-21 17:06:26 +01:00
|
|
|
await app.get(AppService).init();
|
2024-01-31 09:15:54 +01:00
|
|
|
await app.get(MicroAppService).init();
|
2023-10-19 00:02:42 +02:00
|
|
|
|
2023-12-19 03:29:26 +01:00
|
|
|
const port = app.getHttpServer().address().port;
|
|
|
|
const protocol = app instanceof Server ? 'https' : 'http';
|
|
|
|
process.env.IMMICH_INSTANCE_URL = protocol + '://127.0.0.1:' + port;
|
|
|
|
|
|
|
|
return app;
|
2023-10-19 00:02:42 +02:00
|
|
|
},
|
2023-11-15 02:08:22 +01:00
|
|
|
reset: async (options?: ResetOptions) => {
|
|
|
|
await db.reset(options);
|
2024-01-31 09:15:54 +01:00
|
|
|
await app.get(AppService).init();
|
|
|
|
|
|
|
|
await app.get(MicroAppService).init();
|
2023-10-19 00:02:42 +02:00
|
|
|
},
|
2024-01-31 09:15:54 +01:00
|
|
|
get: (member: any) => app.get(member),
|
2023-10-19 00:02:42 +02:00
|
|
|
teardown: async () => {
|
2023-12-19 03:29:26 +01:00
|
|
|
if (app) {
|
2024-01-31 09:15:54 +01:00
|
|
|
await app.get(MicroAppService).teardown();
|
2023-12-19 03:29:26 +01:00
|
|
|
await app.get(AppService).teardown();
|
|
|
|
await app.close();
|
|
|
|
}
|
2023-10-19 00:02:42 +02:00
|
|
|
await db.disconnect();
|
|
|
|
},
|
|
|
|
};
|
2023-09-20 13:16:33 +02:00
|
|
|
|
2024-01-31 09:15:54 +01:00
|
|
|
export function waitForEvent<T>(emitter: EventEmitter, event: string): Promise<T> {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const success = (val: T) => {
|
|
|
|
emitter.off('error', fail);
|
|
|
|
resolve(val);
|
|
|
|
};
|
|
|
|
const fail = (err: Error) => {
|
|
|
|
emitter.off(event, success);
|
|
|
|
reject(err);
|
|
|
|
};
|
|
|
|
emitter.once(event, success);
|
|
|
|
emitter.once('error', fail);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-10-06 23:32:28 +02:00
|
|
|
const directoryExists = async (dirPath: string) =>
|
|
|
|
await fs.promises
|
|
|
|
.access(dirPath)
|
|
|
|
.then(() => true)
|
|
|
|
.catch(() => false);
|
|
|
|
|
|
|
|
export async function restoreTempFolder(): Promise<void> {
|
|
|
|
if (await directoryExists(`${IMMICH_TEST_ASSET_TEMP_PATH}`)) {
|
|
|
|
// Temp directory exists, delete all files inside it
|
|
|
|
await fs.promises.rm(IMMICH_TEST_ASSET_TEMP_PATH, { recursive: true });
|
|
|
|
}
|
|
|
|
// Create temp folder
|
|
|
|
await fs.promises.mkdir(IMMICH_TEST_ASSET_TEMP_PATH);
|
|
|
|
}
|