mirror of
https://github.com/immich-app/immich.git
synced 2025-01-24 20:52:44 +01:00
efa6efd200
* remove external path * open-api * make sql * move library settings to admin panel * Add documentation * show external libraries only * fix library list * make user library settings look good * fix test * fix tests * fix tests * can pick user for library * fix tests * fix e2e * chore: make sql * Use unauth exception * delete user library list * cleanup * fix e2e * fix await lint * chore: remove unused code * chore: cleanup * revert docs * fix: is admin stuff * table alignment --------- Co-authored-by: Jason Rasmussen <jrasm91@gmail.com> Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
229 lines
7.6 KiB
TypeScript
229 lines
7.6 KiB
TypeScript
import { LibraryResponseDto, LibraryService, LoginResponseDto } from '@app/domain';
|
|
import { AssetType, LibraryType } from '@app/infra/entities';
|
|
import fs from 'node:fs/promises';
|
|
import path from 'node:path';
|
|
import {
|
|
IMMICH_TEST_ASSET_PATH,
|
|
IMMICH_TEST_ASSET_TEMP_PATH,
|
|
restoreTempFolder,
|
|
testApp,
|
|
waitForEvent,
|
|
} from '../../../src/test-utils/utils';
|
|
import { api } from '../../client';
|
|
|
|
describe(`Library watcher (e2e)`, () => {
|
|
let server: any;
|
|
let admin: LoginResponseDto;
|
|
let libraryService: LibraryService;
|
|
const configFilePath = process.env.IMMICH_CONFIG_FILE;
|
|
|
|
beforeAll(async () => {
|
|
process.env.IMMICH_CONFIG_FILE = path.normalize(`${__dirname}/../config/library-watcher-e2e-config.json`);
|
|
|
|
const app = await testApp.create();
|
|
server = app.getHttpServer();
|
|
libraryService = testApp.get(LibraryService);
|
|
});
|
|
|
|
beforeEach(async () => {
|
|
await testApp.reset();
|
|
await restoreTempFolder();
|
|
await api.authApi.adminSignUp(server);
|
|
admin = await api.authApi.adminLogin(server);
|
|
});
|
|
|
|
afterEach(async () => {
|
|
await libraryService.unwatchAll();
|
|
});
|
|
|
|
afterAll(async () => {
|
|
await testApp.teardown();
|
|
await restoreTempFolder();
|
|
process.env.IMMICH_CONFIG_FILE = configFilePath;
|
|
});
|
|
|
|
describe('Event handling', () => {
|
|
describe('Single import path', () => {
|
|
beforeEach(async () => {
|
|
await api.libraryApi.create(server, admin.accessToken, {
|
|
type: LibraryType.EXTERNAL,
|
|
importPaths: [`${IMMICH_TEST_ASSET_TEMP_PATH}`],
|
|
});
|
|
});
|
|
|
|
it('should import a new file', async () => {
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const afterAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(afterAssets.length).toEqual(1);
|
|
});
|
|
|
|
it('should import new files with case insensitive extensions', async () => {
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file2.JPG`,
|
|
);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file3.Jpg`,
|
|
);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file4.jpG`,
|
|
);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file5.jPg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
await waitForEvent(libraryService, 'add');
|
|
await waitForEvent(libraryService, 'add');
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const afterAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(afterAssets.length).toEqual(4);
|
|
});
|
|
|
|
it('should update a changed file', async () => {
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const originalAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(originalAssets.length).toEqual(1);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/prairie_falcon.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/file.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'change');
|
|
|
|
const afterAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(afterAssets).toEqual([
|
|
expect.objectContaining({
|
|
// Make sure we keep the original asset id
|
|
id: originalAssets[0].id,
|
|
type: AssetType.IMAGE,
|
|
exifInfo: expect.objectContaining({
|
|
make: 'Canon',
|
|
model: 'Canon EOS R5',
|
|
exifImageWidth: 800,
|
|
exifImageHeight: 533,
|
|
exposureTime: '1/4000',
|
|
}),
|
|
}),
|
|
]);
|
|
});
|
|
});
|
|
|
|
describe('Multiple import paths', () => {
|
|
beforeEach(async () => {
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1`, { recursive: true });
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir2`, { recursive: true });
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir3`, { recursive: true });
|
|
|
|
await api.libraryApi.create(server, admin.accessToken, {
|
|
type: LibraryType.EXTERNAL,
|
|
importPaths: [
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir2`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir3`,
|
|
],
|
|
});
|
|
});
|
|
|
|
it('should add new files in multiple import paths', async () => {
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/el_torcal_rocks.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1/file2.jpg`,
|
|
);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/polemonium_reptans.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir2/file3.jpg`,
|
|
);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/tanners_ridge.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir3/file4.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
await waitForEvent(libraryService, 'add');
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const assets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(assets.length).toEqual(3);
|
|
});
|
|
|
|
it('should offline a removed file', async () => {
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/polemonium_reptans.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1/file.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const addedAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(addedAssets.length).toEqual(1);
|
|
|
|
await fs.unlink(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1/file.jpg`);
|
|
|
|
await waitForEvent(libraryService, 'unlink');
|
|
|
|
const afterAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(afterAssets[0].isOffline).toEqual(true);
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('Configuration', () => {
|
|
let library: LibraryResponseDto;
|
|
|
|
beforeEach(async () => {
|
|
library = await api.libraryApi.create(server, admin.accessToken, {
|
|
type: LibraryType.EXTERNAL,
|
|
importPaths: [
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir2`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir3`,
|
|
],
|
|
});
|
|
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir1`, { recursive: true });
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir2`, { recursive: true });
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir3`, { recursive: true });
|
|
});
|
|
|
|
it('should use an updated import paths', async () => {
|
|
await fs.mkdir(`${IMMICH_TEST_ASSET_TEMP_PATH}/dir4`, { recursive: true });
|
|
|
|
await api.libraryApi.setImportPaths(server, admin.accessToken, library.id, [
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir4`,
|
|
]);
|
|
|
|
await fs.copyFile(
|
|
`${IMMICH_TEST_ASSET_PATH}/albums/nature/polemonium_reptans.jpg`,
|
|
`${IMMICH_TEST_ASSET_TEMP_PATH}/dir4/file.jpg`,
|
|
);
|
|
|
|
await waitForEvent(libraryService, 'add');
|
|
|
|
const afterAssets = await api.assetApi.getAllAssets(server, admin.accessToken);
|
|
expect(afterAssets.length).toEqual(1);
|
|
});
|
|
});
|
|
});
|