2025-02-04 09:52:17 +01:00
|
|
|
import { getPerson, LoginResponseDto, PersonResponseDto } from '@immich/sdk';
|
2024-02-21 08:28:03 -05:00
|
|
|
import { uuidDto } from 'src/fixtures';
|
|
|
|
|
import { errorDto } from 'src/responses';
|
2025-02-04 09:52:17 +01:00
|
|
|
import { app, asBearerAuth, utils } from 'src/utils';
|
2024-02-21 08:28:03 -05:00
|
|
|
import request from 'supertest';
|
|
|
|
|
import { beforeAll, beforeEach, describe, expect, it } from 'vitest';
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('/people', () => {
|
2024-02-21 08:28:03 -05:00
|
|
|
let admin: LoginResponseDto;
|
|
|
|
|
let visiblePerson: PersonResponseDto;
|
|
|
|
|
let hiddenPerson: PersonResponseDto;
|
2024-03-05 00:11:54 +01:00
|
|
|
let multipleAssetsPerson: PersonResponseDto;
|
2024-02-21 08:28:03 -05:00
|
|
|
|
2025-06-13 16:18:44 -03:00
|
|
|
let nameAlicePerson: PersonResponseDto;
|
|
|
|
|
let nameBobPerson: PersonResponseDto;
|
|
|
|
|
let nameCharliePerson: PersonResponseDto;
|
|
|
|
|
let nameNullPerson: PersonResponseDto;
|
|
|
|
|
|
2024-02-21 08:28:03 -05:00
|
|
|
beforeAll(async () => {
|
2024-03-07 10:14:36 -05:00
|
|
|
await utils.resetDatabase();
|
|
|
|
|
admin = await utils.adminSetup();
|
2024-02-21 08:28:03 -05:00
|
|
|
|
2025-06-13 16:18:44 -03:00
|
|
|
[
|
|
|
|
|
visiblePerson,
|
|
|
|
|
hiddenPerson,
|
|
|
|
|
multipleAssetsPerson,
|
|
|
|
|
nameCharliePerson,
|
|
|
|
|
nameBobPerson,
|
|
|
|
|
nameAlicePerson,
|
|
|
|
|
nameNullPerson,
|
|
|
|
|
] = await Promise.all([
|
2024-03-07 10:14:36 -05:00
|
|
|
utils.createPerson(admin.accessToken, {
|
2024-02-21 08:28:03 -05:00
|
|
|
name: 'visible_person',
|
|
|
|
|
}),
|
2024-03-07 10:14:36 -05:00
|
|
|
utils.createPerson(admin.accessToken, {
|
2024-02-21 08:28:03 -05:00
|
|
|
name: 'hidden_person',
|
|
|
|
|
isHidden: true,
|
|
|
|
|
}),
|
2024-03-07 10:14:36 -05:00
|
|
|
utils.createPerson(admin.accessToken, {
|
2024-03-05 00:11:54 +01:00
|
|
|
name: 'multiple_assets_person',
|
|
|
|
|
}),
|
2025-06-13 16:18:44 -03:00
|
|
|
// --- Setup for the specific sorting test ---
|
|
|
|
|
utils.createPerson(admin.accessToken, {
|
|
|
|
|
name: 'Charlie',
|
|
|
|
|
}),
|
|
|
|
|
utils.createPerson(admin.accessToken, {
|
|
|
|
|
name: 'Bob',
|
|
|
|
|
}),
|
|
|
|
|
utils.createPerson(admin.accessToken, {
|
|
|
|
|
name: 'Alice',
|
|
|
|
|
}),
|
|
|
|
|
utils.createPerson(admin.accessToken, {
|
|
|
|
|
name: '',
|
|
|
|
|
}),
|
2024-02-21 08:28:03 -05:00
|
|
|
]);
|
|
|
|
|
|
2024-03-07 10:14:36 -05:00
|
|
|
const asset1 = await utils.createAsset(admin.accessToken);
|
|
|
|
|
const asset2 = await utils.createAsset(admin.accessToken);
|
2025-06-13 16:18:44 -03:00
|
|
|
const asset3 = await utils.createAsset(admin.accessToken);
|
2024-02-21 08:28:03 -05:00
|
|
|
|
|
|
|
|
await Promise.all([
|
2024-03-07 10:14:36 -05:00
|
|
|
utils.createFace({ assetId: asset1.id, personId: visiblePerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: hiddenPerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: multipleAssetsPerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: multipleAssetsPerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset2.id, personId: multipleAssetsPerson.id }),
|
2025-06-13 16:18:44 -03:00
|
|
|
utils.createFace({ assetId: asset3.id, personId: multipleAssetsPerson.id }),
|
|
|
|
|
// Named persons
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: nameCharliePerson.id }), // 1 asset
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: nameBobPerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset2.id, personId: nameBobPerson.id }), // 2 assets
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: nameAlicePerson.id }), // 1 asset
|
|
|
|
|
// Null-named person
|
|
|
|
|
utils.createFace({ assetId: asset1.id, personId: nameNullPerson.id }),
|
|
|
|
|
utils.createFace({ assetId: asset2.id, personId: nameNullPerson.id }), // 2 assets
|
2024-02-21 08:28:03 -05:00
|
|
|
]);
|
|
|
|
|
});
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('GET /people', () => {
|
2024-02-21 08:28:03 -05:00
|
|
|
beforeEach(async () => {});
|
|
|
|
|
it('should return all people (including hidden)', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.get('/people')
|
2024-02-21 08:28:03 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.query({ withHidden: true });
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toEqual({
|
2024-07-25 21:59:28 +02:00
|
|
|
hasNextPage: false,
|
2025-06-13 16:18:44 -03:00
|
|
|
total: 7,
|
2024-02-21 23:03:45 +01:00
|
|
|
hidden: 1,
|
2024-02-21 08:28:03 -05:00
|
|
|
people: [
|
2024-03-05 00:11:54 +01:00
|
|
|
expect.objectContaining({ name: 'multiple_assets_person' }),
|
2025-06-13 16:18:44 -03:00
|
|
|
expect.objectContaining({ name: 'Bob' }),
|
|
|
|
|
expect.objectContaining({ name: 'Alice' }),
|
|
|
|
|
expect.objectContaining({ name: 'Charlie' }),
|
2024-02-21 08:28:03 -05:00
|
|
|
expect.objectContaining({ name: 'visible_person' }),
|
|
|
|
|
expect.objectContaining({ name: 'hidden_person' }),
|
|
|
|
|
],
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2025-06-13 16:18:44 -03:00
|
|
|
it('should sort visible people by asset count (desc), then by name (asc, nulls last)', async () => {
|
|
|
|
|
const { status, body } = await request(app).get('/people').set('Authorization', `Bearer ${admin.accessToken}`);
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body.hasNextPage).toBe(false);
|
|
|
|
|
expect(body.total).toBe(7); // All persons
|
|
|
|
|
expect(body.hidden).toBe(1); // 'hidden_person'
|
|
|
|
|
|
|
|
|
|
const people = body.people as PersonResponseDto[];
|
|
|
|
|
|
|
|
|
|
expect(people.map((p) => p.id)).toEqual([
|
|
|
|
|
multipleAssetsPerson.id, // name: 'multiple_assets_person', count: 3
|
|
|
|
|
nameBobPerson.id, // name: 'Bob', count: 2
|
|
|
|
|
nameAlicePerson.id, // name: 'Alice', count: 1
|
|
|
|
|
nameCharliePerson.id, // name: 'Charlie', count: 1
|
|
|
|
|
visiblePerson.id, // name: 'visible_person', count: 1
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
expect(people.some((p) => p.id === hiddenPerson.id)).toBe(false);
|
|
|
|
|
});
|
|
|
|
|
|
2024-02-21 08:28:03 -05:00
|
|
|
it('should return only visible people', async () => {
|
2024-05-22 13:24:57 -04:00
|
|
|
const { status, body } = await request(app).get('/people').set('Authorization', `Bearer ${admin.accessToken}`);
|
2024-02-21 08:28:03 -05:00
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toEqual({
|
2024-07-25 21:59:28 +02:00
|
|
|
hasNextPage: false,
|
2025-06-13 16:18:44 -03:00
|
|
|
total: 7,
|
2024-02-21 23:03:45 +01:00
|
|
|
hidden: 1,
|
2024-03-05 00:11:54 +01:00
|
|
|
people: [
|
|
|
|
|
expect.objectContaining({ name: 'multiple_assets_person' }),
|
2025-06-13 16:18:44 -03:00
|
|
|
expect.objectContaining({ name: 'Bob' }),
|
|
|
|
|
expect.objectContaining({ name: 'Alice' }),
|
|
|
|
|
expect.objectContaining({ name: 'Charlie' }),
|
2024-03-05 00:11:54 +01:00
|
|
|
expect.objectContaining({ name: 'visible_person' }),
|
|
|
|
|
],
|
2024-02-21 08:28:03 -05:00
|
|
|
});
|
|
|
|
|
});
|
2024-07-25 21:59:28 +02:00
|
|
|
|
|
|
|
|
it('should support pagination', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.get('/people')
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
2025-06-13 16:18:44 -03:00
|
|
|
.query({ withHidden: true, page: 5, size: 1 });
|
2024-07-25 21:59:28 +02:00
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toEqual({
|
|
|
|
|
hasNextPage: true,
|
2025-06-13 16:18:44 -03:00
|
|
|
total: 7,
|
2024-07-25 21:59:28 +02:00
|
|
|
hidden: 1,
|
|
|
|
|
people: [expect.objectContaining({ name: 'visible_person' })],
|
|
|
|
|
});
|
|
|
|
|
});
|
2024-02-21 08:28:03 -05:00
|
|
|
});
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('GET /people/:id', () => {
|
2024-02-21 08:28:03 -05:00
|
|
|
it('should throw error if person with id does not exist', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.get(`/people/${uuidDto.notFound}`)
|
2024-02-21 08:28:03 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`);
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(400);
|
|
|
|
|
expect(body).toEqual(errorDto.badRequest());
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return person information', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.get(`/people/${visiblePerson.id}`)
|
2024-02-21 08:28:03 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`);
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toEqual(expect.objectContaining({ id: visiblePerson.id }));
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('GET /people/:id/statistics', () => {
|
2024-03-05 00:11:54 +01:00
|
|
|
it('should throw error if person with id does not exist', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.get(`/people/${uuidDto.notFound}/statistics`)
|
2024-03-05 00:11:54 +01:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`);
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(400);
|
|
|
|
|
expect(body).toEqual(errorDto.badRequest());
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should return the correct number of assets', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.get(`/people/${multipleAssetsPerson.id}/statistics`)
|
2024-03-05 00:11:54 +01:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`);
|
|
|
|
|
|
|
|
|
|
expect(status).toBe(200);
|
2025-06-13 16:18:44 -03:00
|
|
|
expect(body).toEqual(expect.objectContaining({ assets: 3 }));
|
2024-03-05 00:11:54 +01:00
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('POST /people', () => {
|
2024-03-07 15:34:57 -05:00
|
|
|
it('should create a person', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.post(`/people`)
|
2024-03-07 15:34:57 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({
|
|
|
|
|
name: 'New Person',
|
2024-07-30 01:52:04 +02:00
|
|
|
birthDate: '1990-01-01',
|
2025-02-07 10:06:58 -05:00
|
|
|
color: '#333',
|
2024-03-07 15:34:57 -05:00
|
|
|
});
|
|
|
|
|
expect(status).toBe(201);
|
|
|
|
|
expect(body).toMatchObject({
|
|
|
|
|
id: expect.any(String),
|
|
|
|
|
name: 'New Person',
|
2025-03-09 22:32:05 -04:00
|
|
|
birthDate: '1990-01-01',
|
2024-03-07 15:34:57 -05:00
|
|
|
});
|
|
|
|
|
});
|
2025-02-04 09:52:17 +01:00
|
|
|
|
|
|
|
|
it('should create a favorite person', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.post(`/people`)
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({
|
|
|
|
|
name: 'New Favorite Person',
|
|
|
|
|
isFavorite: true,
|
|
|
|
|
});
|
|
|
|
|
expect(status).toBe(201);
|
|
|
|
|
expect(body).toMatchObject({
|
|
|
|
|
id: expect.any(String),
|
|
|
|
|
name: 'New Favorite Person',
|
|
|
|
|
isFavorite: true,
|
|
|
|
|
});
|
|
|
|
|
});
|
2024-03-07 15:34:57 -05:00
|
|
|
});
|
|
|
|
|
|
2024-05-22 13:24:57 -04:00
|
|
|
describe('PUT /people/:id', () => {
|
2024-02-21 08:28:03 -05:00
|
|
|
it('should update a date of birth', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.put(`/people/${visiblePerson.id}`)
|
2024-02-21 08:28:03 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
2024-07-30 01:52:04 +02:00
|
|
|
.send({ birthDate: '1990-01-01' });
|
2024-02-21 08:28:03 -05:00
|
|
|
expect(status).toBe(200);
|
2025-03-09 22:32:05 -04:00
|
|
|
expect(body).toMatchObject({ birthDate: '1990-01-01' });
|
2024-02-21 08:28:03 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should clear a date of birth', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
2024-05-22 13:24:57 -04:00
|
|
|
.put(`/people/${visiblePerson.id}`)
|
2024-02-21 08:28:03 -05:00
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({ birthDate: null });
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toMatchObject({ birthDate: null });
|
|
|
|
|
});
|
2025-02-04 09:52:17 +01:00
|
|
|
|
2025-02-07 10:06:58 -05:00
|
|
|
it('should set a color', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.put(`/people/${visiblePerson.id}`)
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({ color: '#555' });
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toMatchObject({ color: '#555' });
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
it('should clear a color', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.put(`/people/${visiblePerson.id}`)
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({ color: null });
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body.color).toBeUndefined();
|
|
|
|
|
});
|
|
|
|
|
|
2025-02-04 09:52:17 +01:00
|
|
|
it('should mark a person as favorite', async () => {
|
|
|
|
|
const person = await utils.createPerson(admin.accessToken, {
|
|
|
|
|
name: 'visible_person',
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
expect(person.isFavorite).toBe(false);
|
|
|
|
|
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.put(`/people/${person.id}`)
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({ isFavorite: true });
|
|
|
|
|
expect(status).toBe(200);
|
|
|
|
|
expect(body).toMatchObject({ isFavorite: true });
|
|
|
|
|
|
|
|
|
|
const person2 = await getPerson({ id: person.id }, { headers: asBearerAuth(admin.accessToken) });
|
|
|
|
|
expect(person2).toMatchObject({ id: person.id, isFavorite: true });
|
|
|
|
|
});
|
2024-02-21 08:28:03 -05:00
|
|
|
});
|
2024-07-02 15:56:05 -04:00
|
|
|
|
|
|
|
|
describe('POST /people/:id/merge', () => {
|
|
|
|
|
it('should not supporting merging a person into themselves', async () => {
|
|
|
|
|
const { status, body } = await request(app)
|
|
|
|
|
.post(`/people/${visiblePerson.id}/merge`)
|
|
|
|
|
.set('Authorization', `Bearer ${admin.accessToken}`)
|
|
|
|
|
.send({ ids: [visiblePerson.id] });
|
|
|
|
|
expect(status).toBe(400);
|
|
|
|
|
expect(body).toEqual(errorDto.badRequest('Cannot merge a person into themselves'));
|
|
|
|
|
});
|
|
|
|
|
});
|
2024-02-21 08:28:03 -05:00
|
|
|
});
|