mirror of
https://github.com/immich-app/immich.git
synced 2025-01-09 05:16:47 +01:00
4e9b96ff1a
* Allow building and installing cli * feat: add format fix * docs: remove cli folder * feat: use immich scoped package * feat: rewrite cli readme * docs: add info on running without building * cleanup * chore: remove import functionality from cli * feat: add logout to cli * docs: add todo for file format from server * docs: add compilation step to cli * fix: success message spacing * feat: can create albums * fix: add check step to cli * fix: typos * feat: pull file formats from server * chore: use crawl service from server * chore: fix lint * docs: add cli documentation * chore: rename ignore pattern * chore: add version number to cli * feat: use sdk * fix: cleanup * feat: album name on windows * chore: remove skipped asset field * feat: add more info to server-info command * chore: cleanup * wip * chore: remove unneeded packages * e2e test can start * git ignore for geocode in cli * add cli e2e to github actions * can do e2e tests in the cli * simplify e2e test * cleanup * set matrix strategy in workflow * run npm ci in server * choose different working directory * check out submodules too * increase test timeout * set node version * cli docker e2e tests * fix cli docker file * run cli e2e in correct folder * set docker context * correct docker build * remove cli from dockerignore * chore: fix docs links * feat: add cli v2 milestone * fix: set correct cli date * remove submodule * chore: add npmignore * chore(cli): push to npm * fix: server e2e * run npm ci in server * remove state from e2e * run npm ci in server * reshuffle docker compose files * use new e2e composes in makefile * increase test timeout to 10 minutes * make github actions run makefile e2e tests * cleanup github test names * assert on server version * chore: split cli e2e tests into one file per command * chore: set cli release working dir * chore: add repo url to npmjs * chore: bump node setup to v4 * chore: normalize the github url * check e2e code in lint * fix lint * test key login flow * feat: allow configurable config dir * fix session service tests * create missing dir * cleanup * bump cli version to 2.0.4 * remove form-data * feat: allow single files as argument * add version option * bump dependencies * fix lint * wip use axios as upload * version bump * cApiTALiZaTiON * don't touch package lock * wip: don't use job queues * don't use make for cli e2e * fix server e2e * chore: remove old gha step * add npm ci to server --------- Co-authored-by: Alex <alex.tran1502@gmail.com> Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
162 lines
5 KiB
TypeScript
162 lines
5 KiB
TypeScript
import { AssetCreate, IJobRepository, JobItem, JobItemHandler, LibraryResponseDto, QueueName } from '@app/domain';
|
|
import { AppModule } from '@app/immich';
|
|
import { dataSource, databaseChecks } from '@app/infra';
|
|
import { AssetEntity, AssetType, LibraryType } from '@app/infra/entities';
|
|
import { INestApplication } from '@nestjs/common';
|
|
import { Test } from '@nestjs/testing';
|
|
|
|
import { randomBytes } from 'crypto';
|
|
import * as fs from 'fs';
|
|
import { DateTime } from 'luxon';
|
|
import path from 'path';
|
|
import { Server } from 'tls';
|
|
import { EntityTarget, ObjectLiteral } from 'typeorm';
|
|
import { AppService } from '../src/microservices/app.service';
|
|
|
|
export const IMMICH_TEST_ASSET_PATH = process.env.IMMICH_TEST_ASSET_PATH;
|
|
export const IMMICH_TEST_ASSET_TEMP_PATH = path.normalize(`${IMMICH_TEST_ASSET_PATH}/temp/`);
|
|
|
|
export const today = DateTime.fromObject({ year: 2023, month: 11, day: 3 });
|
|
export const yesterday = today.minus({ days: 1 });
|
|
|
|
export interface ResetOptions {
|
|
entities?: EntityTarget<ObjectLiteral>[];
|
|
}
|
|
export const db = {
|
|
reset: async (options?: ResetOptions) => {
|
|
await databaseChecks();
|
|
await dataSource.transaction(async (em) => {
|
|
const entities = options?.entities || [];
|
|
const tableNames =
|
|
entities.length > 0
|
|
? entities.map((entity) => em.getRepository(entity).metadata.tableName)
|
|
: dataSource.entityMetadatas
|
|
.map((entity) => entity.tableName)
|
|
.filter((tableName) => !tableName.startsWith('geodata'));
|
|
|
|
let deleteUsers = false;
|
|
for (const tableName of tableNames) {
|
|
if (tableName === 'users') {
|
|
deleteUsers = true;
|
|
continue;
|
|
}
|
|
await em.query(`DELETE FROM ${tableName} CASCADE;`);
|
|
}
|
|
if (deleteUsers) {
|
|
await em.query(`DELETE FROM "users" CASCADE;`);
|
|
}
|
|
});
|
|
},
|
|
disconnect: async () => {
|
|
if (dataSource.isInitialized) {
|
|
await dataSource.destroy();
|
|
}
|
|
},
|
|
};
|
|
|
|
let _handler: JobItemHandler = () => Promise.resolve();
|
|
|
|
interface TestAppOptions {
|
|
jobs: boolean;
|
|
}
|
|
|
|
let app: INestApplication;
|
|
|
|
export const testApp = {
|
|
create: async (options?: TestAppOptions): Promise<INestApplication> => {
|
|
const { jobs } = options || { jobs: false };
|
|
|
|
const moduleFixture = await Test.createTestingModule({ imports: [AppModule], providers: [AppService] })
|
|
.overrideProvider(IJobRepository)
|
|
.useValue({
|
|
addHandler: (_queueName: QueueName, _concurrency: number, handler: JobItemHandler) => (_handler = handler),
|
|
addCronJob: jest.fn(),
|
|
updateCronJob: jest.fn(),
|
|
deleteCronJob: jest.fn(),
|
|
validateCronExpression: jest.fn(),
|
|
queue: (item: JobItem) => jobs && _handler(item),
|
|
resume: jest.fn(),
|
|
empty: jest.fn(),
|
|
setConcurrency: jest.fn(),
|
|
getQueueStatus: jest.fn(),
|
|
getJobCounts: jest.fn(),
|
|
pause: jest.fn(),
|
|
clear: jest.fn(),
|
|
} as IJobRepository)
|
|
.compile();
|
|
|
|
app = await moduleFixture.createNestApplication().init();
|
|
await app.listen(0);
|
|
|
|
if (jobs) {
|
|
await app.get(AppService).init();
|
|
}
|
|
|
|
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;
|
|
},
|
|
reset: async (options?: ResetOptions) => {
|
|
await db.reset(options);
|
|
},
|
|
teardown: async () => {
|
|
if (app) {
|
|
await app.get(AppService).teardown();
|
|
await app.close();
|
|
}
|
|
await db.disconnect();
|
|
},
|
|
};
|
|
|
|
export const runAllTests: boolean = process.env.IMMICH_RUN_ALL_TESTS === 'true';
|
|
|
|
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);
|
|
}
|
|
|
|
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,
|
|
};
|
|
}
|