289 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			289 lines
		
	
	
		
			11 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\Validator\Constraints;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\HttpFoundation\File\File;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Mime\MimeTypes;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Validator\Constraint;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Validator\Exception\ConstraintDefinitionException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Validator\Exception\LogicException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Validator\Exception\UnexpectedTypeException;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Validates whether a value is a valid image file and is valid
							 | 
						||
| 
								 | 
							
								 * against minWidth, maxWidth, minHeight and maxHeight constraints.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Benjamin Dulau <benjamin.dulau@gmail.com>
							 | 
						||
| 
								 | 
							
								 * @author Bernhard Schussek <bschussek@gmail.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class ImageValidator extends FileValidator
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    public function validate(mixed $value, Constraint $constraint): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!$constraint instanceof Image) {
							 | 
						||
| 
								 | 
							
								            throw new UnexpectedTypeException($constraint, Image::class);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $violations = \count($this->context->getViolations());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        parent::validate($value, $constraint);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $failed = \count($this->context->getViolations()) !== $violations;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($failed || null === $value || '' === $value) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null === $constraint->minWidth && null === $constraint->maxWidth
							 | 
						||
| 
								 | 
							
								            && null === $constraint->minHeight && null === $constraint->maxHeight
							 | 
						||
| 
								 | 
							
								            && null === $constraint->minPixels && null === $constraint->maxPixels
							 | 
						||
| 
								 | 
							
								            && null === $constraint->minRatio && null === $constraint->maxRatio
							 | 
						||
| 
								 | 
							
								            && $constraint->allowSquare && $constraint->allowLandscape && $constraint->allowPortrait
							 | 
						||
| 
								 | 
							
								            && !$constraint->detectCorrupted) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $isSvg = $this->isSvg($value);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($isSvg) {
							 | 
						||
| 
								 | 
							
								            $size = $this->getSvgSize($value);
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $size = @getimagesize($value);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$size || (0 === $size[0]) || (0 === $size[1])) {
							 | 
						||
| 
								 | 
							
								            $this->context->buildViolation($constraint->sizeNotDetectedMessage)
							 | 
						||
| 
								 | 
							
								                ->setCode(Image::SIZE_NOT_DETECTED_ERROR)
							 | 
						||
| 
								 | 
							
								                ->addViolation();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $width = $size[0];
							 | 
						||
| 
								 | 
							
								        $height = $size[1];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && $constraint->minWidth) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->minWidth)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid minimum width.', $constraint->minWidth));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($width < $constraint->minWidth) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->minWidthMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ min_width }}', $constraint->minWidth)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_NARROW_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && $constraint->maxWidth) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->maxWidth)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid maximum width.', $constraint->maxWidth));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($width > $constraint->maxWidth) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->maxWidthMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ max_width }}', $constraint->maxWidth)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_WIDE_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && $constraint->minHeight) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->minHeight)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid minimum height.', $constraint->minHeight));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($height < $constraint->minHeight) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->minHeightMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ min_height }}', $constraint->minHeight)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_LOW_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && $constraint->maxHeight) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->maxHeight)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid maximum height.', $constraint->maxHeight));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($height > $constraint->maxHeight) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->maxHeightMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ max_height }}', $constraint->maxHeight)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_HIGH_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $pixels = $width * $height;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && null !== $constraint->minPixels) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->minPixels)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid minimum amount of pixels.', $constraint->minPixels));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($pixels < $constraint->minPixels) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->minPixelsMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ pixels }}', $pixels)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ min_pixels }}', $constraint->minPixels)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_FEW_PIXEL_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$isSvg && null !== $constraint->maxPixels) {
							 | 
						||
| 
								 | 
							
								            if (!ctype_digit((string) $constraint->maxPixels)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid maximum amount of pixels.', $constraint->maxPixels));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($pixels > $constraint->maxPixels) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->maxPixelsMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ pixels }}', $pixels)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ max_pixels }}', $constraint->maxPixels)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::TOO_MANY_PIXEL_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $ratio = round($width / $height, 2);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null !== $constraint->minRatio) {
							 | 
						||
