vendor/contao/image/src/Resizer.php line 74

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. /*
  4.  * This file is part of Contao.
  5.  *
  6.  * (c) Leo Feyer
  7.  *
  8.  * @license LGPL-3.0-or-later
  9.  */
  10. namespace Contao\Image;
  11. use Contao\Image\Exception\InvalidArgumentException;
  12. use Contao\Image\Metadata\ImageMetadata;
  13. use Contao\Image\Metadata\MetadataReaderWriter;
  14. use Imagine\Exception\InvalidArgumentException as ImagineInvalidArgumentException;
  15. use Imagine\Exception\RuntimeException as ImagineRuntimeException;
  16. use Imagine\Filter\Basic\Autorotate;
  17. use Imagine\Image\Palette\RGB;
  18. use Symfony\Component\Filesystem\Filesystem;
  19. use Symfony\Component\Filesystem\Path;
  20. /**
  21.  * @method __construct(string $cacheDir, string $secret, ResizeCalculator $calculator = null, Filesystem $filesystem = null, MetadataReaderWriter $metadataReaderWriter = null)
  22.  */
  23. class Resizer implements ResizerInterface
  24. {
  25.     /**
  26.      * @var Filesystem
  27.      *
  28.      * @internal
  29.      */
  30.     protected $filesystem;
  31.     /**
  32.      * @var string
  33.      *
  34.      * @internal
  35.      */
  36.     protected $cacheDir;
  37.     /**
  38.      * @var ResizeCalculator
  39.      */
  40.     private $calculator;
  41.     /**
  42.      * @var MetadataReaderWriter
  43.      */
  44.     private $metadataReaderWriter;
  45.     /**
  46.      * @var string|null
  47.      */
  48.     private $secret;
  49.     /**
  50.      * @param string                    $cacheDir
  51.      * @param string                    $secret
  52.      * @param ResizeCalculator|null     $calculator
  53.      * @param Filesystem|null           $filesystem
  54.      * @param MetadataReaderWriter|null $metadataReaderWriter
  55.      */
  56.     public function __construct(string $cacheDir/*, string $secret, ResizeCalculator $calculator = null, Filesystem $filesystem = null, MetadataReaderWriter $metadataReaderWriter = null*/)
  57.     {
  58.         if (\func_num_args() > && \is_string(func_get_arg(1))) {
  59.             $secret func_get_arg(1);
  60.             $calculator \func_num_args() > func_get_arg(2) : null;
  61.             $filesystem \func_num_args() > func_get_arg(3) : null;
  62.             $metadataReaderWriter \func_num_args() > func_get_arg(4) : null;
  63.         } else {
  64.             trigger_deprecation('contao/image''1.2''Not passing a secret to "%s()" has been deprecated and will no longer work in version 2.0.'__METHOD__);
  65.             $secret null;
  66.             $calculator \func_num_args() > func_get_arg(1) : null;
  67.             $filesystem \func_num_args() > func_get_arg(2) : null;
  68.             $metadataReaderWriter \func_num_args() > func_get_arg(3) : null;
  69.         }
  70.         if (null === $calculator) {
  71.             $calculator = new ResizeCalculator();
  72.         }
  73.         if (null === $filesystem) {
  74.             $filesystem = new Filesystem();
  75.         }
  76.         if (null === $metadataReaderWriter) {
  77.             $metadataReaderWriter = new MetadataReaderWriter();
  78.         }
  79.         if (!$calculator instanceof ResizeCalculator) {
  80.             $type \is_object($calculator) ? \get_class($calculator) : \gettype($calculator);
  81.             throw new \TypeError(sprintf('%s(): Argument #3 ($calculator) must be of type ResizeCalculator|null, %s given'__METHOD__$type));
  82.         }
  83.         if (!$filesystem instanceof Filesystem) {
  84.             $type \is_object($filesystem) ? \get_class($filesystem) : \gettype($filesystem);
  85.             throw new \TypeError(sprintf('%s(): Argument #4 ($filesystem) must be of type ResizeCalculator|null, %s given'__METHOD__$type));
  86.         }
  87.         if (!$metadataReaderWriter instanceof MetadataReaderWriter) {
  88.             $type \is_object($metadataReaderWriter) ? \get_class($metadataReaderWriter) : \gettype($metadataReaderWriter);
  89.             throw new \TypeError(sprintf('%s(): Argument #5 ($metadataReaderWriter) must be of type MetadataReaderWriter|null, %s given'__METHOD__$type));
  90.         }
  91.         if ('' === $secret) {
  92.             throw new InvalidArgumentException('$secret must not be empty');
  93.         }
  94.         $this->cacheDir $cacheDir;
  95.         $this->calculator $calculator;
  96.         $this->filesystem $filesystem;
  97.         $this->metadataReaderWriter $metadataReaderWriter;
  98.         $this->secret $secret;
  99.     }
  100.     /**
  101.      * {@inheritdoc}
  102.      */
  103.     public function resize(ImageInterface $imageResizeConfiguration $configResizeOptions $options): ImageInterface
  104.     {
  105.         if (
  106.             $image->getDimensions()->isUndefined()
  107.             || ($config->isEmpty() && $this->canSkipResize($image$options))
  108.         ) {
  109.             $image $this->createImage($image$image->getPath());
  110.         } else {
  111.             $image $this->processResize($image$config$options);
  112.         }
  113.         if (null !== $options->getTargetPath()) {
  114.             $this->filesystem->copy($image->getPath(), $options->getTargetPath(), true);
  115.             $image $this->createImage($image$options->getTargetPath());
  116.         }
  117.         return $image;
  118.     }
  119.     /**
  120.      * Executes the resize operation via Imagine.
  121.      *
  122.      * @internal Do not call this method in your code; it will be made private in a future version
  123.      */
  124.     protected function executeResize(ImageInterface $imageResizeCoordinates $coordinatesstring $pathResizeOptions $options): ImageInterface
  125.     {
  126.         $dir \dirname($path);
  127.         if (!$this->filesystem->exists($dir)) {
  128.             $this->filesystem->mkdir($dir);
  129.         }
  130.         $imagineOptions $options->getImagineOptions();
  131.         $imagineImage $image->getImagine()->open($image->getPath());
  132.         if (ImageDimensions::ORIENTATION_NORMAL !== $image->getDimensions()->getOrientation()) {
  133.             (new Autorotate())->apply($imagineImage);
  134.         }
  135.         $imagineImage
  136.             ->resize($coordinates->getSize())
  137.             ->crop($coordinates->getCropStart(), $coordinates->getCropSize())
  138.             ->usePalette(new RGB())
  139.             ->strip()
  140.         ;
  141.         if (isset($imagineOptions['interlace'])) {
  142.             try {
  143.                 $imagineImage->interlace($imagineOptions['interlace']);
  144.             } catch (ImagineInvalidArgumentException|ImagineRuntimeException $e) {
  145.                 // Ignore failed interlacing
  146.             }
  147.         }
  148.         if (!isset($imagineOptions['format'])) {
  149.             $imagineOptions['format'] = strtolower(pathinfo($pathPATHINFO_EXTENSION));
  150.         }
  151.         // Fix bug with undefined index notice in Imagine
  152.         if ('webp' === $imagineOptions['format'] && !isset($imagineOptions['webp_quality'])) {
  153.             $imagineOptions['webp_quality'] = 80;
  154.         }
  155.         $tmpPath1 $this->filesystem->tempnam($dir'img');
  156.         $tmpPath2 $this->filesystem->tempnam($dir'img');
  157.         $this->filesystem->chmod([$tmpPath1$tmpPath2], 0666umask());
  158.         if ($options->getPreserveCopyrightMetadata() && ($metadata $this->getMetadata($image))->getAll()) {
  159.             $imagineImage->save($tmpPath1$imagineOptions);
  160.             try {
  161.                 $this->metadataReaderWriter->applyCopyrightToFile($tmpPath1$tmpPath2$metadata$options->getPreserveCopyrightMetadata());
  162.             } catch (\Throwable $exception) {
  163.                 $this->filesystem->rename($tmpPath1$tmpPath2true);
  164.             }
  165.         } else {
  166.             $imagineImage->save($tmpPath2$imagineOptions);
  167.         }
  168.         $this->filesystem->remove($tmpPath1);
  169.         // Atomic write operation
  170.         $this->filesystem->rename($tmpPath2$pathtrue);
  171.         return $this->createImage($image$path);
  172.     }
  173.     /**
  174.      * Creates a new image instance for the specified path.
  175.      *
  176.      * @internal Do not call this method in your code; it will be made private in a future version
  177.      */
  178.     protected function createImage(ImageInterface $imagestring $path): ImageInterface
  179.     {
  180.         return new Image($path$image->getImagine(), $this->filesystem);
  181.     }
  182.     /**
  183.      * Processes the resize and executes it if not already cached.
  184.      *
  185.      * @internal
  186.      */
  187.     protected function processResize(ImageInterface $imageResizeConfiguration $configResizeOptions $options): ImageInterface
  188.     {
  189.         $coordinates $this->calculator->calculate($config$image->getDimensions(), $image->getImportantPart());
  190.         // Skip resizing if it would have no effect
  191.         if (
  192.             $this->canSkipResize($image$options)
  193.             && !$image->getDimensions()->isRelative()
  194.             && $coordinates->isEqualTo($image->getDimensions()->getSize())
  195.         ) {
  196.             return $this->createImage($image$image->getPath());
  197.         }
  198.         $cachePath Path::join($this->cacheDir$this->createCachePath($image->getPath(), $coordinates$optionsfalse));
  199.         if (!$options->getBypassCache()) {
  200.             if ($this->filesystem->exists($cachePath)) {
  201.                 return $this->createImage($image$cachePath);
  202.             }
  203.             $legacyCachePath Path::join($this->cacheDir$this->createCachePath($image->getPath(), $coordinates$optionstrue));
  204.             if ($this->filesystem->exists($legacyCachePath)) {
  205.                 trigger_deprecation('contao/image''1.2''Reusing old cached images like "%s" from version 1.1 has been deprecated and will no longer work in version 2.0. Clear the image cache directory "%s" and regenerate all images to get rid of this message.'$legacyCachePath$this->cacheDir);
  206.                 return $this->createImage($image$legacyCachePath);
  207.             }
  208.         }
  209.         return $this->executeResize($image$coordinates$cachePath$options);
  210.     }
  211.     private function canSkipResize(ImageInterface $imageResizeOptions $options): bool
  212.     {
  213.         if (!$options->getSkipIfDimensionsMatch()) {
  214.             return false;
  215.         }
  216.         if (ImageDimensions::ORIENTATION_NORMAL !== $image->getDimensions()->getOrientation()) {
  217.             return false;
  218.         }
  219.         if (
  220.             isset($options->getImagineOptions()['format'])
  221.             && $options->getImagineOptions()['format'] !== strtolower(pathinfo($image->getPath(), PATHINFO_EXTENSION))
  222.         ) {
  223.             return false;
  224.         }
  225.         return true;
  226.     }
  227.     /**
  228.      * Returns the relative target cache path.
  229.      */
  230.     private function createCachePath(string $pathResizeCoordinates $coordinatesResizeOptions $optionsbool $useLegacyHash): string
  231.     {
  232.         $imagineOptions $options->getImagineOptions();
  233.         ksort($imagineOptions);
  234.         $hashData array_merge(
  235.             [
  236.                 Path::makeRelative($path$this->cacheDir),
  237.                 filemtime($path),
  238.                 $coordinates->getHash(),
  239.             ],
  240.             array_keys($imagineOptions),
  241.             array_map(
  242.                 static function ($value) {
  243.                     return \is_array($value) ? implode(','$value) : $value;
  244.                 },
  245.                 array_values($imagineOptions)
  246.             )
  247.         );
  248.         $preserveMeta $options->getPreserveCopyrightMetadata();
  249.         if ($preserveMeta !== (new ResizeOptions())->getPreserveCopyrightMetadata()) {
  250.             ksort($preserveMetaSORT_STRING);
  251.             $hashData[] = json_encode($preserveMeta);
  252.         }
  253.         if ($useLegacyHash || null === $this->secret) {
  254.             $hash substr(md5(implode('|'$hashData)), 09);
  255.         } else {
  256.             $hash hash_hmac('sha256'implode('|'$hashData), $this->secrettrue);
  257.             $hash substr($this->encodeBase32($hash), 016);
  258.         }
  259.         $pathinfo pathinfo($path);
  260.         $extension $options->getImagineOptions()['format'] ?? strtolower($pathinfo['extension']);
  261.         return Path::join($hash[0], $pathinfo['filename'].'-'.substr($hash1).'.'.$extension);
  262.     }
  263.     private function getMetadata(ImageInterface $image): ImageMetadata
  264.     {
  265.         try {
  266.             return $this->metadataReaderWriter->parse($image->getPath());
  267.         } catch (\Throwable $exception) {
  268.             return new ImageMetadata([]);
  269.         }
  270.     }
  271.     /**
  272.      * Encode a string with Crockford’s Base32 in lowercase
  273.      * (0123456789abcdefghjkmnpqrstvwxyz).
  274.      */
  275.     private function encodeBase32(string $bytes): string
  276.     {
  277.         $result = [];
  278.         foreach (str_split($bytes5) as $chunk) {
  279.             $result[] = substr(
  280.                 str_pad(
  281.                     strtr(
  282.                         base_convert(bin2hex(str_pad($chunk5"\0")), 1632),
  283.                         'ijklmnopqrstuv',
  284.                         'jkmnpqrstvwxyz' // Crockford's Base32
  285.                     ),
  286.                     8,
  287.                     '0',
  288.                     STR_PAD_LEFT
  289.                 ),
  290.                 0,
  291.                 (int) ceil(\strlen($chunk) * 5)
  292.             );
  293.         }
  294.         return implode(''$result);
  295.     }
  296. }