398 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			398 lines
		
	
	
		
			12 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;
 | |
| 
 | |
| use Doctrine\Common\Annotations\AnnotationReader;
 | |
| use Doctrine\Common\Annotations\PsrCachedReader;
 | |
| use Doctrine\Common\Annotations\Reader;
 | |
| use Psr\Cache\CacheItemPoolInterface;
 | |
| use Symfony\Component\Cache\Adapter\ArrayAdapter;
 | |
| use Symfony\Component\Validator\Context\ExecutionContextFactory;
 | |
| use Symfony\Component\Validator\Exception\LogicException;
 | |
| use Symfony\Component\Validator\Exception\ValidatorException;
 | |
| use Symfony\Component\Validator\Mapping\Factory\LazyLoadingMetadataFactory;
 | |
| use Symfony\Component\Validator\Mapping\Factory\MetadataFactoryInterface;
 | |
| use Symfony\Component\Validator\Mapping\Loader\AnnotationLoader;
 | |
| use Symfony\Component\Validator\Mapping\Loader\LoaderChain;
 | |
| use Symfony\Component\Validator\Mapping\Loader\LoaderInterface;
 | |
| use Symfony\Component\Validator\Mapping\Loader\StaticMethodLoader;
 | |
| use Symfony\Component\Validator\Mapping\Loader\XmlFileLoader;
 | |
| use Symfony\Component\Validator\Mapping\Loader\YamlFileLoader;
 | |
| use Symfony\Component\Validator\Validator\RecursiveValidator;
 | |
| use Symfony\Component\Validator\Validator\ValidatorInterface;
 | |
| use Symfony\Contracts\Translation\LocaleAwareInterface;
 | |
| use Symfony\Contracts\Translation\TranslatorInterface;
 | |
| use Symfony\Contracts\Translation\TranslatorTrait;
 | |
| 
 | |
| // Help opcache.preload discover always-needed symbols
 | |
| class_exists(TranslatorInterface::class);
 | |
| class_exists(LocaleAwareInterface::class);
 | |
| class_exists(TranslatorTrait::class);
 | |
| 
 | |
| /**
 | |
|  * @author Bernhard Schussek <bschussek@gmail.com>
 | |
|  */
 | |
| class ValidatorBuilder
 | |