| 
								 | 
							
								            if (!is_numeric((string) $constraint->minRatio)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid minimum ratio.', $constraint->minRatio));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($ratio < round($constraint->minRatio, 2)) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->minRatioMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ ratio }}', $ratio)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ min_ratio }}', round($constraint->minRatio, 2))
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::RATIO_TOO_SMALL_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null !== $constraint->maxRatio) {
							 | 
						||
| 
								 | 
							
								            if (!is_numeric((string) $constraint->maxRatio)) {
							 | 
						||
| 
								 | 
							
								                throw new ConstraintDefinitionException(\sprintf('"%s" is not a valid maximum ratio.', $constraint->maxRatio));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($ratio > round($constraint->maxRatio, 2)) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->maxRatioMessage)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ ratio }}', $ratio)
							 | 
						||
| 
								 | 
							
								                    ->setParameter('{{ max_ratio }}', round($constraint->maxRatio, 2))
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::RATIO_TOO_BIG_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$constraint->allowSquare && $width == $height) {
							 | 
						||
| 
								 | 
							
								            $this->context->buildViolation($constraint->allowSquareMessage)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                ->setCode(Image::SQUARE_NOT_ALLOWED_ERROR)
							 | 
						||
| 
								 | 
							
								                ->addViolation();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$constraint->allowLandscape && $width > $height) {
							 | 
						||
| 
								 | 
							
								            $this->context->buildViolation($constraint->allowLandscapeMessage)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                ->setCode(Image::LANDSCAPE_NOT_ALLOWED_ERROR)
							 | 
						||
| 
								 | 
							
								                ->addViolation();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$constraint->allowPortrait && $width < $height) {
							 | 
						||
| 
								 | 
							
								            $this->context->buildViolation($constraint->allowPortraitMessage)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ width }}', $width)
							 | 
						||
| 
								 | 
							
								                ->setParameter('{{ height }}', $height)
							 | 
						||
| 
								 | 
							
								                ->setCode(Image::PORTRAIT_NOT_ALLOWED_ERROR)
							 | 
						||
| 
								 | 
							
								                ->addViolation();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($constraint->detectCorrupted) {
							 | 
						||
| 
								 | 
							
								            if (!\function_exists('imagecreatefromstring')) {
							 | 
						||
| 
								 | 
							
								                throw new LogicException('Corrupted images detection requires installed and enabled GD extension.');
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $resource = @imagecreatefromstring(file_get_contents($value));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (false === $resource) {
							 | 
						||
| 
								 | 
							
								                $this->context->buildViolation($constraint->corruptedMessage)
							 | 
						||
| 
								 | 
							
								                    ->setCode(Image::CORRUPTED_IMAGE_ERROR)
							 | 
						||
| 
								 | 
							
								                    ->addViolation();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function isSvg(mixed $value): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($value instanceof File) {
							 | 
						||
| 
								 | 
							
								            $mime = $value->getMimeType();
							 | 
						||
| 
								 | 
							
								        } elseif (class_exists(MimeTypes::class)) {
							 | 
						||
| 
								 | 
							
								            $mime = MimeTypes::getDefault()->guessMimeType($value);
							 | 
						||
| 
								 | 
							
								        } elseif (!class_exists(File::class)) {
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $mime = (new File($value))->getMimeType();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return 'image/svg+xml' === $mime;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @return array{int, int}|null index 0 and 1 contains respectively the width and the height of the image, null if size can't be found
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    private function getSvgSize(mixed $value): ?array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if ($value instanceof File) {
							 | 
						||
| 
								 | 
							
								            $content = $value->getContent();
							 | 
						||
| 
								 | 
							
								        } elseif (!class_exists(File::class)) {
							 | 
						||
| 
								 | 
							
								            return null;
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $content = (new File($value))->getContent();
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (1 === preg_match('/<svg[^<>]+width="(?<width>[0-9]+)"[^<>]*>/', $content, $widthMatches)) {
							 | 
						||
| 
								 | 
							
								            $width = (int) $widthMatches['width'];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (1 === preg_match('/<svg[^<>]+height="(?<height>[0-9]+)"[^<>]*>/', $content, $heightMatches)) {
							 | 
						||
| 
								 | 
							
								            $height = (int) $heightMatches['height'];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (1 === preg_match('/<svg[^<>]+viewBox="-?[0-9]+ -?[0-9]+ (?<width>-?[0-9]+) (?<height>-?[0-9]+)"[^<>]*>/', $content, $viewBoxMatches)) {
							 | 
						||
| 
								 | 
							
								            $width ??= (int) $viewBoxMatches['width'];
							 | 
						||
| 
								 | 
							
								            $height ??= (int) $viewBoxMatches['height'];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (isset($width) && isset($height)) {
							 | 
						||
| 
								 | 
							
								            return [$width, $height];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return null;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |