mirror of
https://github.com/immich-app/immich.git
synced 2025-01-24 12:42:44 +01:00
98b72fdb9b
* Person birth date (data layer) * Person birth date (data layer) * Person birth date (service layer) * Person birth date (service layer, API) * Person birth date (service layer, API) * Person birth date (UI) (wip) * Person birth date (UI) (wip) * Person birth date (UI) (wip) * Person birth date (UI) (wip) * UI: Use "date of birth" everywhere * UI: better modal dialog Similar to the API key modal. * UI: set date of birth from people page * Use typed events for modal dispatcher * Date of birth tests (wip) * Regenerate API * Code formatting * Fix Svelte typing * Fix Svelte typing * Fix person model [skip ci] * Minor refactoring [skip ci] * Typed event dispatcher [skip ci] * Refactor typed event dispatcher [skip ci] * Fix unchanged birthdate check [skip ci] * Remove unnecessary custom transformer [skip ci] * PersonUpdate: call search index update job only when needed * Regenerate API * Code formatting * Fix tests * Fix DTO * Regenerate API * chore: verbiage and view mode * feat: show current age * test: person e2e * fix: show name for birth date selection --------- Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
315 lines
12 KiB
TypeScript
315 lines
12 KiB
TypeScript
import { BadRequestException, NotFoundException } from '@nestjs/common';
|
|
import {
|
|
assetStub,
|
|
authStub,
|
|
faceStub,
|
|
newJobRepositoryMock,
|
|
newPersonRepositoryMock,
|
|
newStorageRepositoryMock,
|
|
personStub,
|
|
} from '@test';
|
|
import { BulkIdErrorReason } from '../asset';
|
|
import { IJobRepository, JobName } from '../job';
|
|
import { IStorageRepository } from '../storage';
|
|
import { PersonResponseDto } from './person.dto';
|
|
import { IPersonRepository } from './person.repository';
|
|
import { PersonService } from './person.service';
|
|
|
|
const responseDto: PersonResponseDto = {
|
|
id: 'person-1',
|
|
name: 'Person 1',
|
|
birthDate: null,
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: false,
|
|
};
|
|
|
|
describe(PersonService.name, () => {
|
|
let sut: PersonService;
|
|
let personMock: jest.Mocked<IPersonRepository>;
|
|
let storageMock: jest.Mocked<IStorageRepository>;
|
|
let jobMock: jest.Mocked<IJobRepository>;
|
|
|
|
beforeEach(async () => {
|
|
personMock = newPersonRepositoryMock();
|
|
storageMock = newStorageRepositoryMock();
|
|
jobMock = newJobRepositoryMock();
|
|
sut = new PersonService(personMock, storageMock, jobMock);
|
|
});
|
|
|
|
it('should be defined', () => {
|
|
expect(sut).toBeDefined();
|
|
});
|
|
|
|
describe('getAll', () => {
|
|
it('should get all people with thumbnails', async () => {
|
|
personMock.getAll.mockResolvedValue([personStub.withName, personStub.noThumbnail]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: undefined })).resolves.toEqual({
|
|
total: 1,
|
|
visible: 1,
|
|
people: [responseDto],
|
|
});
|
|
expect(personMock.getAll).toHaveBeenCalledWith(authStub.admin.id, { minimumFaceCount: 1, withHidden: false });
|
|
});
|
|
it('should get all visible people with thumbnails', async () => {
|
|
personMock.getAll.mockResolvedValue([personStub.withName, personStub.hidden]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: false })).resolves.toEqual({
|
|
total: 2,
|
|
visible: 1,
|
|
people: [responseDto],
|
|
});
|
|
expect(personMock.getAll).toHaveBeenCalledWith(authStub.admin.id, { minimumFaceCount: 1, withHidden: false });
|
|
});
|
|
it('should get all hidden and visible people with thumbnails', async () => {
|
|
personMock.getAll.mockResolvedValue([personStub.withName, personStub.hidden]);
|
|
await expect(sut.getAll(authStub.admin, { withHidden: true })).resolves.toEqual({
|
|
total: 2,
|
|
visible: 1,
|
|
people: [
|
|
responseDto,
|
|
{
|
|
id: 'person-1',
|
|
name: '',
|
|
birthDate: null,
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: true,
|
|
},
|
|
],
|
|
});
|
|
expect(personMock.getAll).toHaveBeenCalledWith(authStub.admin.id, { minimumFaceCount: 1, withHidden: true });
|
|
});
|
|
});
|
|
|
|
describe('getById', () => {
|
|
it('should throw a bad request when person is not found', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
await expect(sut.getById(authStub.admin, 'person-1')).rejects.toBeInstanceOf(BadRequestException);
|
|
});
|
|
|
|
it('should get a person by id', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
await expect(sut.getById(authStub.admin, 'person-1')).resolves.toEqual(responseDto);
|
|
expect(personMock.getById).toHaveBeenCalledWith(authStub.admin.id, 'person-1');
|
|
});
|
|
});
|
|
|
|
describe('getThumbnail', () => {
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
await expect(sut.getThumbnail(authStub.admin, 'person-1')).rejects.toBeInstanceOf(NotFoundException);
|
|
expect(storageMock.createReadStream).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should throw an error when person has no thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noThumbnail);
|
|
await expect(sut.getThumbnail(authStub.admin, 'person-1')).rejects.toBeInstanceOf(NotFoundException);
|
|
expect(storageMock.createReadStream).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should serve the thumbnail', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
await sut.getThumbnail(authStub.admin, 'person-1');
|
|
expect(storageMock.createReadStream).toHaveBeenCalledWith('/path/to/thumbnail.jpg', 'image/jpeg');
|
|
});
|
|
});
|
|
|
|
describe('getAssets', () => {
|
|
it("should return a person's assets", async () => {
|
|
personMock.getAssets.mockResolvedValue([assetStub.image, assetStub.video]);
|
|
await sut.getAssets(authStub.admin, 'person-1');
|
|
expect(personMock.getAssets).toHaveBeenCalledWith('admin_id', 'person-1');
|
|
});
|
|
});
|
|
|
|
describe('update', () => {
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
await expect(sut.update(authStub.admin, 'person-1', { name: 'Person 1' })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it("should update a person's name", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noName);
|
|
personMock.update.mockResolvedValue(personStub.withName);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { name: 'Person 1' })).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('admin_id', 'person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', name: 'Person 1' });
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_INDEX_ASSET,
|
|
data: { ids: [assetStub.image.id] },
|
|
});
|
|
});
|
|
|
|
it("should update a person's date of birth", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.noBirthDate);
|
|
personMock.update.mockResolvedValue(personStub.withBirthDate);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { birthDate: new Date('1976-06-30') })).resolves.toEqual({
|
|
id: 'person-1',
|
|
name: 'Person 1',
|
|
birthDate: new Date('1976-06-30'),
|
|
thumbnailPath: '/path/to/thumbnail.jpg',
|
|
isHidden: false,
|
|
});
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('admin_id', 'person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', birthDate: new Date('1976-06-30') });
|
|
expect(jobMock.queue).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should update a person visibility', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.hidden);
|
|
personMock.update.mockResolvedValue(personStub.withName);
|
|
personMock.getAssets.mockResolvedValue([assetStub.image]);
|
|
|
|
await expect(sut.update(authStub.admin, 'person-1', { isHidden: false })).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('admin_id', 'person-1');
|
|
expect(personMock.update).toHaveBeenCalledWith({ id: 'person-1', isHidden: false });
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_INDEX_ASSET,
|
|
data: { ids: [assetStub.image.id] },
|
|
});
|
|
});
|
|
|
|
it("should update a person's thumbnailPath", async () => {
|
|
personMock.getById.mockResolvedValue(personStub.withName);
|
|
personMock.getFaceById.mockResolvedValue(faceStub.face1);
|
|
|
|
await expect(
|
|
sut.update(authStub.admin, 'person-1', { featureFaceAssetId: faceStub.face1.assetId }),
|
|
).resolves.toEqual(responseDto);
|
|
|
|
expect(personMock.getById).toHaveBeenCalledWith('admin_id', 'person-1');
|
|
expect(personMock.getFaceById).toHaveBeenCalledWith({
|
|
assetId: faceStub.face1.assetId,
|
|
personId: 'person-1',
|
|
});
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.GENERATE_FACE_THUMBNAIL,
|
|
data: {
|
|
assetId: faceStub.face1.assetId,
|
|
personId: 'person-1',
|
|
boundingBox: {
|
|
x1: faceStub.face1.boundingBoxX1,
|
|
x2: faceStub.face1.boundingBoxX2,
|
|
y1: faceStub.face1.boundingBoxY1,
|
|
y2: faceStub.face1.boundingBoxY2,
|
|
},
|
|
imageHeight: faceStub.face1.imageHeight,
|
|
imageWidth: faceStub.face1.imageWidth,
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('updateAll', () => {
|
|
it('should throw an error when personId is invalid', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
await expect(
|
|
sut.updatePeople(authStub.admin, { people: [{ id: 'person-1', name: 'Person 1' }] }),
|
|
).resolves.toEqual([{ error: BulkIdErrorReason.UNKNOWN, id: 'person-1', success: false }]);
|
|
expect(personMock.update).not.toHaveBeenCalled();
|
|
});
|
|
});
|
|
|
|
describe('handlePersonCleanup', () => {
|
|
it('should delete people without faces', async () => {
|
|
personMock.getAllWithoutFaces.mockResolvedValue([personStub.noName]);
|
|
|
|
await sut.handlePersonCleanup();
|
|
|
|
expect(personMock.delete).toHaveBeenCalledWith(personStub.noName);
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.DELETE_FILES,
|
|
data: { files: ['/path/to/thumbnail.jpg'] },
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('mergePerson', () => {
|
|
it('should merge two people', async () => {
|
|
personMock.getById.mockResolvedValueOnce(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValueOnce(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([]);
|
|
personMock.delete.mockResolvedValue(personStub.mergePerson);
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: true },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(personMock.reassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(personMock.delete).toHaveBeenCalledWith(personStub.mergePerson);
|
|
});
|
|
|
|
it('should delete conflicting faces before merging', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValue(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([assetStub.image.id]);
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: true },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).toHaveBeenCalledWith({
|
|
newPersonId: personStub.primaryPerson.id,
|
|
oldPersonId: personStub.mergePerson.id,
|
|
});
|
|
|
|
expect(jobMock.queue).toHaveBeenCalledWith({
|
|
name: JobName.SEARCH_REMOVE_FACE,
|
|
data: { assetId: assetStub.image.id, personId: personStub.mergePerson.id },
|
|
});
|
|
});
|
|
|
|
it('should throw an error when the primary person is not found', async () => {
|
|
personMock.getById.mockResolvedValue(null);
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).rejects.toBeInstanceOf(
|
|
BadRequestException,
|
|
);
|
|
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should handle invalid merge ids', async () => {
|
|
personMock.getById.mockResolvedValueOnce(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValueOnce(null);
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: false, error: BulkIdErrorReason.NOT_FOUND },
|
|
]);
|
|
|
|
expect(personMock.prepareReassignFaces).not.toHaveBeenCalled();
|
|
expect(personMock.reassignFaces).not.toHaveBeenCalled();
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
});
|
|
|
|
it('should handle an error reassigning faces', async () => {
|
|
personMock.getById.mockResolvedValue(personStub.primaryPerson);
|
|
personMock.getById.mockResolvedValue(personStub.mergePerson);
|
|
personMock.prepareReassignFaces.mockResolvedValue([assetStub.image.id]);
|
|
personMock.reassignFaces.mockRejectedValue(new Error('update failed'));
|
|
|
|
await expect(sut.mergePerson(authStub.admin, 'person-1', { ids: ['person-2'] })).resolves.toEqual([
|
|
{ id: 'person-2', success: false, error: BulkIdErrorReason.UNKNOWN },
|
|
]);
|
|
|
|
expect(personMock.delete).not.toHaveBeenCalled();
|
|
});
|
|
});
|
|
});
|