| {
 | |
|     private array $initializers = [];
 | |
|     private array $loaders = [];
 | |
|     private array $xmlMappings = [];
 | |
|     private array $yamlMappings = [];
 | |
|     private array $methodMappings = [];
 | |
|     private ?Reader $annotationReader = null;
 | |
|     private bool $enableAnnotationMapping = false;
 | |
|     private ?MetadataFactoryInterface $metadataFactory = null;
 | |
|     private ConstraintValidatorFactoryInterface $validatorFactory;
 | |
|     private ?CacheItemPoolInterface $mappingCache = null;
 | |
|     private ?TranslatorInterface $translator = null;
 | |
|     private ?string $translationDomain = null;
 | |
| 
 | |
|     /**
 | |
|      * Adds an object initializer to the validator.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addObjectInitializer(ObjectInitializerInterface $initializer): static
 | |
|     {
 | |
|         $this->initializers[] = $initializer;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Adds a list of object initializers to the validator.
 | |
|      *
 | |
|      * @param ObjectInitializerInterface[] $initializers
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addObjectInitializers(array $initializers): static
 | |
|     {
 | |
|         $this->initializers = array_merge($this->initializers, $initializers);
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Adds an XML constraint mapping file to the validator.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addXmlMapping(string $path): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->xmlMappings[] = $path;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Adds a list of XML constraint mapping files to the validator.
 | |
|      *
 | |
|      * @param string[] $paths The paths to the mapping files
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addXmlMappings(array $paths): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->xmlMappings = array_merge($this->xmlMappings, $paths);
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Adds a YAML constraint mapping file to the validator.
 | |
|      *
 | |
|      * @param string $path The path to the mapping file
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addYamlMapping(string $path): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->yamlMappings[] = $path;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Adds a list of YAML constraint mappings file to the validator.
 | |
|      *
 | |
|      * @param string[] $paths The paths to the mapping files
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addYamlMappings(array $paths): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->yamlMappings = array_merge($this->yamlMappings, $paths);
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Enables constraint mapping using the given static method.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addMethodMapping(string $methodName): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->methodMappings[] = $methodName;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Enables constraint mapping using the given static methods.
 | |
|      *
 | |
|      * @param string[] $methodNames The names of the methods
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addMethodMappings(array $methodNames): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot add custom mappings after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->methodMappings = array_merge($this->methodMappings, $methodNames);
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Enables annotation based constraint mapping.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function enableAnnotationMapping(): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot enable annotation mapping after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->enableAnnotationMapping = true;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Disables annotation based constraint mapping.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function disableAnnotationMapping(): static
 | |
|     {
 | |
|         $this->enableAnnotationMapping = false;
 | |
|         $this->annotationReader = null;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setDoctrineAnnotationReader(?Reader $reader): static
 | |
|     {
 | |
|         $this->annotationReader = $reader;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addDefaultDoctrineAnnotationReader(): static
 | |
|     {
 | |
|         $this->annotationReader = $this->createAnnotationReader();
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the class metadata factory used by the validator.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setMetadataFactory(MetadataFactoryInterface $metadataFactory): static
 | |
|     {
 | |
|         if (\count($this->xmlMappings) > 0 || \count($this->yamlMappings) > 0 || \count($this->methodMappings) > 0 || $this->enableAnnotationMapping) {
 | |
|             throw new ValidatorException('You cannot set a custom metadata factory after adding custom mappings. You should do either of both.');
 | |
|         }
 | |
| 
 | |
|         $this->metadataFactory = $metadataFactory;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the cache for caching class metadata.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setMappingCache(CacheItemPoolInterface $cache): static
 | |
|     {
 | |
|         if (null !== $this->metadataFactory) {
 | |
|             throw new ValidatorException('You cannot set a custom mapping cache after setting a custom metadata factory. Configure your metadata factory instead.');
 | |
|         }
 | |
| 
 | |
|         $this->mappingCache = $cache;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the constraint validator factory used by the validator.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setConstraintValidatorFactory(ConstraintValidatorFactoryInterface $validatorFactory): static
 | |
|     {
 | |
|         $this->validatorFactory = $validatorFactory;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the translator used for translating violation messages.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setTranslator(TranslatorInterface $translator): static
 | |
|     {
 | |
|         $this->translator = $translator;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Sets the default translation domain of violation messages.
 | |
|      *
 | |
|      * The same message can have different translations in different domains.
 | |
|      * Pass the domain that is used for violation messages by default to this
 | |
|      * method.
 | |
|      *
 | |
|      * @return $this
 | |
|      */
 | |
|     public function setTranslationDomain(?string $translationDomain): static
 | |
|     {
 | |
|         $this->translationDomain = $translationDomain;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return $this
 | |
|      */
 | |
|     public function addLoader(LoaderInterface $loader): static
 | |
|     {
 | |
|         $this->loaders[] = $loader;
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return LoaderInterface[]
 | |
|      */
 | |
|     public function getLoaders(): array
 | |
|     {
 | |
|         $loaders = [];
 | |
| 
 | |
|         foreach ($this->xmlMappings as $xmlMapping) {
 | |
|             $loaders[] = new XmlFileLoader($xmlMapping);
 | |
|         }
 | |
| 
 | |
|         foreach ($this->yamlMappings as $yamlMappings) {
 | |
|             $loaders[] = new YamlFileLoader($yamlMappings);
 | |
|         }
 | |
| 
 | |
|         foreach ($this->methodMappings as $methodName) {
 | |
|             $loaders[] = new StaticMethodLoader($methodName);
 | |
|         }
 | |
| 
 | |
|         if ($this->enableAnnotationMapping) {
 | |
|             $loaders[] = new AnnotationLoader($this->annotationReader);
 | |
|         }
 | |
| 
 | |
|         return array_merge($loaders, $this->loaders);
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Builds and returns a new validator object.
 | |
|      */
 | |
|     public function getValidator(): ValidatorInterface
 | |
|     {
 | |
|         $metadataFactory = $this->metadataFactory;
 | |
| 
 | |
|         if (!$metadataFactory) {
 | |
|             $loaders = $this->getLoaders();
 | |
|             $loader = null;
 | |
| 
 | |
|             if (\count($loaders) > 1) {
 | |
|                 $loader = new LoaderChain($loaders);
 | |
|             } elseif (1 === \count($loaders)) {
 | |
|                 $loader = $loaders[0];
 | |
|             }
 | |
| 
 | |
|             $metadataFactory = new LazyLoadingMetadataFactory($loader, $this->mappingCache);
 | |
|         }
 | |
| 
 | |
|         $validatorFactory = $this->validatorFactory ?? new ConstraintValidatorFactory();
 | |
|         $translator = $this->translator;
 | |
| 
 | |
|         if (null === $translator) {
 | |
|             $translator = new class() implements TranslatorInterface, LocaleAwareInterface {
 | |
|                 use TranslatorTrait;
 | |
|             };
 | |
|             // Force the locale to be 'en' when no translator is provided rather than relying on the Intl default locale
 | |
|             // This avoids depending on Intl or the stub implementation being available. It also ensures that Symfony
 | |
|             // validation messages are pluralized properly even when the default locale gets changed because they are in
 | |
|             // English.
 | |
|             $translator->setLocale('en');
 | |
|         }
 | |
| 
 | |
|         $contextFactory = new ExecutionContextFactory($translator, $this->translationDomain);
 | |
| 
 | |
|         return new RecursiveValidator($contextFactory, $metadataFactory, $validatorFactory, $this->initializers);
 | |
|     }
 | |
| 
 | |
|     private function createAnnotationReader(): Reader
 | |
|     {
 | |
|         if (!class_exists(AnnotationReader::class)) {
 | |
|             throw new LogicException('Enabling annotation based constraint mapping requires the packages doctrine/annotations and symfony/cache to be installed.');
 | |
|         }
 | |
| 
 | |
|         if (class_exists(ArrayAdapter::class)) {
 | |
|             return new PsrCachedReader(new AnnotationReader(), new ArrayAdapter());
 | |
|         }
 | |
| 
 | |
|         throw new LogicException('Enabling annotation based constraint mapping requires the packages doctrine/annotations and symfony/cache to be installed.');
 | |
|     }
 | |
| }
 |