| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  | <?php | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace BookStack\Uploads; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | use BookStack\Exceptions\ImageUploadException; | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | use Exception; | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  | use GuzzleHttp\Psr7\Utils; | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | use Illuminate\Support\Facades\Cache; | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  | use Illuminate\Support\Facades\Log; | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  | use Intervention\Image\Decoders\BinaryImageDecoder; | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  | use Intervention\Image\Drivers\Gd\Decoders\NativeObjectDecoder; | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  | use Intervention\Image\Drivers\Gd\Driver; | 
					
						
							|  |  |  | use Intervention\Image\Encoders\AutoEncoder; | 
					
						
							|  |  |  | use Intervention\Image\Encoders\PngEncoder; | 
					
						
							|  |  |  | use Intervention\Image\Interfaces\ImageInterface as InterventionImage; | 
					
						
							|  |  |  | use Intervention\Image\ImageManager; | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  | use Intervention\Image\Origin; | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | class ImageResizer | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |     protected const THUMBNAIL_CACHE_TIME = 604_800; // 1 week
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |     public function __construct( | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |         protected ImageStorage $storage, | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |     ) { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Load gallery thumbnails for a set of images. | 
					
						
							|  |  |  |      * @param iterable<Image> $images | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function loadGalleryThumbnailsForMany(iterable $images, bool $shouldCreate = false): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         foreach ($images as $image) { | 
					
						
							|  |  |  |             $this->loadGalleryThumbnailsForImage($image, $shouldCreate); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Load gallery thumbnails into the given image instance. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function loadGalleryThumbnailsForImage(Image $image, bool $shouldCreate): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $thumbs = ['gallery' => null, 'display' => null]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try { | 
					
						
							|  |  |  |             $thumbs['gallery'] = $this->resizeToThumbnailUrl($image, 150, 150, false, $shouldCreate); | 
					
						
							|  |  |  |             $thumbs['display'] = $this->resizeToThumbnailUrl($image, 1680, null, true, $shouldCreate); | 
					
						
							|  |  |  |         } catch (Exception $exception) { | 
					
						
							|  |  |  |             // Prevent thumbnail errors from stopping execution
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $image->setAttribute('thumbs', $thumbs); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Get the thumbnail for an image. | 
					
						
							|  |  |  |      * If $keepRatio is true only the width will be used. | 
					
						
							|  |  |  |      * Checks the cache then storage to avoid creating / accessing the filesystem on every check. | 
					
						
							|  |  |  |      * | 
					
						
							|  |  |  |      * @throws Exception | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     public function resizeToThumbnailUrl( | 
					
						
							|  |  |  |         Image $image, | 
					
						
							|  |  |  |         ?int $width, | 
					
						
							|  |  |  |         ?int $height, | 
					
						
							|  |  |  |         bool $keepRatio = false, | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |         bool $shouldCreate = false | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     ): ?string { | 
					
						
							|  |  |  |         // Do not resize GIF images where we're not cropping
 | 
					
						
							|  |  |  |         if ($keepRatio && $this->isGif($image)) { | 
					
						
							|  |  |  |             return $this->storage->getPublicUrl($image->path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $thumbDirName = '/' . ($keepRatio ? 'scaled-' : 'thumbs-') . $width . '-' . $height . '/'; | 
					
						
							|  |  |  |         $imagePath = $image->path; | 
					
						
							|  |  |  |         $thumbFilePath = dirname($imagePath) . $thumbDirName . basename($imagePath); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $thumbCacheKey = 'images::' . $image->id . '::' . $thumbFilePath; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Return path if in cache
 | 
					
						
							|  |  |  |         $cachedThumbPath = Cache::get($thumbCacheKey); | 
					
						
							|  |  |  |         if ($cachedThumbPath && !$shouldCreate) { | 
					
						
							|  |  |  |             return $this->storage->getPublicUrl($cachedThumbPath); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If thumbnail has already been generated, serve that and cache path
 | 
					
						
							|  |  |  |         $disk = $this->storage->getDisk($image->type); | 
					
						
							|  |  |  |         if (!$shouldCreate && $disk->exists($thumbFilePath)) { | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |             Cache::put($thumbCacheKey, $thumbFilePath, static::THUMBNAIL_CACHE_TIME); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return $this->storage->getPublicUrl($thumbFilePath); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $imageData = $disk->get($imagePath); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  |         // Do not resize animated images where we're not cropping
 | 
					
						
							|  |  |  |         if ($keepRatio && $this->isAnimated($image, $imageData)) { | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |             Cache::put($thumbCacheKey, $image->path, static::THUMBNAIL_CACHE_TIME); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return $this->storage->getPublicUrl($image->path); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // If not in cache and thumbnail does not exist, generate thumb and cache path
 | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  |         $thumbData = $this->resizeImageData($imageData, $width, $height, $keepRatio, $this->getExtension($image)); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |         $disk->put($thumbFilePath, $thumbData, true); | 
					
						
							| 
									
										
										
										
											2023-10-01 20:05:18 +08:00
										 |  |  |         Cache::put($thumbCacheKey, $thumbFilePath, static::THUMBNAIL_CACHE_TIME); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return $this->storage->getPublicUrl($thumbFilePath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Resize the image of given data to the specified size, and return the new image data. | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |      * Format will remain the same as the input format, unless specified. | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |      * | 
					
						
							|  |  |  |      * @throws ImageUploadException | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2023-11-19 23:57:19 +08:00
										 |  |  |     public function resizeImageData( | 
					
						
							|  |  |  |         string $imageData, | 
					
						
							|  |  |  |         ?int $width, | 
					
						
							|  |  |  |         ?int $height, | 
					
						
							|  |  |  |         bool $keepRatio, | 
					
						
							|  |  |  |         ?string $format = null, | 
					
						
							|  |  |  |     ): string { | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |         try { | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  |             $thumb = $this->interventionFromImageData($imageData, $format); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |         } catch (Exception $e) { | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |             throw new ImageUploadException(trans('errors.cannot_create_thumbs')); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $this->orientImageToOriginalExif($thumb, $imageData); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($keepRatio) { | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  |             $thumb->scaleDown($width, $height); | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  |             $thumb->cover($width, $height); | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  |         $encoder = match ($format) { | 
					
						
							|  |  |  |             'png' => new PngEncoder(), | 
					
						
							|  |  |  |             default => new AutoEncoder(), | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $thumbData = (string) $thumb->encode($encoder); | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Use original image data if we're keeping the ratio
 | 
					
						
							|  |  |  |         // and the resizing does not save any space.
 | 
					
						
							|  |  |  |         if ($keepRatio && strlen($thumbData) > strlen($imageData)) { | 
					
						
							|  |  |  |             return $imageData; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $thumbData; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-20 00:34:29 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Create an intervention image instance from the given image data. | 
					
						
							|  |  |  |      * Performs some manual library usage to ensure image is specifically loaded | 
					
						
							|  |  |  |      * from given binary data instead of data being misinterpreted. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  |     protected function interventionFromImageData(string $imageData, ?string $fileType): InterventionImage | 
					
						
							| 
									
										
										
										
											2023-11-20 00:34:29 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-02-01 05:29:38 +08:00
										 |  |  |         $manager = new ImageManager( | 
					
						
							|  |  |  |             new Driver(), | 
					
						
							|  |  |  |             autoOrientation: false, | 
					
						
							|  |  |  |         ); | 
					
						
							| 
									
										
										
										
											2024-03-18 00:03:12 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  |         // Ensure gif images are decoded natively instead of deferring to intervention GIF
 | 
					
						
							|  |  |  |         // handling since we don't need the added animation support.
 | 
					
						
							|  |  |  |         $isGif = $fileType === 'gif'; | 
					
						
							|  |  |  |         $decoder = $isGif ? NativeObjectDecoder::class : BinaryImageDecoder::class; | 
					
						
							|  |  |  |         $input = $isGif ? @imagecreatefromstring($imageData) : $imageData; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $image = $manager->read($input, $decoder); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($isGif) { | 
					
						
							|  |  |  |             $image->setOrigin(new Origin('image/gif')); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return $image; | 
					
						
							| 
									
										
										
										
											2023-11-20 00:34:29 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  |     /** | 
					
						
							|  |  |  |      * Orientate the given intervention image based upon the given original image data. | 
					
						
							|  |  |  |      * Intervention does have an `orientate` method but the exif data it needs is lost before it | 
					
						
							|  |  |  |      * can be used (At least when created using binary string data) so we need to do some | 
					
						
							|  |  |  |      * implementation on our side to use the original image data. | 
					
						
							|  |  |  |      * Bulk of logic taken from: https://github.com/Intervention/image/blob/b734a4988b2148e7d10364b0609978a88d277536/src/Intervention/Image/Commands/OrientateCommand.php | 
					
						
							|  |  |  |      * Copyright (c) Oliver Vogel, MIT License. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function orientImageToOriginalExif(InterventionImage $image, string $originalData): void | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         if (!extension_loaded('exif')) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         $stream = Utils::streamFor($originalData)->detach(); | 
					
						
							|  |  |  |         $exif = @exif_read_data($stream); | 
					
						
							|  |  |  |         $orientation = $exif ? ($exif['Orientation'] ?? null) : null; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         switch ($orientation) { | 
					
						
							|  |  |  |             case 2: | 
					
						
							|  |  |  |                 $image->flip(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 3: | 
					
						
							|  |  |  |                 $image->rotate(180); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 4: | 
					
						
							|  |  |  |                 $image->rotate(180)->flip(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 5: | 
					
						
							|  |  |  |                 $image->rotate(270)->flip(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 6: | 
					
						
							|  |  |  |                 $image->rotate(270); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 7: | 
					
						
							|  |  |  |                 $image->rotate(90)->flip(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             case 8: | 
					
						
							|  |  |  |                 $image->rotate(90); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Checks if the image is a gif. Returns true if it is, else false. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isGif(Image $image): bool | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2024-06-09 23:58:23 +08:00
										 |  |  |         return $this->getExtension($image) === 'gif'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Get the extension for the given image, normalised to lower-case. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function getExtension(Image $image): string | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         return strtolower(pathinfo($image->path, PATHINFO_EXTENSION)); | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the given image and image data is apng. | 
					
						
							|  |  |  |      */ | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  |     protected function isApngData(string &$imageData): bool | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  |         $initialHeader = substr($imageData, 0, strpos($imageData, 'IDAT')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return str_contains($initialHeader, 'acTL'); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the given avif image data represents an animated image. | 
					
						
							|  |  |  |      * This is based up the answer here: https://stackoverflow.com/a/79457313 | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isAnimatedAvifData(string &$imageData): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $stszPos = strpos($imageData, 'stsz'); | 
					
						
							|  |  |  |         if ($stszPos === false) { | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  |         // Look 12 bytes after the start of 'stsz'
 | 
					
						
							|  |  |  |         $start = $stszPos + 12; | 
					
						
							|  |  |  |         $end = $start + 4; | 
					
						
							|  |  |  |         if ($end > strlen($imageData) - 1) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-05-23 23:12:03 +08:00
										 |  |  |         $data = substr($imageData, $start, 4); | 
					
						
							|  |  |  |         $count = unpack('Nvalue', $data)['value']; | 
					
						
							|  |  |  |         return $count > 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /** | 
					
						
							|  |  |  |      * Check if the given image is animated. | 
					
						
							|  |  |  |      */ | 
					
						
							|  |  |  |     protected function isAnimated(Image $image, string &$imageData): bool | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         $extension = strtolower(pathinfo($image->path, PATHINFO_EXTENSION)); | 
					
						
							|  |  |  |         if ($extension === 'png') { | 
					
						
							|  |  |  |             return $this->isApngData($imageData); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ($extension === 'avif') { | 
					
						
							|  |  |  |             return $this->isAnimatedAvifData($imageData); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2023-10-01 03:00:48 +08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-10-01 01:28:42 +08:00
										 |  |  | } |