2024-01-26 15:19:13 +01:00
|
|
|
import { BadRequestException } from '@nestjs/common';
|
2024-03-20 23:53:07 +01:00
|
|
|
import { DownloadResponseDto } from 'src/dtos/download.dto';
|
2024-04-16 16:44:45 +02:00
|
|
|
import { AssetEntity } from 'src/entities/asset.entity';
|
2024-03-21 12:59:49 +01:00
|
|
|
import { IAssetRepository } from 'src/interfaces/asset.interface';
|
|
|
|
import { IStorageRepository } from 'src/interfaces/storage.interface';
|
2024-03-21 00:07:30 +01:00
|
|
|
import { DownloadService } from 'src/services/download.service';
|
2024-03-21 04:15:09 +01:00
|
|
|
import { CacheControl, ImmichFileResponse } from 'src/utils/file';
|
2024-03-20 19:32:04 +01:00
|
|
|
import { assetStub } from 'test/fixtures/asset.stub';
|
|
|
|
import { authStub } from 'test/fixtures/auth.stub';
|
|
|
|
import { IAccessRepositoryMock, newAccessRepositoryMock } from 'test/repositories/access.repository.mock';
|
|
|
|
import { newAssetRepositoryMock } from 'test/repositories/asset.repository.mock';
|
|
|
|
import { newStorageRepositoryMock } from 'test/repositories/storage.repository.mock';
|
2024-01-26 15:19:13 +01:00
|
|
|
import { Readable } from 'typeorm/platform/PlatformTools.js';
|
2024-04-16 16:44:45 +02:00
|
|
|
import { Mocked, vitest } from 'vitest';
|
2024-01-26 15:19:13 +01:00
|
|
|
|
|
|
|
const downloadResponse: DownloadResponseDto = {
|
|
|
|
totalSize: 105_000,
|
|
|
|
archives: [
|
|
|
|
{
|
|
|
|
assetIds: ['asset-id', 'asset-id'],
|
|
|
|
size: 105_000,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
};
|
|
|
|
|
|
|
|
describe(DownloadService.name, () => {
|
|
|
|
let sut: DownloadService;
|
|
|
|
let accessMock: IAccessRepositoryMock;
|
2024-04-16 16:44:45 +02:00
|
|
|
let assetMock: Mocked<IAssetRepository>;
|
|
|
|
let storageMock: Mocked<IStorageRepository>;
|
2024-01-26 15:19:13 +01:00
|
|
|
|
|
|
|
it('should work', () => {
|
|
|
|
expect(sut).toBeDefined();
|
|
|
|
});
|
|
|
|
|
2024-03-05 23:23:06 +01:00
|
|
|
beforeEach(() => {
|
2024-01-26 15:19:13 +01:00
|
|
|
accessMock = newAccessRepositoryMock();
|
|
|
|
assetMock = newAssetRepositoryMock();
|
|
|
|
storageMock = newStorageRepositoryMock();
|
|
|
|
|
|
|
|
sut = new DownloadService(accessMock, assetMock, storageMock);
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('downloadFile', () => {
|
|
|
|
it('should require the asset.download permission', async () => {
|
|
|
|
await expect(sut.downloadFile(authStub.admin, 'asset-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
|
|
|
expect(accessMock.asset.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['asset-1']));
|
|
|
|
expect(accessMock.asset.checkAlbumAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['asset-1']));
|
|
|
|
expect(accessMock.asset.checkPartnerAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['asset-1']));
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw an error if the asset is not found', async () => {
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1']));
|
|
|
|
assetMock.getByIds.mockResolvedValue([]);
|
|
|
|
|
|
|
|
await expect(sut.downloadFile(authStub.admin, 'asset-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
|
|
|
expect(assetMock.getByIds).toHaveBeenCalledWith(['asset-1']);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should throw an error if the asset is offline', async () => {
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1']));
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.offline]);
|
|
|
|
|
|
|
|
await expect(sut.downloadFile(authStub.admin, 'asset-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
|
|
|
|
expect(assetMock.getByIds).toHaveBeenCalledWith(['asset-1']);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should download a file', async () => {
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1']));
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image]);
|
|
|
|
|
|
|
|
await expect(sut.downloadFile(authStub.admin, 'asset-1')).resolves.toEqual(
|
|
|
|
new ImmichFileResponse({
|
|
|
|
path: '/original/path.jpg',
|
|
|
|
contentType: 'image/jpeg',
|
|
|
|
cacheControl: CacheControl.NONE,
|
|
|
|
}),
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should download an archive', async () => {
|
|
|
|
const archiveMock = {
|
2024-04-16 16:44:45 +02:00
|
|
|
addFile: vitest.fn(),
|
|
|
|
finalize: vitest.fn(),
|
2024-01-26 15:19:13 +01:00
|
|
|
stream: new Readable(),
|
|
|
|
};
|
|
|
|
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1', 'asset-2']));
|
2024-03-05 22:04:43 +01:00
|
|
|
assetMock.getByIds.mockResolvedValue([
|
|
|
|
{ ...assetStub.noResizePath, id: 'asset-1' },
|
|
|
|
{ ...assetStub.noWebpPath, id: 'asset-2' },
|
|
|
|
]);
|
2024-01-26 15:19:13 +01:00
|
|
|
storageMock.createZipStream.mockReturnValue(archiveMock);
|
|
|
|
|
|
|
|
await expect(sut.downloadArchive(authStub.admin, { assetIds: ['asset-1', 'asset-2'] })).resolves.toEqual({
|
|
|
|
stream: archiveMock.stream,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(archiveMock.addFile).toHaveBeenCalledTimes(2);
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(1, 'upload/library/IMG_123.jpg', 'IMG_123.jpg');
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(2, 'upload/library/IMG_456.jpg', 'IMG_456.jpg');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should handle duplicate file names', async () => {
|
|
|
|
const archiveMock = {
|
2024-04-16 16:44:45 +02:00
|
|
|
addFile: vitest.fn(),
|
|
|
|
finalize: vitest.fn(),
|
2024-01-26 15:19:13 +01:00
|
|
|
stream: new Readable(),
|
|
|
|
};
|
|
|
|
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1', 'asset-2']));
|
2024-03-05 22:04:43 +01:00
|
|
|
assetMock.getByIds.mockResolvedValue([
|
|
|
|
{ ...assetStub.noResizePath, id: 'asset-1' },
|
|
|
|
{ ...assetStub.noResizePath, id: 'asset-2' },
|
|
|
|
]);
|
|
|
|
storageMock.createZipStream.mockReturnValue(archiveMock);
|
|
|
|
|
|
|
|
await expect(sut.downloadArchive(authStub.admin, { assetIds: ['asset-1', 'asset-2'] })).resolves.toEqual({
|
|
|
|
stream: archiveMock.stream,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(archiveMock.addFile).toHaveBeenCalledTimes(2);
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(1, 'upload/library/IMG_123.jpg', 'IMG_123.jpg');
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(2, 'upload/library/IMG_123.jpg', 'IMG_123+1.jpg');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should be deterministic', async () => {
|
|
|
|
const archiveMock = {
|
2024-04-16 16:44:45 +02:00
|
|
|
addFile: vitest.fn(),
|
|
|
|
finalize: vitest.fn(),
|
2024-03-05 22:04:43 +01:00
|
|
|
stream: new Readable(),
|
|
|
|
};
|
|
|
|
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1', 'asset-2']));
|
|
|
|
assetMock.getByIds.mockResolvedValue([
|
|
|
|
{ ...assetStub.noResizePath, id: 'asset-2' },
|
|
|
|
{ ...assetStub.noResizePath, id: 'asset-1' },
|
|
|
|
]);
|
2024-01-26 15:19:13 +01:00
|
|
|
storageMock.createZipStream.mockReturnValue(archiveMock);
|
|
|
|
|
|
|
|
await expect(sut.downloadArchive(authStub.admin, { assetIds: ['asset-1', 'asset-2'] })).resolves.toEqual({
|
|
|
|
stream: archiveMock.stream,
|
|
|
|
});
|
|
|
|
|
|
|
|
expect(archiveMock.addFile).toHaveBeenCalledTimes(2);
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(1, 'upload/library/IMG_123.jpg', 'IMG_123.jpg');
|
|
|
|
expect(archiveMock.addFile).toHaveBeenNthCalledWith(2, 'upload/library/IMG_123.jpg', 'IMG_123+1.jpg');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe('getDownloadInfo', () => {
|
|
|
|
it('should throw an error for an invalid dto', async () => {
|
|
|
|
await expect(sut.getDownloadInfo(authStub.admin, {})).rejects.toBeInstanceOf(BadRequestException);
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return a list of archives (assetIds)', async () => {
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(['asset-1', 'asset-2']));
|
|
|
|
assetMock.getByIds.mockResolvedValue([assetStub.image, assetStub.video]);
|
|
|
|
|
|
|
|
const assetIds = ['asset-1', 'asset-2'];
|
|
|
|
await expect(sut.getDownloadInfo(authStub.admin, { assetIds })).resolves.toEqual(downloadResponse);
|
|
|
|
|
2024-03-14 06:58:09 +01:00
|
|
|
expect(assetMock.getByIds).toHaveBeenCalledWith(['asset-1', 'asset-2'], { exifInfo: true });
|
2024-01-26 15:19:13 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
it('should return a list of archives (albumId)', async () => {
|
|
|
|
accessMock.album.checkOwnerAccess.mockResolvedValue(new Set(['album-1']));
|
|
|
|
assetMock.getByAlbumId.mockResolvedValue({
|
|
|
|
items: [assetStub.image, assetStub.video],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
await expect(sut.getDownloadInfo(authStub.admin, { albumId: 'album-1' })).resolves.toEqual(downloadResponse);
|
|
|
|
|
|
|
|
expect(accessMock.album.checkOwnerAccess).toHaveBeenCalledWith(authStub.admin.user.id, new Set(['album-1']));
|
|
|
|
expect(assetMock.getByAlbumId).toHaveBeenCalledWith({ take: 2500, skip: 0 }, 'album-1');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should return a list of archives (userId)', async () => {
|
|
|
|
accessMock.library.checkOwnerAccess.mockResolvedValue(new Set([authStub.admin.user.id]));
|
|
|
|
assetMock.getByUserId.mockResolvedValue({
|
|
|
|
items: [assetStub.image, assetStub.video],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
await expect(sut.getDownloadInfo(authStub.admin, { userId: authStub.admin.user.id })).resolves.toEqual(
|
|
|
|
downloadResponse,
|
|
|
|
);
|
|
|
|
|
|
|
|
expect(assetMock.getByUserId).toHaveBeenCalledWith({ take: 2500, skip: 0 }, authStub.admin.user.id, {
|
|
|
|
isVisible: true,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should split archives by size', async () => {
|
|
|
|
accessMock.library.checkOwnerAccess.mockResolvedValue(new Set([authStub.admin.user.id]));
|
|
|
|
|
|
|
|
assetMock.getByUserId.mockResolvedValue({
|
|
|
|
items: [
|
|
|
|
{ ...assetStub.image, id: 'asset-1' },
|
|
|
|
{ ...assetStub.video, id: 'asset-2' },
|
|
|
|
{ ...assetStub.withLocation, id: 'asset-3' },
|
|
|
|
{ ...assetStub.noWebpPath, id: 'asset-4' },
|
|
|
|
],
|
|
|
|
hasNextPage: false,
|
|
|
|
});
|
|
|
|
|
|
|
|
await expect(
|
|
|
|
sut.getDownloadInfo(authStub.admin, {
|
|
|
|
userId: authStub.admin.user.id,
|
|
|
|
archiveSize: 30_000,
|
|
|
|
}),
|
|
|
|
).resolves.toEqual({
|
|
|
|
totalSize: 251_456,
|
|
|
|
archives: [
|
|
|
|
{ assetIds: ['asset-1', 'asset-2'], size: 105_000 },
|
|
|
|
{ assetIds: ['asset-3', 'asset-4'], size: 146_456 },
|
|
|
|
],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
it('should include the video portion of a live photo', async () => {
|
|
|
|
const assetIds = [assetStub.livePhotoStillAsset.id];
|
2024-04-16 16:44:45 +02:00
|
|
|
const assets = [assetStub.livePhotoStillAsset, assetStub.livePhotoMotionAsset];
|
2024-01-26 15:19:13 +01:00
|
|
|
|
|
|
|
accessMock.asset.checkOwnerAccess.mockResolvedValue(new Set(assetIds));
|
2024-04-16 16:44:45 +02:00
|
|
|
assetMock.getByIds.mockImplementation(
|
|
|
|
(ids) =>
|
|
|
|
Promise.resolve(
|
|
|
|
ids.map((id) => assets.find((asset) => asset.id === id)).filter((asset) => !!asset),
|
|
|
|
) as Promise<AssetEntity[]>,
|
|
|
|
);
|
2024-01-26 15:19:13 +01:00
|
|
|
|
|
|
|
await expect(sut.getDownloadInfo(authStub.admin, { assetIds })).resolves.toEqual({
|
|
|
|
totalSize: 125_000,
|
|
|
|
archives: [
|
|
|
|
{
|
|
|
|
assetIds: [assetStub.livePhotoStillAsset.id, assetStub.livePhotoMotionAsset.id],
|
|
|
|
size: 125_000,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|