2024-01-10 05:04:16 +01:00
|
|
|
import { AssetCreate, IJobRepository, IMetadataRepository, LibraryResponseDto } 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';
|
2023-12-08 17:15:46 +01:00
|
|
|
import { AssetEntity, AssetType, LibraryType } from '@app/infra/entities';
|
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 { randomBytes } from 'crypto';
|
|
|
|
import { DateTime } from 'luxon';
|
2023-11-14 23:47:15 +01:00
|
|
|
import { EntityTarget, ObjectLiteral } from 'typeorm';
|
2024-01-10 05:04:16 +01:00
|
|
|
import { AppService } from '../../src/microservices/app.service';
|
|
|
|
import { newJobRepositoryMock, newMetadataRepositoryMock } from '../../test';
|
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();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
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-10 05:04:16 +01:00
|
|
|
create: async (): Promise<INestApplication> => {
|
2023-10-19 00:02:42 +02:00
|
|
|
const moduleFixture = await Test.createTestingModule({ imports: [AppModule], providers: [AppService] })
|
2024-01-09 23:07:01 +01:00
|
|
|
.overrideModule(InfraModule)
|
|
|
|
.useModule(InfraTestModule)
|
2023-10-19 00:02:42 +02:00
|
|
|
.overrideProvider(IJobRepository)
|
2024-01-10 05:04:16 +01:00
|
|
|
.useValue(newJobRepositoryMock())
|
|
|
|
.overrideProvider(IMetadataRepository)
|
|
|
|
.useValue(newMetadataRepositoryMock())
|
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-21 17:06:26 +01:00
|
|
|
await app.get(AppService).init();
|
2023-10-19 00:02:42 +02:00
|
|
|
|
2023-12-19 03:29:26 +01:00
|
|
|
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);
|
2023-10-19 00:02:42 +02:00
|
|
|
},
|
|
|
|
teardown: async () => {
|
2023-12-19 03:29:26 +01:00
|
|
|
if (app) {
|
|
|
|
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
|
|
|
|
2023-12-08 17:15:46 +01:00
|
|
|
function randomDate(start: Date, end: Date): Date {
|
|
|
|
return new Date(start.getTime() + Math.random() * (end.getTime() - start.getTime()));
|
|
|
|
}
|
|
|
|
|
|
|
|
let assetCount = 0;
|
|
|
|
export function generateAsset(
|
|
|
|
userId: string,
|
|
|
|
libraries: LibraryResponseDto[],
|
|
|
|
other: Partial<AssetEntity> = {},
|
|
|
|
): AssetCreate {
|
|
|
|
const id = assetCount++;
|
|
|
|
const { fileCreatedAt = randomDate(new Date(1970, 1, 1), new Date(2023, 1, 1)) } = other;
|
|
|
|
|
|
|
|
return {
|
|
|
|
createdAt: today.toJSDate(),
|
|
|
|
updatedAt: today.toJSDate(),
|
|
|
|
ownerId: userId,
|
|
|
|
checksum: randomBytes(20),
|
|
|
|
originalPath: `/tests/test_${id}`,
|
|
|
|
deviceAssetId: `test_${id}`,
|
|
|
|
deviceId: 'e2e-test',
|
|
|
|
libraryId: (
|
|
|
|
libraries.find(({ ownerId, type }) => ownerId === userId && type === LibraryType.UPLOAD) as LibraryResponseDto
|
|
|
|
).id,
|
|
|
|
isVisible: true,
|
|
|
|
fileCreatedAt,
|
|
|
|
fileModifiedAt: new Date(),
|
|
|
|
localDateTime: fileCreatedAt,
|
|
|
|
type: AssetType.IMAGE,
|
|
|
|
originalFileName: `test_${id}`,
|
|
|
|
...other,
|
|
|
|
};
|
|
|
|
}
|