mirror of
				https://github.com/immich-app/immich.git
				synced 2025-10-26 08:24:27 -04:00 
			
		
		
		
	fix(server): use preview image when generating person thumbnail from video (#10240)
This commit is contained in:
		
							parent
							
								
									c642150b85
								
							
						
					
					
						commit
						fb641c74be
					
				| @ -949,6 +949,32 @@ describe(PersonService.name, () => { | |||||||
|         }, |         }, | ||||||
|       ); |       ); | ||||||
|     }); |     }); | ||||||
|  | 
 | ||||||
|  |     it('should use preview path for videos', async () => { | ||||||
|  |       personMock.getById.mockResolvedValue({ ...personStub.primaryPerson, faceAssetId: faceStub.end.assetId }); | ||||||
|  |       personMock.getFaceByIdWithAssets.mockResolvedValue(faceStub.end); | ||||||
|  |       assetMock.getById.mockResolvedValue(assetStub.video); | ||||||
|  |       mediaMock.getImageDimensions.mockResolvedValue({ width: 2560, height: 1440 }); | ||||||
|  | 
 | ||||||
|  |       await sut.handleGeneratePersonThumbnail({ id: personStub.primaryPerson.id }); | ||||||
|  | 
 | ||||||
|  |       expect(mediaMock.generateThumbnail).toHaveBeenCalledWith( | ||||||
|  |         assetStub.video.previewPath, | ||||||
|  |         'upload/thumbs/admin_id/pe/rs/person-1.jpeg', | ||||||
|  |         { | ||||||
|  |           format: 'jpeg', | ||||||
|  |           size: 250, | ||||||
|  |           quality: 80, | ||||||
|  |           colorspace: Colorspace.P3, | ||||||
|  |           crop: { | ||||||
|  |             left: 1741, | ||||||
|  |             top: 851, | ||||||
|  |             width: 588, | ||||||
|  |             height: 588, | ||||||
|  |           }, | ||||||
|  |         }, | ||||||
|  |       ); | ||||||
|  |     }); | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   describe('mergePerson', () => { |   describe('mergePerson', () => { | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ import { | |||||||
|   mapFaces, |   mapFaces, | ||||||
|   mapPerson, |   mapPerson, | ||||||
| } from 'src/dtos/person.dto'; | } from 'src/dtos/person.dto'; | ||||||
|  | import { AssetEntity, AssetType } from 'src/entities/asset.entity'; | ||||||
| import { PersonPathType } from 'src/entities/move.entity'; | import { PersonPathType } from 'src/entities/move.entity'; | ||||||
| import { PersonEntity } from 'src/entities/person.entity'; | import { PersonEntity } from 'src/entities/person.entity'; | ||||||
| import { IAccessRepository } from 'src/interfaces/access.interface'; | import { IAccessRepository } from 'src/interfaces/access.interface'; | ||||||
| @ -39,7 +40,7 @@ import { | |||||||
|   QueueName, |   QueueName, | ||||||
| } from 'src/interfaces/job.interface'; | } from 'src/interfaces/job.interface'; | ||||||
| import { ILoggerRepository } from 'src/interfaces/logger.interface'; | import { ILoggerRepository } from 'src/interfaces/logger.interface'; | ||||||
| import { IMachineLearningRepository } from 'src/interfaces/machine-learning.interface'; | import { BoundingBox, IMachineLearningRepository } from 'src/interfaces/machine-learning.interface'; | ||||||
| import { CropOptions, IMediaRepository, ImageDimensions } from 'src/interfaces/media.interface'; | import { CropOptions, IMediaRepository, ImageDimensions } from 'src/interfaces/media.interface'; | ||||||
| import { IMoveRepository } from 'src/interfaces/move.interface'; | import { IMoveRepository } from 'src/interfaces/move.interface'; | ||||||
| import { IPersonRepository, UpdateFacesData } from 'src/interfaces/person.interface'; | import { IPersonRepository, UpdateFacesData } from 'src/interfaces/person.interface'; | ||||||
| @ -509,61 +510,30 @@ export class PersonService { | |||||||
|       boundingBoxX2: x2, |       boundingBoxX2: x2, | ||||||
|       boundingBoxY1: y1, |       boundingBoxY1: y1, | ||||||
|       boundingBoxY2: y2, |       boundingBoxY2: y2, | ||||||
|       imageWidth, |       imageWidth: oldWidth, | ||||||
|       imageHeight, |       imageHeight: oldHeight, | ||||||
|     } = face; |     } = face; | ||||||
| 
 | 
 | ||||||
|     const asset = await this.assetRepository.getById(assetId, { exifInfo: true }); |     const asset = await this.assetRepository.getById(assetId, { exifInfo: true }); | ||||||
|     if (!asset?.exifInfo?.exifImageHeight || !asset.exifInfo.exifImageWidth) { |     if (!asset) { | ||||||
|       this.logger.error(`Could not generate person thumbnail: asset ${assetId} dimensions are unknown`); |       this.logger.error(`Could not generate person thumbnail: asset ${assetId} does not exist`); | ||||||
|       return JobStatus.FAILED; |       return JobStatus.FAILED; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     this.logger.verbose(`Cropping face for person: ${person.id}`); |     const { width, height, inputPath } = await this.getInputDimensions(asset); | ||||||
|  | 
 | ||||||
|     const thumbnailPath = StorageCore.getPersonThumbnailPath(person); |     const thumbnailPath = StorageCore.getPersonThumbnailPath(person); | ||||||
|     this.storageCore.ensureFolders(thumbnailPath); |     this.storageCore.ensureFolders(thumbnailPath); | ||||||
| 
 | 
 | ||||||
|     const { width: exifWidth, height: exifHeight } = this.withOrientation(asset.exifInfo.orientation as Orientation, { |  | ||||||
|       width: asset.exifInfo.exifImageWidth, |  | ||||||
|       height: asset.exifInfo.exifImageHeight, |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     const widthScale = exifWidth / imageWidth; |  | ||||||
|     const heightScale = exifHeight / imageHeight; |  | ||||||
| 
 |  | ||||||
|     const halfWidth = (widthScale * (x2 - x1)) / 2; |  | ||||||
|     const halfHeight = (heightScale * (y2 - y1)) / 2; |  | ||||||
| 
 |  | ||||||
|     const middleX = Math.round(widthScale * x1 + halfWidth); |  | ||||||
|     const middleY = Math.round(heightScale * 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(exifWidth - 1, middleX + targetHalfSize) - middleX, |  | ||||||
|       Math.min(exifHeight - 1, middleY + targetHalfSize) - middleY, |  | ||||||
|     ); |  | ||||||
| 
 |  | ||||||
|     const cropOptions: CropOptions = { |  | ||||||
|       left: middleX - newHalfSize, |  | ||||||
|       top: middleY - newHalfSize, |  | ||||||
|       width: newHalfSize * 2, |  | ||||||
|       height: newHalfSize * 2, |  | ||||||
|     }; |  | ||||||
| 
 |  | ||||||
|     const thumbnailOptions = { |     const thumbnailOptions = { | ||||||
|       format: ImageFormat.JPEG, |       format: ImageFormat.JPEG, | ||||||
|       size: FACE_THUMBNAIL_SIZE, |       size: FACE_THUMBNAIL_SIZE, | ||||||
|       colorspace: image.colorspace, |       colorspace: image.colorspace, | ||||||
|       quality: image.quality, |       quality: image.quality, | ||||||
|       crop: cropOptions, |       crop: this.getCrop({ old: { width: oldWidth, height: oldHeight }, new: { width, height } }, { x1, y1, x2, y2 }), | ||||||
|     } as const; |     } as const; | ||||||
| 
 | 
 | ||||||
|     await this.mediaRepository.generateThumbnail(asset.originalPath, thumbnailPath, thumbnailOptions); |     await this.mediaRepository.generateThumbnail(inputPath, thumbnailPath, thumbnailOptions); | ||||||
|     await this.repository.update({ id: person.id, thumbnailPath }); |     await this.repository.update({ id: person.id, thumbnailPath }); | ||||||
| 
 | 
 | ||||||
|     return JobStatus.SUCCESS; |     return JobStatus.SUCCESS; | ||||||
| @ -631,6 +601,27 @@ export class PersonService { | |||||||
|     return person; |     return person; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  |   private async getInputDimensions(asset: AssetEntity): Promise<ImageDimensions & { inputPath: string }> { | ||||||
|  |     if (!asset.exifInfo?.exifImageHeight || !asset.exifInfo.exifImageWidth) { | ||||||
|  |       throw new Error(`Asset ${asset.id} dimensions are unknown`); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (!asset.previewPath) { | ||||||
|  |       throw new Error(`Asset ${asset.id} has no preview path`); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     if (asset.type === AssetType.IMAGE) { | ||||||
|  |       const { width, height } = this.withOrientation(asset.exifInfo.orientation as Orientation, { | ||||||
|  |         width: asset.exifInfo.exifImageWidth, | ||||||
|  |         height: asset.exifInfo.exifImageHeight, | ||||||
|  |       }); | ||||||
|  |       return { width, height, inputPath: asset.originalPath }; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     const { width, height } = await this.mediaRepository.getImageDimensions(asset.previewPath); | ||||||
|  |     return { width, height, inputPath: asset.previewPath }; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|   private withOrientation(orientation: Orientation, { width, height }: ImageDimensions): ImageDimensions { |   private withOrientation(orientation: Orientation, { width, height }: ImageDimensions): ImageDimensions { | ||||||
|     switch (orientation) { |     switch (orientation) { | ||||||
|       case Orientation.MirrorHorizontalRotate270CW: |       case Orientation.MirrorHorizontalRotate270CW: | ||||||
| @ -644,4 +635,33 @@ export class PersonService { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   private getCrop(dims: { old: ImageDimensions; new: ImageDimensions }, { x1, y1, x2, y2 }: BoundingBox): CropOptions { | ||||||
|  |     const widthScale = dims.new.width / dims.old.width; | ||||||
|  |     const heightScale = dims.new.height / dims.old.height; | ||||||
|  | 
 | ||||||
|  |     const halfWidth = (widthScale * (x2 - x1)) / 2; | ||||||
|  |     const halfHeight = (heightScale * (y2 - y1)) / 2; | ||||||
|  | 
 | ||||||
|  |     const middleX = Math.round(widthScale * x1 + halfWidth); | ||||||
|  |     const middleY = Math.round(heightScale * 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(dims.new.width - 1, middleX + targetHalfSize) - middleX, | ||||||
|  |       Math.min(dims.new.height - 1, middleY + targetHalfSize) - middleY, | ||||||
|  |     ); | ||||||
|  | 
 | ||||||
|  |     return { | ||||||
|  |       left: middleX - newHalfSize, | ||||||
|  |       top: middleY - newHalfSize, | ||||||
|  |       width: newHalfSize * 2, | ||||||
|  |       height: newHalfSize * 2, | ||||||
|  |     }; | ||||||
|  |   } | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										2
									
								
								server/test/fixtures/asset.stub.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								server/test/fixtures/asset.stub.ts
									
									
									
									
										vendored
									
									
								
							| @ -436,6 +436,8 @@ export const assetStub = { | |||||||
|     sidecarPath: null, |     sidecarPath: null, | ||||||
|     exifInfo: { |     exifInfo: { | ||||||
|       fileSizeInByte: 100_000, |       fileSizeInByte: 100_000, | ||||||
|  |       exifImageHeight: 2160, | ||||||
|  |       exifImageWidth: 3840, | ||||||
|     } as ExifEntity, |     } as ExifEntity, | ||||||
|     deletedAt: null, |     deletedAt: null, | ||||||
|     duplicateId: null, |     duplicateId: null, | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user