158 lines
4.7 KiB
PHP
158 lines
4.7 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\HttpFoundation\File\MimeType\MimeTypeGuesser;
|
|
use Symfony\Component\HttpFoundation\File\MimeType\MimeTypeGuesserInterface;
|
|
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
|
|
{
|
|
private static $supportedTypes = array(
|
|
\SplFileInfo::class => true,
|
|
\SplFileObject::class => true,
|
|
File::class => true,
|
|
);
|
|
|
|
/**
|
|
* @var MimeTypeGuesserInterface
|
|
*/
|
|
private $mimeTypeGuesser;
|
|
|
|
public function __construct(MimeTypeGuesserInterface $mimeTypeGuesser = null)
|
|
{
|
|
if (null === $mimeTypeGuesser && class_exists('Symfony\Component\HttpFoundation\File\MimeType\MimeTypeGuesser')) {
|
|
$mimeTypeGuesser = MimeTypeGuesser::getInstance();
|
|
}
|
|
|
|
$this->mimeTypeGuesser = $mimeTypeGuesser;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function normalize($object, $format = null, array $context = array())
|
|
{
|
|
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));
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function supportsNormalization($data, $format = null)
|
|
{
|
|
return $data instanceof \SplFileInfo;
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*
|
|
* Regex adapted from Brian Grinstead code.
|
|
*
|
|
* @see https://gist.github.com/bgrins/6194623
|
|
*
|
|
* @throws InvalidArgumentException
|
|
* @throws NotNormalizableValueException
|
|
*/
|
|
public function denormalize($data, $class, $format = null, array $context = array())
|
|
{
|
|
if (!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 new NotNormalizableValueException('The provided "data:" URI is not valid.');
|
|
}
|
|
|
|
try {
|
|
switch ($class) {
|
|
case 'Symfony\Component\HttpFoundation\File\File':
|
|
return new File($data, false);
|
|
|
|
case 'SplFileObject':
|
|
case 'SplFileInfo':
|
|
return new \SplFileObject($data);
|
|
}
|
|
} catch (\RuntimeException $exception) {
|
|
throw new NotNormalizableValueException($exception->getMessage(), $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".', $class));
|
|
}
|
|
|
|
/**
|
|
* {@inheritdoc}
|
|
*/
|
|
public function supportsDenormalization($data, $type, $format = null)
|
|
{
|
|
return isset(self::$supportedTypes[$type]);
|
|
}
|
|
|
|
/**
|
|
* Gets the mime type of the object. Defaults to application/octet-stream.
|
|
*
|
|
* @param \SplFileInfo $object
|
|
*
|
|
* @return string
|
|
*/
|
|
private function getMimeType(\SplFileInfo $object)
|
|
{
|
|
if ($object instanceof File) {
|
|
return $object->getMimeType();
|
|
}
|
|
|
|
if ($this->mimeTypeGuesser && $mimeType = $this->mimeTypeGuesser->guess($object->getPathname())) {
|
|
return $mimeType;
|
|
}
|
|
|
|
return 'application/octet-stream';
|
|
}
|
|
|
|
/**
|
|
* Returns the \SplFileObject instance associated with the given \SplFileInfo instance.
|
|
*
|
|
* @param \SplFileInfo $object
|
|
*
|
|
* @return \SplFileObject
|
|
*/
|
|
private function extractSplFileObject(\SplFileInfo $object)
|
|
{
|
|
if ($object instanceof \SplFileObject) {
|
|
return $object;
|
|
}
|
|
|
|
return $object->openFile();
|
|
}
|
|
}
|