2023-09-27 22:46:46 +02:00
|
|
|
import { PersonEntity } from '@app/infra/entities';
|
2023-10-11 04:14:44 +02:00
|
|
|
import { PersonPathType } from '@app/infra/entities/move.entity';
|
2023-05-17 19:07:17 +02:00
|
|
|
import { BadRequestException, Inject, Injectable, Logger, NotFoundException } from '@nestjs/common';
|
2023-10-09 16:25:03 +02:00
|
|
|
import { AccessCore, Permission } from '../access';
|
|
|
|
import { AssetResponseDto, BulkIdErrorReason, BulkIdResponseDto, mapAsset } from '../asset';
|
2023-12-10 05:34:12 +01:00
|
|
|
import { AuthDto } from '../auth';
|
2023-07-10 19:56:45 +02:00
|
|
|
import { mimeTypes } from '../domain.constant';
|
2023-09-27 22:46:46 +02:00
|
|
|
import { usePagination } from '../domain.util';
|
2023-10-09 16:25:03 +02:00
|
|
|
import { IBaseJob, IEntityJob, JOBS_ASSET_PAGINATION_SIZE, JobName } from '../job';
|
|
|
|
import { FACE_THUMBNAIL_SIZE } from '../media';
|
|
|
|
import {
|
|
|
|
CropOptions,
|
|
|
|
IAccessRepository,
|
|
|
|
IAssetRepository,
|
|
|
|
IJobRepository,
|
|
|
|
IMachineLearningRepository,
|
|
|
|
IMediaRepository,
|
2023-10-11 04:14:44 +02:00
|
|
|
IMoveRepository,
|
2023-10-09 16:25:03 +02:00
|
|
|
IPersonRepository,
|
2023-12-08 17:15:46 +01:00
|
|
|
ISmartInfoRepository,
|
2023-10-09 16:25:03 +02:00
|
|
|
IStorageRepository,
|
|
|
|
ISystemConfigRepository,
|
|
|
|
ImmichReadStream,
|
|
|
|
UpdateFacesData,
|
|
|
|
WithoutProperty,
|
|
|
|
} from '../repositories';
|
2023-10-11 04:14:44 +02:00
|
|
|
import { StorageCore } from '../storage';
|
2023-10-09 16:25:03 +02:00
|
|
|
import { SystemConfigCore } from '../system-config';
|
2023-07-18 20:09:43 +02:00
|
|
|
import {
|
2023-12-05 16:43:15 +01:00
|
|
|
AssetFaceResponseDto,
|
|
|
|
AssetFaceUpdateDto,
|
|
|
|
FaceDto,
|
2023-07-18 20:09:43 +02:00
|
|
|
MergePersonDto,
|
|
|
|
PeopleResponseDto,
|
2023-07-23 05:00:43 +02:00
|
|
|
PeopleUpdateDto,
|
2023-07-18 20:09:43 +02:00
|
|
|
PersonResponseDto,
|
|
|
|
PersonSearchDto,
|
2023-10-24 17:53:49 +02:00
|
|
|
PersonStatisticsResponseDto,
|
2023-07-18 20:09:43 +02:00
|
|
|
PersonUpdateDto,
|
2023-12-05 16:43:15 +01:00
|
|
|
mapFaces,
|
2023-09-04 21:45:59 +02:00
|
|
|
mapPerson,
|
2023-07-18 20:09:43 +02:00
|
|
|
} from './person.dto';
|
2023-05-17 19:07:17 +02:00
|
|
|
|
|
|
|
@Injectable()
|
|
|
|
export class PersonService {
|
2023-09-18 23:22:44 +02:00
|
|
|
private access: AccessCore;
|
2023-09-18 06:05:35 +02:00
|
|
|
private configCore: SystemConfigCore;
|
2023-09-27 22:46:46 +02:00
|
|
|
private storageCore: StorageCore;
|
2023-05-17 19:07:17 +02:00
|
|
|
readonly logger = new Logger(PersonService.name);
|
|
|
|
|
|
|
|
constructor(
|
2023-09-27 22:46:46 +02:00
|
|
|
@Inject(IAccessRepository) accessRepository: IAccessRepository,
|
|
|
|
@Inject(IAssetRepository) private assetRepository: IAssetRepository,
|
|
|
|
@Inject(IMachineLearningRepository) private machineLearningRepository: IMachineLearningRepository,
|
2023-10-11 04:14:44 +02:00
|
|
|
@Inject(IMoveRepository) moveRepository: IMoveRepository,
|
2023-09-27 22:46:46 +02:00
|
|
|
@Inject(IMediaRepository) private mediaRepository: IMediaRepository,
|
2023-05-17 19:07:17 +02:00
|
|
|
@Inject(IPersonRepository) private repository: IPersonRepository,
|
2023-09-18 06:05:35 +02:00
|
|
|
@Inject(ISystemConfigRepository) configRepository: ISystemConfigRepository,
|
2023-05-17 19:07:17 +02:00
|
|
|
@Inject(IStorageRepository) private storageRepository: IStorageRepository,
|
|
|
|
@Inject(IJobRepository) private jobRepository: IJobRepository,
|
2023-12-08 17:15:46 +01:00
|
|
|
@Inject(ISmartInfoRepository) private smartInfoRepository: ISmartInfoRepository,
|
2023-09-18 06:05:35 +02:00
|
|
|
) {
|
2023-10-23 14:37:51 +02:00
|
|
|
this.access = AccessCore.create(accessRepository);
|
2023-10-09 02:51:03 +02:00
|
|
|
this.configCore = SystemConfigCore.create(configRepository);
|
2023-10-23 17:52:21 +02:00
|
|
|
this.storageCore = StorageCore.create(assetRepository, moveRepository, repository, storageRepository);
|
2023-09-18 06:05:35 +02:00
|
|
|
}
|
2023-05-17 19:07:17 +02:00
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getAll(auth: AuthDto, dto: PersonSearchDto): Promise<PeopleResponseDto> {
|
2023-09-18 06:05:35 +02:00
|
|
|
const { machineLearning } = await this.configCore.getConfig();
|
2023-12-10 05:34:12 +01:00
|
|
|
const people = await this.repository.getAllForUser(auth.user.id, {
|
2023-09-18 06:05:35 +02:00
|
|
|
minimumFaceCount: machineLearning.facialRecognition.minFaces,
|
2023-08-16 02:06:49 +02:00
|
|
|
withHidden: dto.withHidden || false,
|
|
|
|
});
|
2023-08-14 18:09:26 +02:00
|
|
|
const persons: PersonResponseDto[] = people
|
2023-07-18 20:09:43 +02:00
|
|
|
// with thumbnails
|
|
|
|
.filter((person) => !!person.thumbnailPath)
|
|
|
|
.map((person) => mapPerson(person));
|
|
|
|
|
|
|
|
return {
|
|
|
|
people: persons.filter((person) => dto.withHidden || !person.isHidden),
|
|
|
|
total: persons.length,
|
|
|
|
visible: persons.filter((person: PersonResponseDto) => !person.isHidden).length,
|
|
|
|
};
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
createPerson(auth: AuthDto): Promise<PersonResponseDto> {
|
|
|
|
return this.repository.create({ ownerId: auth.user.id });
|
2023-12-05 16:43:15 +01:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async reassignFaces(auth: AuthDto, personId: string, dto: AssetFaceUpdateDto): Promise<PersonResponseDto[]> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_WRITE, personId);
|
2023-12-05 16:43:15 +01:00
|
|
|
const person = await this.findOrFail(personId);
|
|
|
|
const result: PersonResponseDto[] = [];
|
|
|
|
const changeFeaturePhoto: string[] = [];
|
|
|
|
for (const data of dto.data) {
|
|
|
|
const faces = await this.repository.getFacesByIds([{ personId: data.personId, assetId: data.assetId }]);
|
|
|
|
|
|
|
|
for (const face of faces) {
|
2023-12-10 05:34:12 +01:00
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_CREATE, face.id);
|
2023-12-05 16:43:15 +01:00
|
|
|
if (person.faceAssetId === null) {
|
|
|
|
changeFeaturePhoto.push(person.id);
|
|
|
|
}
|
|
|
|
if (face.person && face.person.faceAssetId === face.id) {
|
|
|
|
changeFeaturePhoto.push(face.person.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
await this.repository.reassignFace(face.id, personId);
|
|
|
|
}
|
|
|
|
|
|
|
|
result.push(person);
|
|
|
|
}
|
|
|
|
if (changeFeaturePhoto.length > 0) {
|
|
|
|
// Remove duplicates
|
|
|
|
await this.createNewFeaturePhoto(Array.from(new Set(changeFeaturePhoto)));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async reassignFacesById(auth: AuthDto, personId: string, dto: FaceDto): Promise<PersonResponseDto> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_WRITE, personId);
|
2023-12-05 16:43:15 +01:00
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_CREATE, dto.id);
|
2023-12-05 16:43:15 +01:00
|
|
|
const face = await this.repository.getFaceById(dto.id);
|
|
|
|
const person = await this.findOrFail(personId);
|
|
|
|
|
|
|
|
await this.repository.reassignFace(face.id, personId);
|
|
|
|
if (person.faceAssetId === null) {
|
|
|
|
await this.createNewFeaturePhoto([person.id]);
|
|
|
|
}
|
|
|
|
if (face.person && face.person.faceAssetId === face.id) {
|
|
|
|
await this.createNewFeaturePhoto([face.person.id]);
|
|
|
|
}
|
|
|
|
|
|
|
|
return await this.findOrFail(personId).then(mapPerson);
|
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getFacesById(auth: AuthDto, dto: FaceDto): Promise<AssetFaceResponseDto[]> {
|
|
|
|
await this.access.requirePermission(auth, Permission.ASSET_READ, dto.id);
|
2023-12-05 16:43:15 +01:00
|
|
|
const faces = await this.repository.getFaces(dto.id);
|
2023-12-10 05:34:12 +01:00
|
|
|
return faces.map((asset) => mapFaces(asset, auth));
|
2023-12-05 16:43:15 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
async createNewFeaturePhoto(changeFeaturePhoto: string[]) {
|
|
|
|
this.logger.debug(
|
|
|
|
`Changing feature photos for ${changeFeaturePhoto.length} ${changeFeaturePhoto.length > 1 ? 'people' : 'person'}`,
|
|
|
|
);
|
|
|
|
for (const personId of changeFeaturePhoto) {
|
|
|
|
const assetFace = await this.repository.getRandomFace(personId);
|
|
|
|
|
|
|
|
if (assetFace !== null) {
|
|
|
|
await this.repository.update({
|
|
|
|
id: personId,
|
|
|
|
faceAssetId: assetFace.id,
|
|
|
|
});
|
|
|
|
|
|
|
|
await this.jobRepository.queue({
|
|
|
|
name: JobName.GENERATE_PERSON_THUMBNAIL,
|
|
|
|
data: {
|
|
|
|
id: personId,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getById(auth: AuthDto, id: string): Promise<PersonResponseDto> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_READ, id);
|
2023-09-18 23:22:44 +02:00
|
|
|
return this.findOrFail(id).then(mapPerson);
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getStatistics(auth: AuthDto, id: string): Promise<PersonStatisticsResponseDto> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_READ, id);
|
2023-10-24 17:53:49 +02:00
|
|
|
return this.repository.getStatistics(id);
|
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getThumbnail(auth: AuthDto, id: string): Promise<ImmichReadStream> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_READ, id);
|
2023-09-18 23:22:44 +02:00
|
|
|
const person = await this.repository.getById(id);
|
2023-05-17 19:07:17 +02:00
|
|
|
if (!person || !person.thumbnailPath) {
|
|
|
|
throw new NotFoundException();
|
|
|
|
}
|
|
|
|
|
2023-07-10 19:56:45 +02:00
|
|
|
return this.storageRepository.createReadStream(person.thumbnailPath, mimeTypes.lookup(person.thumbnailPath));
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async getAssets(auth: AuthDto, id: string): Promise<AssetResponseDto[]> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_READ, id);
|
2023-09-18 23:22:44 +02:00
|
|
|
const assets = await this.repository.getAssets(id);
|
2023-10-14 03:46:30 +02:00
|
|
|
return assets.map((asset) => mapAsset(asset));
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async update(auth: AuthDto, id: string, dto: PersonUpdateDto): Promise<PersonResponseDto> {
|
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_WRITE, id);
|
2023-09-18 23:22:44 +02:00
|
|
|
let person = await this.findOrFail(id);
|
2023-05-17 19:07:17 +02:00
|
|
|
|
2023-09-26 09:03:22 +02:00
|
|
|
const { name, birthDate, isHidden, featureFaceAssetId: assetId } = dto;
|
|
|
|
|
|
|
|
if (name !== undefined || birthDate !== undefined || isHidden !== undefined) {
|
|
|
|
person = await this.repository.update({ id, name, birthDate, isHidden });
|
2023-07-03 00:46:20 +02:00
|
|
|
}
|
|
|
|
|
2023-09-26 09:03:22 +02:00
|
|
|
if (assetId) {
|
2023-12-10 05:34:12 +01:00
|
|
|
await this.access.requirePermission(auth, Permission.ASSET_READ, assetId);
|
2023-09-27 22:46:46 +02:00
|
|
|
const [face] = await this.repository.getFacesByIds([{ personId: id, assetId }]);
|
2023-07-11 23:52:41 +02:00
|
|
|
if (!face) {
|
|
|
|
throw new BadRequestException('Invalid assetId for feature face');
|
|
|
|
}
|
2023-07-03 00:46:20 +02:00
|
|
|
|
2023-12-05 16:43:15 +01:00
|
|
|
person = await this.repository.update({ id, faceAssetId: face.id });
|
2023-09-26 09:03:22 +02:00
|
|
|
await this.jobRepository.queue({ name: JobName.GENERATE_PERSON_THUMBNAIL, data: { id } });
|
2023-07-03 00:46:20 +02:00
|
|
|
}
|
|
|
|
|
2023-07-11 23:52:41 +02:00
|
|
|
return mapPerson(person);
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async updatePeople(auth: AuthDto, dto: PeopleUpdateDto): Promise<BulkIdResponseDto[]> {
|
2023-07-23 05:00:43 +02:00
|
|
|
const results: BulkIdResponseDto[] = [];
|
|
|
|
for (const person of dto.people) {
|
|
|
|
try {
|
2023-12-10 05:34:12 +01:00
|
|
|
await this.update(auth, person.id, {
|
2023-07-23 05:00:43 +02:00
|
|
|
isHidden: person.isHidden,
|
|
|
|
name: person.name,
|
2023-08-18 22:10:29 +02:00
|
|
|
birthDate: person.birthDate,
|
2023-07-23 05:00:43 +02:00
|
|
|
featureFaceAssetId: person.featureFaceAssetId,
|
|
|
|
}),
|
|
|
|
results.push({ id: person.id, success: true });
|
|
|
|
} catch (error: Error | any) {
|
|
|
|
this.logger.error(`Unable to update ${person.id} : ${error}`, error?.stack);
|
|
|
|
results.push({ id: person.id, success: false, error: BulkIdErrorReason.UNKNOWN });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2023-10-03 03:15:11 +02:00
|
|
|
async handlePersonDelete({ id }: IEntityJob) {
|
|
|
|
const person = await this.repository.getById(id);
|
|
|
|
if (!person) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
await this.repository.delete(person);
|
|
|
|
await this.storageRepository.unlink(person.thumbnailPath);
|
|
|
|
} catch (error: Error | any) {
|
|
|
|
this.logger.error(`Unable to delete person: ${error}`, error?.stack);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-26 21:43:24 +02:00
|
|
|
async handlePersonCleanup() {
|
2023-05-17 19:07:17 +02:00
|
|
|
const people = await this.repository.getAllWithoutFaces();
|
|
|
|
for (const person of people) {
|
|
|
|
this.logger.debug(`Person ${person.name || person.id} no longer has any faces, deleting.`);
|
2023-10-03 03:15:11 +02:00
|
|
|
await this.jobRepository.queue({ name: JobName.PERSON_DELETE, data: { id: person.id } });
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
2023-05-26 21:43:24 +02:00
|
|
|
|
|
|
|
return true;
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|
2023-07-11 23:52:41 +02:00
|
|
|
|
2023-09-27 22:46:46 +02:00
|
|
|
async handleQueueRecognizeFaces({ force }: IBaseJob) {
|
|
|
|
const { machineLearning } = await this.configCore.getConfig();
|
|
|
|
if (!machineLearning.enabled || !machineLearning.facialRecognition.enabled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const assetPagination = usePagination(JOBS_ASSET_PAGINATION_SIZE, (pagination) => {
|
|
|
|
return force
|
|
|
|
? this.assetRepository.getAll(pagination, { order: 'DESC' })
|
|
|
|
: this.assetRepository.getWithout(pagination, WithoutProperty.FACES);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (force) {
|
2023-10-03 03:15:11 +02:00
|
|
|
const people = await this.repository.getAll();
|
|
|
|
for (const person of people) {
|
|
|
|
await this.jobRepository.queue({ name: JobName.PERSON_DELETE, data: { id: person.id } });
|
|
|
|
}
|
2023-12-08 17:15:46 +01:00
|
|
|
this.logger.debug(`Deleted ${people.length} people`);
|
2023-09-27 22:46:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for await (const assets of assetPagination) {
|
|
|
|
for (const asset of assets) {
|
|
|
|
await this.jobRepository.queue({ name: JobName.RECOGNIZE_FACES, data: { id: asset.id } });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
async handleRecognizeFaces({ id }: IEntityJob) {
|
|
|
|
const { machineLearning } = await this.configCore.getConfig();
|
|
|
|
if (!machineLearning.enabled || !machineLearning.facialRecognition.enabled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-11-06 03:15:12 +01:00
|
|
|
const relations = {
|
|
|
|
exifInfo: true,
|
|
|
|
faces: {
|
|
|
|
person: true,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
const [asset] = await this.assetRepository.getByIds([id], relations);
|
2023-11-05 17:07:29 +01:00
|
|
|
if (!asset || !asset.resizePath || asset.faces?.length > 0) {
|
2023-09-27 22:46:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const faces = await this.machineLearningRepository.detectFaces(
|
|
|
|
machineLearning.url,
|
|
|
|
{ imagePath: asset.resizePath },
|
|
|
|
machineLearning.facialRecognition,
|
|
|
|
);
|
|
|
|
|
|
|
|
this.logger.debug(`${faces.length} faces detected in ${asset.resizePath}`);
|
2023-12-08 17:15:46 +01:00
|
|
|
this.logger.verbose(faces.map((face) => ({ ...face, embedding: `vector(${face.embedding.length})` })));
|
2023-09-27 22:46:46 +02:00
|
|
|
|
|
|
|
for (const { embedding, ...rest } of faces) {
|
2023-12-08 17:15:46 +01:00
|
|
|
const matches = await this.smartInfoRepository.searchFaces({
|
|
|
|
ownerId: asset.ownerId,
|
|
|
|
embedding,
|
|
|
|
numResults: 1,
|
|
|
|
maxDistance: machineLearning.facialRecognition.maxDistance,
|
|
|
|
});
|
2023-09-27 22:46:46 +02:00
|
|
|
|
2023-12-08 17:15:46 +01:00
|
|
|
let personId = matches[0]?.personId || null;
|
2023-09-27 22:46:46 +02:00
|
|
|
let newPerson: PersonEntity | null = null;
|
|
|
|
if (!personId) {
|
|
|
|
this.logger.debug('No matches, creating a new person.');
|
|
|
|
newPerson = await this.repository.create({ ownerId: asset.ownerId });
|
|
|
|
personId = newPerson.id;
|
|
|
|
}
|
|
|
|
|
2023-12-05 16:43:15 +01:00
|
|
|
const face = await this.repository.createFace({
|
|
|
|
assetId: asset.id,
|
|
|
|
personId,
|
2023-09-27 22:46:46 +02:00
|
|
|
embedding,
|
|
|
|
imageHeight: rest.imageHeight,
|
|
|
|
imageWidth: rest.imageWidth,
|
|
|
|
boundingBoxX1: rest.boundingBox.x1,
|
|
|
|
boundingBoxX2: rest.boundingBox.x2,
|
|
|
|
boundingBoxY1: rest.boundingBox.y1,
|
|
|
|
boundingBoxY2: rest.boundingBox.y2,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (newPerson) {
|
2023-12-05 16:43:15 +01:00
|
|
|
await this.repository.update({ id: personId, faceAssetId: face.id });
|
2023-09-27 22:46:46 +02:00
|
|
|
await this.jobRepository.queue({ name: JobName.GENERATE_PERSON_THUMBNAIL, data: { id: newPerson.id } });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-10 02:55:00 +01:00
|
|
|
await this.assetRepository.upsertJobStatus({
|
|
|
|
assetId: asset.id,
|
|
|
|
facesRecognizedAt: new Date(),
|
|
|
|
});
|
|
|
|
|
2023-09-27 22:46:46 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
async handlePersonMigration({ id }: IEntityJob) {
|
|
|
|
const person = await this.repository.getById(id);
|
|
|
|
if (!person) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-10-11 04:14:44 +02:00
|
|
|
await this.storageCore.movePersonFile(person, PersonPathType.FACE);
|
2023-09-27 22:46:46 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
async handleGeneratePersonThumbnail(data: IEntityJob) {
|
|
|
|
const { machineLearning, thumbnail } = await this.configCore.getConfig();
|
|
|
|
if (!machineLearning.enabled || !machineLearning.facialRecognition.enabled) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
const person = await this.repository.getById(data.id);
|
|
|
|
if (!person?.faceAssetId) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2023-12-05 16:43:15 +01:00
|
|
|
const face = await this.repository.getFaceByIdWithAssets(person.faceAssetId);
|
|
|
|
if (face === null) {
|
2023-09-27 22:46:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const {
|
|
|
|
assetId,
|
|
|
|
boundingBoxX1: x1,
|
|
|
|
boundingBoxX2: x2,
|
|
|
|
boundingBoxY1: y1,
|
|
|
|
boundingBoxY2: y2,
|
|
|
|
imageWidth,
|
|
|
|
imageHeight,
|
|
|
|
} = face;
|
|
|
|
|
|
|
|
const [asset] = await this.assetRepository.getByIds([assetId]);
|
|
|
|
if (!asset?.resizePath) {
|
|
|
|
return false;
|
|
|
|
}
|
2023-12-05 16:43:15 +01:00
|
|
|
this.logger.verbose(`Cropping face for person: ${person.id}`);
|
2023-10-23 17:52:21 +02:00
|
|
|
const thumbnailPath = StorageCore.getPersonThumbnailPath(person);
|
2023-10-11 04:14:44 +02:00
|
|
|
this.storageCore.ensureFolders(thumbnailPath);
|
2023-09-27 22:46:46 +02:00
|
|
|
|
|
|
|
const halfWidth = (x2 - x1) / 2;
|
|
|
|
const halfHeight = (y2 - y1) / 2;
|
|
|
|
|
|
|
|
const middleX = Math.round(x1 + halfWidth);
|
|
|
|
const middleY = Math.round(y1 + halfHeight);
|
|
|
|
|
|
|
|
// zoom out 10%
|
|
|
|
const targetHalfSize = Math.floor(Math.max(halfWidth, halfHeight) * 1.1);
|
|
|
|
|
|
|
|
// get the longest distance from the center of the image without overflowing
|
|
|
|
const newHalfSize = Math.min(
|
|
|
|
middleX - Math.max(0, middleX - targetHalfSize),
|
|
|
|
middleY - Math.max(0, middleY - targetHalfSize),
|
|
|
|
Math.min(imageWidth - 1, middleX + targetHalfSize) - middleX,
|
|
|
|
Math.min(imageHeight - 1, middleY + targetHalfSize) - middleY,
|
|
|
|
);
|
|
|
|
|
|
|
|
const cropOptions: CropOptions = {
|
|
|
|
left: middleX - newHalfSize,
|
|
|
|
top: middleY - newHalfSize,
|
|
|
|
width: newHalfSize * 2,
|
|
|
|
height: newHalfSize * 2,
|
|
|
|
};
|
|
|
|
|
|
|
|
const croppedOutput = await this.mediaRepository.crop(asset.resizePath, cropOptions);
|
|
|
|
const thumbnailOptions = {
|
|
|
|
format: 'jpeg',
|
|
|
|
size: FACE_THUMBNAIL_SIZE,
|
|
|
|
colorspace: thumbnail.colorspace,
|
|
|
|
quality: thumbnail.quality,
|
|
|
|
} as const;
|
|
|
|
|
|
|
|
await this.mediaRepository.resize(croppedOutput, thumbnailPath, thumbnailOptions);
|
2023-10-24 15:12:42 +02:00
|
|
|
await this.repository.update({ id: person.id, thumbnailPath });
|
2023-09-27 22:46:46 +02:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
async mergePerson(auth: AuthDto, id: string, dto: MergePersonDto): Promise<BulkIdResponseDto[]> {
|
2023-07-11 23:52:41 +02:00
|
|
|
const mergeIds = dto.ids;
|
2023-12-10 05:34:12 +01:00
|
|
|
await this.access.requirePermission(auth, Permission.PERSON_WRITE, id);
|
2023-09-18 23:22:44 +02:00
|
|
|
const primaryPerson = await this.findOrFail(id);
|
2023-07-11 23:52:41 +02:00
|
|
|
const primaryName = primaryPerson.name || primaryPerson.id;
|
|
|
|
|
|
|
|
const results: BulkIdResponseDto[] = [];
|
|
|
|
|
2023-12-10 05:34:12 +01:00
|
|
|
const allowedIds = await this.access.checkAccess(auth, Permission.PERSON_MERGE, mergeIds);
|
2023-09-18 23:22:44 +02:00
|
|
|
|
2023-11-23 05:04:52 +01:00
|
|
|
for (const mergeId of mergeIds) {
|
|
|
|
const hasAccess = allowedIds.has(mergeId);
|
|
|
|
if (!hasAccess) {
|
2023-09-18 23:22:44 +02:00
|
|
|
results.push({ id: mergeId, success: false, error: BulkIdErrorReason.NO_PERMISSION });
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2023-07-11 23:52:41 +02:00
|
|
|
try {
|
2023-09-18 23:22:44 +02:00
|
|
|
const mergePerson = await this.repository.getById(mergeId);
|
2023-07-11 23:52:41 +02:00
|
|
|
if (!mergePerson) {
|
|
|
|
results.push({ id: mergeId, success: false, error: BulkIdErrorReason.NOT_FOUND });
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
const mergeName = mergePerson.name || mergePerson.id;
|
|
|
|
const mergeData: UpdateFacesData = { oldPersonId: mergeId, newPersonId: id };
|
|
|
|
this.logger.log(`Merging ${mergeName} into ${primaryName}`);
|
|
|
|
|
|
|
|
await this.repository.reassignFaces(mergeData);
|
2023-10-03 03:15:11 +02:00
|
|
|
await this.jobRepository.queue({ name: JobName.PERSON_DELETE, data: { id: mergePerson.id } });
|
2023-07-11 23:52:41 +02:00
|
|
|
|
|
|
|
this.logger.log(`Merged ${mergeName} into ${primaryName}`);
|
|
|
|
results.push({ id: mergeId, success: true });
|
|
|
|
} catch (error: Error | any) {
|
|
|
|
this.logger.error(`Unable to merge ${mergeId} into ${id}: ${error}`, error?.stack);
|
|
|
|
results.push({ id: mergeId, success: false, error: BulkIdErrorReason.UNKNOWN });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return results;
|
|
|
|
}
|
|
|
|
|
2023-09-18 23:22:44 +02:00
|
|
|
private async findOrFail(id: string) {
|
|
|
|
const person = await this.repository.getById(id);
|
2023-07-11 23:52:41 +02:00
|
|
|
if (!person) {
|
|
|
|
throw new BadRequestException('Person not found');
|
|
|
|
}
|
|
|
|
return person;
|
|
|
|
}
|
2023-05-17 19:07:17 +02:00
|
|
|
}
|