mirror of
https://github.com/immich-app/immich.git
synced 2026-02-10 19:07:55 +03:00
156 lines
5.1 KiB
TypeScript
156 lines
5.1 KiB
TypeScript
import { Kysely } from 'kysely';
|
|
import { Stats } from 'node:fs';
|
|
import { writeFile } from 'node:fs/promises';
|
|
import { tmpdir } from 'node:os';
|
|
import { join } from 'node:path';
|
|
import { AssetJobRepository } from 'src/repositories/asset-job.repository';
|
|
import { AssetRepository } from 'src/repositories/asset.repository';
|
|
import { ConfigRepository } from 'src/repositories/config.repository';
|
|
import { EventRepository } from 'src/repositories/event.repository';
|
|
import { LoggingRepository } from 'src/repositories/logging.repository';
|
|
import { MetadataRepository } from 'src/repositories/metadata.repository';
|
|
import { StorageRepository } from 'src/repositories/storage.repository';
|
|
import { SystemMetadataRepository } from 'src/repositories/system-metadata.repository';
|
|
import { TagRepository } from 'src/repositories/tag.repository';
|
|
import { DB } from 'src/schema';
|
|
import { MetadataService } from 'src/services/metadata.service';
|
|
import { newMediumService } from 'test/medium.factory';
|
|
import { getKyselyDB, newRandomImage } from 'test/utils';
|
|
|
|
type TimeZoneTest = {
|
|
description: string;
|
|
serverTimeZone?: string;
|
|
exifData: Record<string, any>;
|
|
expected: {
|
|
localDateTime: string;
|
|
dateTimeOriginal: string;
|
|
timeZone: string | null;
|
|
};
|
|
};
|
|
|
|
let defaultDatabase: Kysely<DB>;
|
|
|
|
const setup = (db?: Kysely<DB>) => {
|
|
const { sut, ctx } = newMediumService(MetadataService, {
|
|
database: db || defaultDatabase,
|
|
real: [
|
|
AssetRepository,
|
|
AssetJobRepository,
|
|
ConfigRepository,
|
|
MetadataRepository,
|
|
SystemMetadataRepository,
|
|
TagRepository,
|
|
],
|
|
mock: [EventRepository, StorageRepository, LoggingRepository],
|
|
});
|
|
|
|
ctx.getMock(StorageRepository).stat.mockResolvedValue({
|
|
size: 123_456,
|
|
mtime: new Date(654_321),
|
|
mtimeMs: 654_321,
|
|
birthtimeMs: 654_322,
|
|
} as Stats);
|
|
|
|
return { sut, ctx };
|
|
};
|
|
|
|
const createTestFile = async (exifData: Record<string, any>) => {
|
|
const { ctx } = setup();
|
|
const data = newRandomImage();
|
|
const filePath = join(tmpdir(), 'test.png');
|
|
await writeFile(filePath, data);
|
|
await ctx.get(MetadataRepository).writeTags(filePath, exifData);
|
|
return { filePath };
|
|
};
|
|
|
|
beforeAll(async () => {
|
|
defaultDatabase = await getKyselyDB();
|
|
});
|
|
|
|
describe(MetadataService.name, () => {
|
|
afterEach(() => {
|
|
vi.unstubAllEnvs();
|
|
});
|
|
|
|
it('should be defined', () => {
|
|
const { sut } = setup();
|
|
expect(sut).toBeDefined();
|
|
});
|
|
|
|
describe('handleMetadataExtraction', () => {
|
|
const timeZoneTests: TimeZoneTest[] = [
|
|
{
|
|
description: 'should handle no time zone information',
|
|
exifData: {
|
|
DateTimeOriginal: '2022:01:01 00:00:00',
|
|
},
|
|
expected: {
|
|
localDateTime: '2022-01-01T00:00:00.000Z',
|
|
dateTimeOriginal: '2022-01-01T00:00:00.000Z',
|
|
timeZone: null,
|
|
},
|
|
},
|
|
{
|
|
description: 'should handle a +13:00 time zone',
|
|
exifData: {
|
|
DateTimeOriginal: '2022:01:01 00:00:00+13:00',
|
|
},
|
|
expected: {
|
|
localDateTime: '2022-01-01T00:00:00.000Z',
|
|
dateTimeOriginal: '2021-12-31T11:00:00.000Z',
|
|
timeZone: 'UTC+13',
|
|
},
|
|
},
|
|
];
|
|
|
|
it.each(timeZoneTests)('$description', async ({ exifData, serverTimeZone, expected }) => {
|
|
vi.stubEnv('TZ', serverTimeZone);
|
|
|
|
const { sut, ctx } = setup();
|
|
ctx.getMock(EventRepository).emit.mockResolvedValue();
|
|
const { filePath } = await createTestFile(exifData);
|
|
const { user } = await ctx.newUser();
|
|
const { asset } = await ctx.newAsset({ originalPath: filePath, ownerId: user.id });
|
|
await ctx.newExif({ assetId: asset.id, description: '' });
|
|
|
|
await sut.handleMetadataExtraction({ id: asset.id });
|
|
|
|
await expect(
|
|
ctx.database
|
|
.selectFrom('asset_exif')
|
|
.select(['dateTimeOriginal', 'timeZone', 'lockedProperties'])
|
|
.where('assetId', '=', asset.id)
|
|
.executeTakeFirstOrThrow(),
|
|
).resolves.toEqual({
|
|
dateTimeOriginal: new Date(expected.dateTimeOriginal),
|
|
timeZone: expected.timeZone,
|
|
lockedProperties: null,
|
|
});
|
|
|
|
await expect(ctx.get(AssetRepository).getById(asset.id)).resolves.toEqual(
|
|
expect.objectContaining({ localDateTime: new Date(expected.localDateTime) }),
|
|
);
|
|
});
|
|
|
|
it('should handle dates far in the future', async () => {
|
|
const { sut, ctx } = setup();
|
|
ctx.getMock(EventRepository).emit.mockResolvedValue();
|
|
const { filePath } = await createTestFile({ CreateDate: '42603:05:04 04:12:48' });
|
|
const { user } = await ctx.newUser();
|
|
const { asset } = await ctx.newAsset({ originalPath: filePath, ownerId: user.id });
|
|
await ctx.newExif({ assetId: asset.id, description: '' });
|
|
|
|
await sut.handleMetadataExtraction({ id: asset.id });
|
|
|
|
await expect(
|
|
ctx.database
|
|
.selectFrom('asset_exif')
|
|
.where('assetId', '=', asset.id)
|
|
.select('dateTimeOriginal')
|
|
.executeTakeFirstOrThrow(),
|
|
// note that this date is technically wrong. it does not throw though and should get the user's attention either way.
|
|
).resolves.toEqual({ dateTimeOriginal: new Date('4260-03-05T04:04:12.000Z') });
|
|
});
|
|
});
|
|
});
|