154 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			154 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			PHP
		
	
	
| <?php
 | |
| 
 | |
| /*
 | |
|  * This file is part of the Symfony package.
 | |
|  *
 | |
|  * (c) Fabien Potencier <fabien@symfony.com>
 | |
|  *
 | |
|  * For the full copyright and license information, please view the LICENSE
 | |
|  * file that was distributed with this source code.
 | |
|  */
 | |
| 
 | |
| namespace Symfony\Component\Serializer\Normalizer;
 | |
| 
 | |
| use Symfony\Component\HttpFoundation\File\File;
 | |
| use Symfony\Component\Mime\MimeTypeGuesserInterface;
 | |
| use Symfony\Component\Mime\MimeTypes;
 | |
| use Symfony\Component\Serializer\Exception\InvalidArgumentException;
 | |
| use Symfony\Component\Serializer\Exception\NotNormalizableValueException;
 | |
| 
 | |
| /**
 | |
|  * Normalizes an {@see \SplFileInfo} object to a data URI.
 | |
|  * Denormalizes a data URI to a {@see \SplFileObject} object.
 | |
|  *
 | |
|  * @author Kévin Dunglas <dunglas@gmail.com>
 | |
|  */
 | |
| class DataUriNormalizer implements NormalizerInterface, DenormalizerInterface, CacheableSupportsMethodInterface
 | |
| {
 | |
|     private const SUPPORTED_TYPES = [
 | |
|         \SplFileInfo::class => true,
 | |
|         \SplFileObject::class => true,
 | |
|         File::class => true,
 | |
|     ];
 | |
| 
 | |
|     /**
 | |
|      * @var MimeTypeGuesserInterface|null
 | |
|      */
 | |
|     private $mimeTypeGuesser;
 | |
| 
 | |
|     public function __construct(MimeTypeGuesserInterface $mimeTypeGuesser = null)
 | |
|     {
 | |
|         if (!$mimeTypeGuesser && class_exists(MimeTypes::class)) {
 | |
|             $mimeTypeGuesser = MimeTypes::getDefault();
 | |
|         }
 | |
| 
 | |
|         $this->mimeTypeGuesser = $mimeTypeGuesser;
 | |
|     }
 | |
| 
 | |
|     public function normalize(mixed $object, string $format = null, array $context = []): string
 | |
|     {
 | |
|         if (!$object instanceof \SplFileInfo) {
 | |
|             throw new InvalidArgumentException('The object must be an instance of "\SplFileInfo".');
 | |
|         }
 | |
| 
 | |
|         $mimeType = $this->getMimeType($object);
 | |
|         $splFileObject = $this->extractSplFileObject($object);
 | |
| 
 | |
|         $data = '';
 | |
| 
 | |
|         $splFileObject->rewind();
 | |
|         while (!$splFileObject->eof()) {
 | |
|             $data .= $splFileObject->fgets();
 | |
|         }
 | |
| 
 | |
|         if ('text' === explode('/', $mimeType, 2)[0]) {
 | |
|             return sprintf('data:%s,%s', $mimeType, rawurlencode($data));
 | |
|         }
 | |
| 
 | |
|         return sprintf('data:%s;base64,%s', $mimeType, base64_encode($data));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param array $context
 | |
|      */
 | |
|     public function supportsNormalization(mixed $data, string $format = null /* , array $context = [] */): bool
 | |
|     {
 | |
|         return $data instanceof \SplFileInfo;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Regex adapted from Brian Grinstead code.
 | |
|      *
 | |
|      * @see https://gist.github.com/bgrins/6194623
 | |
|      *
 | |
|      * @throws InvalidArgumentException
 | |
|      * @throws NotNormalizableValueException
 | |
|      */
 | |
|     public function denormalize(mixed $data, string $type, string $format = null, array $context = []): \SplFileInfo
 | |
|     {
 | |
|         if (null === $data || !preg_match('/^data:([a-z0-9][a-z0-9\!\#\$\&\-\^\_\+\.]{0,126}\/[a-z0-9][a-z0-9\!\#\$\&\-\^\_\+\.]{0,126}(;[a-z0-9\-]+\=[a-z0-9\-]+)?)?(;base64)?,[a-z0-9\!\$\&\\\'\,\(\)\*\+\,\;\=\-\.\_\~\:\@\/\?\%\s]*\s*$/i', $data)) {
 | |
|             throw NotNormalizableValueException::createForUnexpectedDataType('The provided "data:" URI is not valid.', $data, ['string'], $context['deserialization_path'] ?? null, true);
 | |
|         }
 | |
| 
 | |
|         try {
 | |
|             switch ($type) {
 | |
|                 case File::class:
 | |
|                     if (!class_exists(File::class)) {
 | |
|                         throw new InvalidArgumentException(sprintf('Cannot denormalize to a "%s" without the HttpFoundation component installed. Try running "composer require symfony/http-foundation".', File::class));
 | |
|                     }
 | |
| 
 | |
|                     return new File($data, false);
 | |
| 
 | |
|                 case 'SplFileObject':
 | |
|                 case 'SplFileInfo':
 | |
|                     return new \SplFileObject($data);
 | |
|             }
 | |
|         } catch (\RuntimeException $exception) {
 | |
|             throw NotNormalizableValueException::createForUnexpectedDataType($exception->getMessage(), $data, ['string'], $context['deserialization_path'] ?? null, false, $exception->getCode(), $exception);
 | |
|         }
 | |
| 
 | |
|         throw new InvalidArgumentException(sprintf('The class parameter "%s" is not supported. It must be one of "SplFileInfo", "SplFileObject" or "Symfony\Component\HttpFoundation\File\File".', $type));
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param array $context
 | |
|      */
 | |
|     public function supportsDenormalization(mixed $data, string $type, string $format = null /* , array $context = [] */): bool
 | |
|     {
 | |
|         return isset(self::SUPPORTED_TYPES[$type]);
 | |
|     }
 | |
| 
 | |
|     public function hasCacheableSupportsMethod(): bool
 | |
|     {
 | |
|         return __CLASS__ === static::class;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Gets the mime type of the object. Defaults to application/octet-stream.
 | |
|      */
 | |
|     private function getMimeType(\SplFileInfo $object): string
 | |
|     {
 | |
|         if ($object instanceof File) {
 | |
|             return $object->getMimeType();
 | |
|         }
 | |
| 
 | |
|         if ($this->mimeTypeGuesser && $mimeType = $this->mimeTypeGuesser->guessMimeType($object->getPathname())) {
 | |
|             return $mimeType;
 | |
|         }
 | |
| 
 | |
|         return 'application/octet-stream';
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns the \SplFileObject instance associated with the given \SplFileInfo instance.
 | |
|      */
 | |
|     private function extractSplFileObject(\SplFileInfo $object): \SplFileObject
 | |
|     {
 | |
|         if ($object instanceof \SplFileObject) {
 | |
|             return $object;
 | |
|         }
 | |
| 
 | |
|         return $object->openFile();
 | |
|     }
 | |
| }
 |