vendor/sulu/sulu/src/Sulu/Bundle/MediaBundle/Media/FormatManager/FormatManager.php line 109

  1. <?php
  2. /*
  3.  * This file is part of Sulu.
  4.  *
  5.  * (c) Sulu GmbH
  6.  *
  7.  * This source file is subject to the MIT license that is bundled
  8.  * with this source code in the file LICENSE.
  9.  */
  10. namespace Sulu\Bundle\MediaBundle\Media\FormatManager;
  11. use Psr\Log\LoggerInterface;
  12. use Psr\Log\NullLogger;
  13. use Sulu\Bundle\MediaBundle\Entity\FileVersion;
  14. use Sulu\Bundle\MediaBundle\Entity\MediaInterface;
  15. use Sulu\Bundle\MediaBundle\Entity\MediaRepository;
  16. use Sulu\Bundle\MediaBundle\Entity\MediaRepositoryInterface;
  17. use Sulu\Bundle\MediaBundle\Media\Exception\FormatNotFoundException;
  18. use Sulu\Bundle\MediaBundle\Media\Exception\ImageProxyInvalidImageFormat;
  19. use Sulu\Bundle\MediaBundle\Media\Exception\ImageProxyInvalidUrl;
  20. use Sulu\Bundle\MediaBundle\Media\Exception\ImageProxyMediaNotFoundException;
  21. use Sulu\Bundle\MediaBundle\Media\Exception\InvalidMimeTypeForPreviewException;
  22. use Sulu\Bundle\MediaBundle\Media\Exception\MediaException;
  23. use Sulu\Bundle\MediaBundle\Media\FormatCache\FormatCacheInterface;
  24. use Sulu\Bundle\MediaBundle\Media\ImageConverter\ImageConverterInterface;
  25. use Symfony\Component\Filesystem\Filesystem;
  26. use Symfony\Component\HttpFoundation\ParameterBag;
  27. use Symfony\Component\HttpFoundation\Response;
  28. /**
  29.  * Sulu format manager for media.
  30.  */
  31. class FormatManager implements FormatManagerInterface
  32. {
  33.     /**
  34.      * The repository for communication with the database.
  35.      *
  36.      * @var MediaRepository
  37.      */
  38.     private $mediaRepository;
  39.     /**
  40.      * @var FormatCacheInterface
  41.      */
  42.     private $formatCache;
  43.     /**
  44.      * @var ImageConverterInterface
  45.      */
  46.     private $converter;
  47.     /**
  48.      * @var bool
  49.      */
  50.     private $saveImage false;
  51.     /**
  52.      * @var array
  53.      */
  54.     private $responseHeaders = [];
  55.     /**
  56.      * @var Filesystem
  57.      */
  58.     private $fileSystem;
  59.     /**
  60.      * @var array
  61.      */
  62.     private $formats;
  63.     /**
  64.      * @var LoggerInterface
  65.      */
  66.     private $logger;
  67.     /**
  68.      * @var ParameterBag|null
  69.      */
  70.     private $supportedImageFormats;
  71.     /**
  72.      * @param string $saveImage
  73.      * @param array $responseHeaders
  74.      * @param array $formats
  75.      * @param LoggerInterface $logger
  76.      */
  77.     public function __construct(
  78.         MediaRepositoryInterface $mediaRepository,
  79.         FormatCacheInterface $formatCache,
  80.         ImageConverterInterface $converter,
  81.         $saveImage,
  82.         $responseHeaders,
  83.         $formats,
  84.         LoggerInterface $logger null
  85.     ) {
  86.         $this->mediaRepository $mediaRepository;
  87.         $this->formatCache $formatCache;
  88.         $this->converter $converter;
  89.         $this->saveImage 'true' == $saveImage true false;
  90.         $this->responseHeaders $responseHeaders;
  91.         $this->fileSystem = new Filesystem();
  92.         $this->formats $formats;
  93.         $this->logger $logger ?: new NullLogger();
  94.     }
  95.     public function returnImage($id$formatKey$fileName)
  96.     {
  97.         $setExpireHeaders false;
  98.         try {
  99.             $info \pathinfo($fileName);
  100.             if (!isset($info['extension'])) {
  101.                 throw new ImageProxyInvalidUrl('No `extension` was found in the url');
  102.             }
  103.             $imageFormat $info['extension'];
  104.             $media $this->mediaRepository->findMediaByIdForRendering($id$formatKey);
  105.             if (!$media) {
  106.                 throw new ImageProxyMediaNotFoundException('Media was not found');
  107.             }
  108.             $fileVersion $this->getLatestFileVersion($media);
  109.             $supportedImageFormats $this->converter->getSupportedOutputImageFormats($fileVersion->getMimeType());
  110.             if (empty($supportedImageFormats)) {
  111.                 throw new InvalidMimeTypeForPreviewException($fileVersion->getMimeType() ?? '-null-');
  112.             }
  113.             if (!\in_array($imageFormat$supportedImageFormats)) {
  114.                 throw new ImageProxyInvalidImageFormat(
  115.                     \sprintf(
  116.                         'Image format "%s" not supported. Supported image formats are: %s',
  117.                         $imageFormat,
  118.                         \implode(', '$supportedImageFormats)
  119.                     ));
  120.             }
  121.             // Convert Media to format.
  122.             $responseContent $this->converter->convert($fileVersion$formatKey$imageFormat);
  123.             // HTTP Headers
  124.             $status 200;
  125.             $setExpireHeaders true;
  126.             $finfo = new \finfo(\FILEINFO_MIME_TYPE);
  127.             $mimeType $finfo->buffer($responseContent);
  128.             // Save image.
  129.             if ($this->saveImage) {
  130.                 $this->formatCache->save(
  131.                     $responseContent,
  132.                     $media->getId(),
  133.                     $this->replaceExtension($fileVersion->getName(), $imageFormat),
  134.                     $formatKey
  135.                 );
  136.             }
  137.         } catch (MediaException $e) {
  138.             $this->logger->error($e->getMessage(), ['exception' => $e]);
  139.             $responseContent null;
  140.             $status 404;
  141.             $mimeType null;
  142.         }
  143.         // Set header.
  144.         $headers $this->getResponseHeaders($mimeType$setExpireHeaders);
  145.         // Return image.
  146.         return new Response($responseContent$status$headers);
  147.     }
  148.     public function getFormats($id$fileName$version$subVersion$mimeType)
  149.     {
  150.         $formats = [];
  151.         $extensions $this->converter->getSupportedOutputImageFormats($mimeType);
  152.         if (empty($extensions)) {
  153.             return [];
  154.         }
  155.         $originalExtension $extensions[0];
  156.         foreach ($this->formats as $format) {
  157.             foreach ($extensions as $extension) {
  158.                 $formatUrl $this->formatCache->getMediaUrl(
  159.                     $id,
  160.                     $this->replaceExtension($fileName$extension),
  161.                     $format['key'],
  162.                     $version,
  163.                     $subVersion
  164.                 );
  165.                 if ($extension === $originalExtension) {
  166.                     $formats[$format['key']] = $formatUrl;
  167.                 }
  168.                 $formats[$format['key'] . '.' $extension] = $formatUrl;
  169.             }
  170.         }
  171.         return $formats;
  172.     }
  173.     public function purge($idMedia$fileName$mimeType)
  174.     {
  175.         $extensions $this->converter->getSupportedOutputImageFormats($mimeType);
  176.         if (empty($extensions)) {
  177.             return true;
  178.         }
  179.         foreach ($this->formats as $format) {
  180.             foreach ($extensions as $extension) {
  181.                 $this->formatCache->purge($idMedia$this->replaceExtension($fileName$extension), $format['key']);
  182.             }
  183.         }
  184.         return true;
  185.     }
  186.     public function clearCache()
  187.     {
  188.         $this->formatCache->clear();
  189.     }
  190.     public function getFormatDefinition($formatKey$locale null)
  191.     {
  192.         if (!isset($this->formats[$formatKey])) {
  193.             throw new FormatNotFoundException($formatKey);
  194.         }
  195.         $format $this->formats[$formatKey];
  196.         $title $format['key'];
  197.         if (\array_key_exists($locale$format['meta']['title'])) {
  198.             $title $format['meta']['title'][$locale];
  199.         } elseif (\count($format['meta']['title']) > 0) {
  200.             $title \array_values($format['meta']['title'])[0];
  201.         }
  202.         $formatArray = [
  203.             'internal' => $format['internal'],
  204.             'key' => $format['key'],
  205.             'title' => $title,
  206.             'scale' => $format['scale'],
  207.         ];
  208.         return $formatArray;
  209.     }
  210.     public function getFormatDefinitions($locale null)
  211.     {
  212.         $definitionsArray = [];
  213.         foreach ($this->formats as $format) {
  214.             $definitionsArray[$format['key']] = $this->getFormatDefinition($format['key'], $locale);
  215.         }
  216.         return $definitionsArray;
  217.     }
  218.     /**
  219.      * @param string $mimeType
  220.      * @param bool $setExpireHeaders
  221.      *
  222.      * @return array
  223.      */
  224.     protected function getResponseHeaders($mimeType ''$setExpireHeaders false)
  225.     {
  226.         $headers = [];
  227.         if (!empty($mimeType)) {
  228.             $headers['Content-Type'] = $mimeType;
  229.         }
  230.         if (empty($this->responseHeaders)) {
  231.             return $headers;
  232.         }
  233.         $headers \array_merge(
  234.             $headers,
  235.             $this->responseHeaders
  236.         );
  237.         if (isset($this->responseHeaders['Expires']) && $setExpireHeaders) {
  238.             $date = new \DateTime();
  239.             $date->modify($this->responseHeaders['Expires']);
  240.             $headers['Expires'] = $date->format('D, d M Y H:i:s \G\M\T');
  241.         } else {
  242.             // will remove exist set expire header
  243.             $headers['Expires'] = null;
  244.             $headers['Cache-Control'] = 'no-cache';
  245.             $headers['Pragma'] = null;
  246.         }
  247.         return $headers;
  248.     }
  249.     /**
  250.      * Replace extension.
  251.      *
  252.      * @param string $filename
  253.      * @param string $newExtension
  254.      *
  255.      * @return string
  256.      */
  257.     private function replaceExtension($filename$newExtension)
  258.     {
  259.         $info \pathinfo($filename);
  260.         return $info['filename'] . '.' $newExtension;
  261.     }
  262.     /**
  263.      * @return FileVersion
  264.      *
  265.      * @throws ImageProxyMediaNotFoundException
  266.      */
  267.     private function getLatestFileVersion(MediaInterface $media)
  268.     {
  269.         foreach ($media->getFiles() as $file) {
  270.             $version $file->getVersion();
  271.             foreach ($file->getFileVersions() as $fileVersion) {
  272.                 if ($fileVersion->getVersion() == $version) {
  273.                     return $fileVersion;
  274.                 }
  275.             }
  276.             break;
  277.         }
  278.         throw new ImageProxyMediaNotFoundException('Media file version was not found');
  279.     }
  280. }