mirror of
https://github.com/immich-app/immich.git
synced 2026-01-20 16:43:16 -08:00
* feat: ProcessRepository#createSpawnDuplexStream
* test: write tests for ProcessRepository#createSpawnDuplexStream
* feat: StorageRepository#createGzip,createGunzip,createPlainReadStream
* feat: backups util (args, create, restore, progress)
* feat: wait on maintenance operation lock on boot
* chore: use backup util from backup.service.ts
test: update backup.service.ts tests with new util
* feat: list/delete backups (maintenance services)
* chore: open api
fix: missing action in cli.service.ts
* chore: add missing repositories to MaintenanceModule
* refactor: move logSecret into module init
* feat: initialise StorageCore in maintenance mode
* feat: authenticate websocket requests in maintenance mode
* test: add mock for new storage fns
* feat: add MaintenanceEphemeralStateRepository
refactor: cache the secret in memory
* test: update service worker tests
* feat: add external maintenance mode status
* feat: synchronised status, restore db action
* test: backup restore service tests
* refactor: DRY end maintenance
* feat: list and delete backup routes
* feat: start action on boot
* fix: should set status on restore end
* refactor: add maintenanceStore to hold writables
* feat: sync status to web app
* feat: web impl.
* test: various utils for testings
* test: web e2e tests
* test: e2e maintenance spec
* test: update cli spec
* chore: e2e lint
* chore: lint fixes
* chore: lint fixes
* feat: start restore flow route
* test: update e2e tests
* chore: remove neon lights on maintenance action pages
* fix: use 'startRestoreFlow' on onboarding page
* chore: ignore any library folder in `docker/`
* fix: load status on boot
* feat: upload backups
* refactor: permit any .sql(.gz) to be listed/restored
* feat: download backups from list
* fix: permit uploading just .sql files
* feat: restore just .sql files
* fix: don't show backups list if logged out
* feat: system integrity check in restore flow
* test: not providing failed backups in API anymore
* test: util should also not try to use failedBackups
* fix: actually assign inputStream
* test: correct test backup prep.
* fix: ensure task is defined to show error
* test: fix docker cp command
* test: update e2e web spec to select next button
* test: update e2e api tests
* test: refactor timeouts
* chore: remove `showDelete` from maint. settings
* chore: lint
* chore: lint
* fix: make sure backups are correctly sorted for clean up
* test: update service spec
* test: adjust e2e timeout
* test: increase web timeouts for ci
* chore: move gitignore changes
* chore: additional filename validation
* refactor: better typings for integrity API
* feat: higher accuracy progress tracking
* chore: delay lock retry
* refactor: remove old maintenance settings
* refactor: clean up tailwind classes
* refactor: use while loop rather than recursive calls
* test: update service specs
* chore: check canParse too
* chore: lint
* fix: logic error causing infinite loop
* refactor: use <ProgressBar /> from ui library
* fix: create or overwrite file
* chore: i18n pass, update progress bar
* fix: wrong translation string
* chore: update colour variables
* test: update web test for new maint. page
* chore: format, fix key
* test: update tests to be more linter complaint & use new routines
* chore: update onClick -> onAction, title -> breadcrumbs
* fix: use wrench icon in admin settings sidebar
* chore: add translation strings to accordion
* chore: lint
* refactor: move maintenance worker init into service
* refactor: `maintenanceStatus` -> `getMaintenanceStatus`
refactor: `integrityCheck` -> `detectPriorInstall`
chore: add `v2.4.0` version
refactor: `/backups/list` -> `/backups`
refactor: use sendFile in download route
refactor: use separate backups permissions
chore: correct descriptions
refactor: permit handler that doesn't return promise for sendfile
* refactor: move status impl into service
refactor: add active flag to maintenance status
* refactor: split into database backup controller
* test: split api e2e tests and passing
* fix: move end button into authed default maint page
* fix: also show in restore flow
* fix: import getMaintenanceStatus
* test: split web e2e tests
* refactor: ensure detect install is consistently named
* chore: ensure admin for detect install while out of maint.
* refactor: remove state repository
* test: update maint. worker service spec
* test: split backup service spec
* refactor: rename db backup routes
* refactor: instead of param, allow bulk backup deletion
* test: update sdk use in e2e test
* test: correct deleteBackup call
* fix: correct type for serverinstall response dto
* chore: validate filename for deletion
* test: wip
* test: backups no longer take path param
* refactor: scope util to database-backups instead of backups
* fix: update worker controller with new route
* chore: use new admin page actions
* chore: remove stray comment
* test: rename outdated test
* refactor: getter pattern for maintenance secret
* refactor: `createSpawnDuplexStream` -> `spawnDuplexStream`
* refactor: prefer `Object.assign`
* refactor: remove useless try {} block
* refactor: prefer `type Props`
refactor: prefer arrow function
* refactor: use luxon API for minutesAgo
* chore: remove change to gitignore
* refactor: prefer `type Props`
* refactor: remove async from onMount
* refactor: use luxon toRelative for relative time
* refactor: duplicate logic check
* chore: open api
* refactor: begin moving code into web//services
* refactor: don't use template string with $t
* test: use dialog role to match prompt
* refactor: split actions into flow/restore
* test: fix action value
* refactor: move more service calls into web//services
* chore: should void fn return
* chore: bump 2.4.0 to 2.5.0 in controller
* chore: bump 2.4.0 to 2.5.0 in controller
* refactor: use events for web//services
* chore: open api
* chore: open api
* refactor: don't await returned promise
* refactor: remove redundant check
* refactor: add `type: command` to actions
* refactor: split backup entries into own component
* refactor: split restore flow into separate components
* refactor(web): split BackupDelete event
* chore: stylings
* chore: stylings
* fix: don't log query failure on first boot
* feat: support pg_dumpall backups
* feat: display information about each backup
* chore: i18n
* feat: rollback to restore point on migrations failure
* feat: health check after restore
* chore: format
* refactor: split health check into separate function
* refactor: split health into repository
test: write tests covering rollbacks
* fix: omit 'health' requirement from createDbBackup
* test(e2e): rollback test
* fix: wrap text in backup entry
* fix: don't shrink context menu button
* fix: correct CREATE DB syntax for postgres
* test: rename backups generated by test
* feat: add filesize to backup response dto
* feat: restore list
* feat: ui work
* fix: e2e test
* fix: e2e test
* pr feedback
* pr feedback
---------
Co-authored-by: Alex <alex.tran1502@gmail.com>
Co-authored-by: Jason Rasmussen <jason@rasm.me>
587 lines
23 KiB
TypeScript
587 lines
23 KiB
TypeScript
import { CallHandler, ExecutionContext, Provider, ValidationPipe } from '@nestjs/common';
|
|
import { APP_GUARD, APP_PIPE } from '@nestjs/core';
|
|
import { transformException } from '@nestjs/platform-express/multer/multer/multer.utils';
|
|
import { Test } from '@nestjs/testing';
|
|
import { ClassConstructor } from 'class-transformer';
|
|
import { NextFunction } from 'express';
|
|
import { Kysely } from 'kysely';
|
|
import multer from 'multer';
|
|
import { ChildProcessWithoutNullStreams } from 'node:child_process';
|
|
import { Duplex, Readable, Writable } from 'node:stream';
|
|
import { PNG } from 'pngjs';
|
|
import postgres from 'postgres';
|
|
import { UploadFieldName } from 'src/dtos/asset-media.dto';
|
|
import { AssetUploadInterceptor } from 'src/middleware/asset-upload.interceptor';
|
|
import { AuthGuard } from 'src/middleware/auth.guard';
|
|
import { FileUploadInterceptor } from 'src/middleware/file-upload.interceptor';
|
|
import { AccessRepository } from 'src/repositories/access.repository';
|
|
import { ActivityRepository } from 'src/repositories/activity.repository';
|
|
import { AlbumUserRepository } from 'src/repositories/album-user.repository';
|
|
import { AlbumRepository } from 'src/repositories/album.repository';
|
|
import { ApiKeyRepository } from 'src/repositories/api-key.repository';
|
|
import { AppRepository } from 'src/repositories/app.repository';
|
|
import { AssetEditRepository } from 'src/repositories/asset-edit.repository';
|
|
import { AssetJobRepository } from 'src/repositories/asset-job.repository';
|
|
import { AssetRepository } from 'src/repositories/asset.repository';
|
|
import { AuditRepository } from 'src/repositories/audit.repository';
|
|
import { ConfigRepository } from 'src/repositories/config.repository';
|
|
import { CronRepository } from 'src/repositories/cron.repository';
|
|
import { CryptoRepository } from 'src/repositories/crypto.repository';
|
|
import { DatabaseRepository } from 'src/repositories/database.repository';
|
|
import { DownloadRepository } from 'src/repositories/download.repository';
|
|
import { DuplicateRepository } from 'src/repositories/duplicate.repository';
|
|
import { EmailRepository } from 'src/repositories/email.repository';
|
|
import { EventRepository } from 'src/repositories/event.repository';
|
|
import { JobRepository } from 'src/repositories/job.repository';
|
|
import { LibraryRepository } from 'src/repositories/library.repository';
|
|
import { LoggingRepository } from 'src/repositories/logging.repository';
|
|
import { MachineLearningRepository } from 'src/repositories/machine-learning.repository';
|
|
import { MapRepository } from 'src/repositories/map.repository';
|
|
import { MediaRepository } from 'src/repositories/media.repository';
|
|
import { MemoryRepository } from 'src/repositories/memory.repository';
|
|
import { MetadataRepository } from 'src/repositories/metadata.repository';
|
|
import { MoveRepository } from 'src/repositories/move.repository';
|
|
import { NotificationRepository } from 'src/repositories/notification.repository';
|
|
import { OAuthRepository } from 'src/repositories/oauth.repository';
|
|
import { OcrRepository } from 'src/repositories/ocr.repository';
|
|
import { PartnerRepository } from 'src/repositories/partner.repository';
|
|
import { PersonRepository } from 'src/repositories/person.repository';
|
|
import { PluginRepository } from 'src/repositories/plugin.repository';
|
|
import { ProcessRepository } from 'src/repositories/process.repository';
|
|
import { SearchRepository } from 'src/repositories/search.repository';
|
|
import { ServerInfoRepository } from 'src/repositories/server-info.repository';
|
|
import { SessionRepository } from 'src/repositories/session.repository';
|
|
import { SharedLinkAssetRepository } from 'src/repositories/shared-link-asset.repository';
|
|
import { SharedLinkRepository } from 'src/repositories/shared-link.repository';
|
|
import { StackRepository } from 'src/repositories/stack.repository';
|
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
|
import { SyncCheckpointRepository } from 'src/repositories/sync-checkpoint.repository';
|
|
import { SyncRepository } from 'src/repositories/sync.repository';
|
|
import { SystemMetadataRepository } from 'src/repositories/system-metadata.repository';
|
|
import { TagRepository } from 'src/repositories/tag.repository';
|
|
import { TelemetryRepository } from 'src/repositories/telemetry.repository';
|
|
import { TrashRepository } from 'src/repositories/trash.repository';
|
|
import { UserRepository } from 'src/repositories/user.repository';
|
|
import { VersionHistoryRepository } from 'src/repositories/version-history.repository';
|
|
import { ViewRepository } from 'src/repositories/view-repository';
|
|
import { WebsocketRepository } from 'src/repositories/websocket.repository';
|
|
import { WorkflowRepository } from 'src/repositories/workflow.repository';
|
|
import { DB } from 'src/schema';
|
|
import { AuthService } from 'src/services/auth.service';
|
|
import { BaseService } from 'src/services/base.service';
|
|
import { RepositoryInterface } from 'src/types';
|
|
import { asPostgresConnectionConfig, getKyselyConfig } from 'src/utils/database';
|
|
import { IAccessRepositoryMock, newAccessRepositoryMock } from 'test/repositories/access.repository.mock';
|
|
import { newAssetRepositoryMock } from 'test/repositories/asset.repository.mock';
|
|
import { newConfigRepositoryMock } from 'test/repositories/config.repository.mock';
|
|
import { newCryptoRepositoryMock } from 'test/repositories/crypto.repository.mock';
|
|
import { newDatabaseRepositoryMock } from 'test/repositories/database.repository.mock';
|
|
import { newJobRepositoryMock } from 'test/repositories/job.repository.mock';
|
|
import { newMediaRepositoryMock } from 'test/repositories/media.repository.mock';
|
|
import { newMetadataRepositoryMock } from 'test/repositories/metadata.repository.mock';
|
|
import { newStorageRepositoryMock } from 'test/repositories/storage.repository.mock';
|
|
import { newSystemMetadataRepositoryMock } from 'test/repositories/system-metadata.repository.mock';
|
|
import { ITelemetryRepositoryMock, newTelemetryRepositoryMock } from 'test/repositories/telemetry.repository.mock';
|
|
import { assert, Mock, Mocked, vitest } from 'vitest';
|
|
|
|
export type ControllerContext = {
|
|
authenticate: Mock;
|
|
getHttpServer: () => any;
|
|
reset: () => void;
|
|
close: () => Promise<void>;
|
|
};
|
|
|
|
export const controllerSetup = async (controller: ClassConstructor<unknown>, providers: Provider[]) => {
|
|
const noopInterceptor = { intercept: (ctx: never, next: CallHandler<unknown>) => next.handle() };
|
|
const upload = multer({ storage: multer.memoryStorage() });
|
|
const memoryFileInterceptor = {
|
|
intercept: async (ctx: ExecutionContext, next: CallHandler<unknown>) => {
|
|
const context = ctx.switchToHttp();
|
|
const handler = upload.fields([
|
|
{ name: UploadFieldName.ASSET_DATA, maxCount: 1 },
|
|
{ name: UploadFieldName.SIDECAR_DATA, maxCount: 1 },
|
|
]);
|
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
const next: NextFunction = (error) => (error ? reject(transformException(error)) : resolve());
|
|
const maybePromise = handler(context.getRequest(), context.getResponse(), next);
|
|
Promise.resolve(maybePromise).catch((error) => reject(error));
|
|
});
|
|
|
|
return next.handle();
|
|
},
|
|
};
|
|
const moduleRef = await Test.createTestingModule({
|
|
controllers: [controller],
|
|
providers: [
|
|
{ provide: APP_PIPE, useValue: new ValidationPipe({ transform: true, whitelist: true }) },
|
|
{ provide: APP_GUARD, useClass: AuthGuard },
|
|
{ provide: LoggingRepository, useValue: LoggingRepository.create() },
|
|
{ provide: AuthService, useValue: { authenticate: vi.fn() } },
|
|
...providers,
|
|
],
|
|
})
|
|
.overrideInterceptor(FileUploadInterceptor)
|
|
.useValue(memoryFileInterceptor)
|
|
.overrideInterceptor(AssetUploadInterceptor)
|
|
.useValue(noopInterceptor)
|
|
.compile();
|
|
const app = moduleRef.createNestApplication();
|
|
await app.init();
|
|
|
|
// allow the AuthController to override the AuthService itself
|
|
const authenticate = app.get<Mocked<AuthService>>(AuthService).authenticate as Mock;
|
|
|
|
return {
|
|
authenticate,
|
|
getHttpServer: () => app.getHttpServer(),
|
|
reset: () => {
|
|
authenticate.mockReset();
|
|
},
|
|
close: async () => {
|
|
await app.close();
|
|
},
|
|
};
|
|
};
|
|
|
|
export type AutoMocked<T> = Mocked<T> & { resetAllMocks: () => void };
|
|
|
|
const mockFn = (label: string, { strict }: { strict: boolean }) => {
|
|
const message = `Called a mock function without a mock implementation (${label})`;
|
|
return vitest.fn(() => {
|
|
{
|
|
if (strict) {
|
|
assert.fail(message);
|
|
} else {
|
|
// console.warn(message);
|
|
}
|
|
}
|
|
});
|
|
};
|
|
|
|
export const mockBaseService = <T extends BaseService>(service: ClassConstructor<T>) => {
|
|
return automock(service, { args: [{ setContext: () => {} }], strict: false });
|
|
};
|
|
|
|
export const automock = <T>(
|
|
Dependency: ClassConstructor<T>,
|
|
options?: {
|
|
args?: ConstructorParameters<ClassConstructor<T>>;
|
|
strict?: boolean;
|
|
},
|
|
): AutoMocked<T> => {
|
|
const mock: Record<string, unknown> = {};
|
|
const strict = options?.strict ?? true;
|
|
const args = options?.args ?? [];
|
|
|
|
const mocks: Mock[] = [];
|
|
|
|
const instance = new Dependency(...args);
|
|
for (const property of Object.getOwnPropertyNames(Dependency.prototype)) {
|
|
if (property === 'constructor') {
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
const label = `${Dependency.name}.${property}`;
|
|
// console.log(`Automocking ${label}`);
|
|
|
|
const target = instance[property as keyof T];
|
|
if (typeof target === 'function') {
|
|
const mockImplementation = mockFn(label, { strict });
|
|
mock[property] = mockImplementation;
|
|
mocks.push(mockImplementation);
|
|
continue;
|
|
}
|
|
} catch {
|
|
// noop
|
|
}
|
|
}
|
|
|
|
const result = mock as AutoMocked<T>;
|
|
result.resetAllMocks = () => {
|
|
for (const mock of mocks) {
|
|
mock.mockReset();
|
|
}
|
|
};
|
|
|
|
return result;
|
|
};
|
|
|
|
export type ServiceOverrides = {
|
|
access: AccessRepository;
|
|
activity: ActivityRepository;
|
|
album: AlbumRepository;
|
|
albumUser: AlbumUserRepository;
|
|
apiKey: ApiKeyRepository;
|
|
app: AppRepository;
|
|
audit: AuditRepository;
|
|
asset: AssetRepository;
|
|
assetEdit: AssetEditRepository;
|
|
assetJob: AssetJobRepository;
|
|
config: ConfigRepository;
|
|
cron: CronRepository;
|
|
crypto: CryptoRepository;
|
|
database: DatabaseRepository;
|
|
downloadRepository: DownloadRepository;
|
|
duplicateRepository: DuplicateRepository;
|
|
email: EmailRepository;
|
|
event: EventRepository;
|
|
job: JobRepository;
|
|
library: LibraryRepository;
|
|
logger: LoggingRepository;
|
|
machineLearning: MachineLearningRepository;
|
|
map: MapRepository;
|
|
media: MediaRepository;
|
|
memory: MemoryRepository;
|
|
metadata: MetadataRepository;
|
|
move: MoveRepository;
|
|
notification: NotificationRepository;
|
|
ocr: OcrRepository;
|
|
oauth: OAuthRepository;
|
|
partner: PartnerRepository;
|
|
person: PersonRepository;
|
|
plugin: PluginRepository;
|
|
process: ProcessRepository;
|
|
search: SearchRepository;
|
|
serverInfo: ServerInfoRepository;
|
|
session: SessionRepository;
|
|
sharedLink: SharedLinkRepository;
|
|
sharedLinkAsset: SharedLinkAssetRepository;
|
|
stack: StackRepository;
|
|
storage: StorageRepository;
|
|
sync: SyncRepository;
|
|
syncCheckpoint: SyncCheckpointRepository;
|
|
systemMetadata: SystemMetadataRepository;
|
|
tag: TagRepository;
|
|
telemetry: TelemetryRepository;
|
|
trash: TrashRepository;
|
|
user: UserRepository;
|
|
versionHistory: VersionHistoryRepository;
|
|
view: ViewRepository;
|
|
websocket: WebsocketRepository;
|
|
workflow: WorkflowRepository;
|
|
};
|
|
|
|
type As<T> = T extends RepositoryInterface<infer U> ? U : never;
|
|
type IAccessRepository = { [K in keyof AccessRepository]: RepositoryInterface<AccessRepository[K]> };
|
|
|
|
export type ServiceMocks = {
|
|
[K in keyof Omit<ServiceOverrides, 'access' | 'telemetry'>]: Mocked<RepositoryInterface<ServiceOverrides[K]>>;
|
|
} & { access: IAccessRepositoryMock; telemetry: ITelemetryRepositoryMock };
|
|
|
|
type BaseServiceArgs = ConstructorParameters<typeof BaseService>;
|
|
type Constructor<Type, Args extends Array<any>> = {
|
|
new (...deps: Args): Type;
|
|
};
|
|
|
|
export const getMocks = () => {
|
|
const loggerMock = { setContext: () => {} };
|
|
const configMock = { getEnv: () => ({}) };
|
|
|
|
const mocks: ServiceMocks = {
|
|
access: newAccessRepositoryMock(),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
logger: automock(LoggingRepository, { args: [, configMock], strict: false }),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
cron: automock(CronRepository, { args: [, loggerMock] }),
|
|
crypto: newCryptoRepositoryMock(),
|
|
activity: automock(ActivityRepository),
|
|
audit: automock(AuditRepository),
|
|
album: automock(AlbumRepository, { strict: false }),
|
|
albumUser: automock(AlbumUserRepository),
|
|
asset: newAssetRepositoryMock(),
|
|
assetEdit: automock(AssetEditRepository),
|
|
assetJob: automock(AssetJobRepository),
|
|
app: automock(AppRepository, { strict: false }),
|
|
config: newConfigRepositoryMock(),
|
|
database: newDatabaseRepositoryMock(),
|
|
downloadRepository: automock(DownloadRepository, { strict: false }),
|
|
duplicateRepository: automock(DuplicateRepository),
|
|
email: automock(EmailRepository, { args: [loggerMock] }),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
event: automock(EventRepository, { args: [, , loggerMock], strict: false }),
|
|
job: newJobRepositoryMock(),
|
|
apiKey: automock(ApiKeyRepository),
|
|
library: automock(LibraryRepository, { strict: false }),
|
|
machineLearning: automock(MachineLearningRepository, { args: [loggerMock], strict: false }),
|
|
map: automock(MapRepository, { args: [undefined, undefined, { setContext: () => {} }] }),
|
|
media: newMediaRepositoryMock(),
|
|
memory: automock(MemoryRepository),
|
|
metadata: newMetadataRepositoryMock(),
|
|
move: automock(MoveRepository, { strict: false }),
|
|
notification: automock(NotificationRepository),
|
|
ocr: automock(OcrRepository, { strict: false }),
|
|
oauth: automock(OAuthRepository, { args: [loggerMock] }),
|
|
partner: automock(PartnerRepository, { strict: false }),
|
|
person: automock(PersonRepository, { strict: false }),
|
|
plugin: automock(PluginRepository, { strict: true }),
|
|
process: automock(ProcessRepository),
|
|
search: automock(SearchRepository, { strict: false }),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
serverInfo: automock(ServerInfoRepository, { args: [, loggerMock], strict: false }),
|
|
session: automock(SessionRepository),
|
|
sharedLink: automock(SharedLinkRepository),
|
|
sharedLinkAsset: automock(SharedLinkAssetRepository),
|
|
stack: automock(StackRepository),
|
|
storage: newStorageRepositoryMock(),
|
|
sync: automock(SyncRepository),
|
|
syncCheckpoint: automock(SyncCheckpointRepository),
|
|
systemMetadata: newSystemMetadataRepositoryMock(),
|
|
// systemMetadata: automock(SystemMetadataRepository, { strict: false }),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
tag: automock(TagRepository, { args: [, loggerMock], strict: false }),
|
|
telemetry: newTelemetryRepositoryMock(),
|
|
trash: automock(TrashRepository),
|
|
user: automock(UserRepository, { strict: false }),
|
|
versionHistory: automock(VersionHistoryRepository),
|
|
view: automock(ViewRepository),
|
|
// eslint-disable-next-line no-sparse-arrays
|
|
websocket: automock(WebsocketRepository, { args: [, loggerMock], strict: false }),
|
|
workflow: automock(WorkflowRepository, { strict: true }),
|
|
};
|
|
|
|
return mocks;
|
|
};
|
|
|
|
export const newTestService = <T extends BaseService>(
|
|
Service: Constructor<T, BaseServiceArgs>,
|
|
overrides: Partial<ServiceOverrides> = {},
|
|
) => {
|
|
const mocks = getMocks();
|
|
|
|
const sut = new Service(
|
|
overrides.logger || (mocks.logger as As<LoggingRepository>),
|
|
overrides.access || (mocks.access as IAccessRepository as AccessRepository),
|
|
overrides.activity || (mocks.activity as As<ActivityRepository>),
|
|
overrides.album || (mocks.album as As<AlbumRepository>),
|
|
overrides.albumUser || (mocks.albumUser as As<AlbumUserRepository>),
|
|
overrides.apiKey || (mocks.apiKey as As<ApiKeyRepository>),
|
|
overrides.app || (mocks.app as As<AppRepository>),
|
|
overrides.asset || (mocks.asset as As<AssetRepository>),
|
|
overrides.assetEdit || (mocks.assetEdit as As<AssetEditRepository>),
|
|
overrides.assetJob || (mocks.assetJob as As<AssetJobRepository>),
|
|
overrides.audit || (mocks.audit as As<AuditRepository>),
|
|
overrides.config || (mocks.config as As<ConfigRepository> as ConfigRepository),
|
|
overrides.cron || (mocks.cron as As<CronRepository>),
|
|
overrides.crypto || (mocks.crypto as As<CryptoRepository>),
|
|
overrides.database || (mocks.database as As<DatabaseRepository>),
|
|
overrides.downloadRepository || (mocks.downloadRepository as As<DownloadRepository>),
|
|
overrides.duplicateRepository || (mocks.duplicateRepository as As<DuplicateRepository>),
|
|
overrides.email || (mocks.email as As<EmailRepository>),
|
|
overrides.event || (mocks.event as As<EventRepository>),
|
|
overrides.job || (mocks.job as As<JobRepository>),
|
|
overrides.library || (mocks.library as As<LibraryRepository>),
|
|
overrides.machineLearning || (mocks.machineLearning as As<MachineLearningRepository>),
|
|
overrides.map || (mocks.map as As<MapRepository>),
|
|
overrides.media || (mocks.media as As<MediaRepository>),
|
|
overrides.memory || (mocks.memory as As<MemoryRepository>),
|
|
overrides.metadata || (mocks.metadata as As<MetadataRepository>),
|
|
overrides.move || (mocks.move as As<MoveRepository>),
|
|
overrides.notification || (mocks.notification as As<NotificationRepository>),
|
|
overrides.oauth || (mocks.oauth as As<OAuthRepository>),
|
|
overrides.ocr || (mocks.ocr as As<OcrRepository>),
|
|
overrides.partner || (mocks.partner as As<PartnerRepository>),
|
|
overrides.person || (mocks.person as As<PersonRepository>),
|
|
overrides.plugin || (mocks.plugin as As<PluginRepository>),
|
|
overrides.process || (mocks.process as As<ProcessRepository>),
|
|
overrides.search || (mocks.search as As<SearchRepository>),
|
|
overrides.serverInfo || (mocks.serverInfo as As<ServerInfoRepository>),
|
|
overrides.session || (mocks.session as As<SessionRepository>),
|
|
overrides.sharedLink || (mocks.sharedLink as As<SharedLinkRepository>),
|
|
overrides.sharedLinkAsset || (mocks.sharedLinkAsset as As<SharedLinkAssetRepository>),
|
|
overrides.stack || (mocks.stack as As<StackRepository>),
|
|
overrides.storage || (mocks.storage as As<StorageRepository>),
|
|
overrides.sync || (mocks.sync as As<SyncRepository>),
|
|
overrides.syncCheckpoint || (mocks.syncCheckpoint as As<SyncCheckpointRepository>),
|
|
overrides.systemMetadata || (mocks.systemMetadata as As<SystemMetadataRepository>),
|
|
overrides.tag || (mocks.tag as As<TagRepository>),
|
|
overrides.telemetry || (mocks.telemetry as unknown as TelemetryRepository),
|
|
overrides.trash || (mocks.trash as As<TrashRepository>),
|
|
overrides.user || (mocks.user as As<UserRepository>),
|
|
overrides.versionHistory || (mocks.versionHistory as As<VersionHistoryRepository>),
|
|
overrides.view || (mocks.view as As<ViewRepository>),
|
|
overrides.websocket || (mocks.websocket as As<WebsocketRepository>),
|
|
overrides.workflow || (mocks.workflow as As<WorkflowRepository>),
|
|
);
|
|
|
|
return {
|
|
sut,
|
|
mocks,
|
|
};
|
|
};
|
|
|
|
const createPNG = (r: number, g: number, b: number) => {
|
|
const image = new PNG({ width: 1, height: 1 });
|
|
image.data[0] = r;
|
|
image.data[1] = g;
|
|
image.data[2] = b;
|
|
image.data[3] = 255;
|
|
return PNG.sync.write(image);
|
|
};
|
|
|
|
function* newPngFactory() {
|
|
for (let r = 0; r < 255; r++) {
|
|
for (let g = 0; g < 255; g++) {
|
|
for (let b = 0; b < 255; b++) {
|
|
yield createPNG(r, g, b);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const pngFactory = newPngFactory();
|
|
|
|
const templateName = 'mich';
|
|
|
|
const withDatabase = (url: string, name: string) => url.replace(`/${templateName}`, `/${name}`);
|
|
|
|
export const getKyselyDB = async (suffix?: string): Promise<Kysely<DB>> => {
|
|
const testUrl = process.env.IMMICH_TEST_POSTGRES_URL!;
|
|
const sql = postgres({
|
|
...asPostgresConnectionConfig({
|
|
connectionType: 'url',
|
|
url: withDatabase(testUrl, 'postgres'),
|
|
}),
|
|
max: 1,
|
|
});
|
|
|
|
const randomSuffix = Math.random().toString(36).slice(2, 7);
|
|
const dbName = `immich_${suffix ?? randomSuffix}`;
|
|
await sql.unsafe(`CREATE DATABASE ${dbName} WITH TEMPLATE ${templateName} OWNER postgres;`);
|
|
|
|
return new Kysely<DB>(getKyselyConfig({ connectionType: 'url', url: withDatabase(testUrl, dbName) }));
|
|
};
|
|
|
|
export const newRandomImage = () => {
|
|
const { value } = pngFactory.next();
|
|
if (!value) {
|
|
throw new Error('Ran out of random asset data');
|
|
}
|
|
|
|
return value;
|
|
};
|
|
|
|
export const mockSpawn = vitest.fn((exitCode: number, stdout: string, stderr: string, error?: unknown) => {
|
|
return {
|
|
stdout: new Readable({
|
|
read() {
|
|
this.push(stdout); // write mock data to stdout
|
|
this.push(null); // end stream
|
|
},
|
|
}),
|
|
stderr: new Readable({
|
|
read() {
|
|
this.push(stderr); // write mock data to stderr
|
|
this.push(null); // end stream
|
|
},
|
|
}),
|
|
stdin: new Writable({
|
|
write(chunk, encoding, callback) {
|
|
callback();
|
|
},
|
|
}),
|
|
exitCode,
|
|
on: vitest.fn((event, callback: any) => {
|
|
if (event === 'close') {
|
|
callback(0);
|
|
}
|
|
if (event === 'error' && error) {
|
|
callback(error);
|
|
}
|
|
if (event === 'exit') {
|
|
callback(exitCode);
|
|
}
|
|
}),
|
|
} as unknown as ChildProcessWithoutNullStreams;
|
|
});
|
|
|
|
export const mockDuplex = vitest.fn(
|
|
(command: string, exitCode: number, stdout: string, stderr: string, error?: unknown) => {
|
|
const duplex = new Duplex({
|
|
write(_chunk, _encoding, callback) {
|
|
callback();
|
|
},
|
|
|
|
read() {},
|
|
|
|
final(callback) {
|
|
callback();
|
|
},
|
|
});
|
|
|
|
setImmediate(() => {
|
|
if (error) {
|
|
duplex.destroy(error as Error);
|
|
} else if (exitCode === 0) {
|
|
/* eslint-disable unicorn/prefer-single-call */
|
|
duplex.push(stdout);
|
|
duplex.push(null);
|
|
/* eslint-enable unicorn/prefer-single-call */
|
|
} else {
|
|
duplex.destroy(new Error(`${command} non-zero exit code (${exitCode})\n${stderr}`));
|
|
}
|
|
});
|
|
|
|
return duplex;
|
|
},
|
|
);
|
|
|
|
export const mockFork = vitest.fn((exitCode: number, stdout: string, stderr: string, error?: unknown) => {
|
|
const stdoutStream = new Readable({
|
|
read() {
|
|
this.push(stdout); // write mock data to stdout
|
|
this.push(null); // end stream
|
|
},
|
|
});
|
|
|
|
return {
|
|
stdout: stdoutStream,
|
|
stderr: new Readable({
|
|
read() {
|
|
this.push(stderr); // write mock data to stderr
|
|
this.push(null); // end stream
|
|
},
|
|
}),
|
|
stdin: new Writable({
|
|
write(chunk, encoding, callback) {
|
|
callback();
|
|
},
|
|
}),
|
|
exitCode,
|
|
on: vitest.fn((event, callback: any) => {
|
|
if (event === 'close') {
|
|
stdoutStream.once('end', () => callback(0));
|
|
}
|
|
if (event === 'error' && error) {
|
|
stdoutStream.once('end', () => callback(error));
|
|
}
|
|
if (event === 'exit') {
|
|
stdoutStream.once('end', () => callback(exitCode));
|
|
}
|
|
}),
|
|
kill: vitest.fn(),
|
|
} as unknown as ChildProcessWithoutNullStreams;
|
|
});
|
|
|
|
export async function* makeStream<T>(items: T[] = []): AsyncIterableIterator<T> {
|
|
for (const item of items) {
|
|
await Promise.resolve();
|
|
yield item;
|
|
}
|
|
}
|
|
|
|
export const wait = (ms: number): Promise<void> => {
|
|
return new Promise((resolve) => {
|
|
const target = performance.now() + ms;
|
|
const checkDone = () => {
|
|
if (performance.now() >= target) {
|
|
resolve();
|
|
} else {
|
|
setTimeout(checkDone, 1); // Check again after 1ms
|
|
}
|
|
};
|
|
setTimeout(checkDone, ms);
|
|
});
|
|
};
|