This commit is contained in:
Marek
2026-03-24 00:04:55 +01:00
commit c5229e48ed
4225 changed files with 511461 additions and 0 deletions

View File

@@ -0,0 +1,84 @@
<?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\DependencyInjection;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Reference;
/**
* Injects the automapping configuration as last argument of loaders tagged with the "validator.auto_mapper" tag.
*
* @author Kévin Dunglas <dunglas@gmail.com>
*/
class AddAutoMappingConfigurationPass implements CompilerPassInterface
{
public function process(ContainerBuilder $container): void
{
if (!$container->hasParameter('validator.auto_mapping') || !$container->hasDefinition('validator.builder')) {
return;
}
$config = $container->getParameter('validator.auto_mapping');
$globalNamespaces = [];
$servicesToNamespaces = [];
foreach ($config as $namespace => $value) {
if ([] === $value['services']) {
$globalNamespaces[] = $namespace;
continue;
}
foreach ($value['services'] as $service) {
$servicesToNamespaces[$service][] = $namespace;
}
}
$validatorBuilder = $container->getDefinition('validator.builder');
foreach ($container->findTaggedServiceIds('validator.auto_mapper') as $id => $tags) {
$regexp = $this->getRegexp(array_merge($globalNamespaces, $servicesToNamespaces[$id] ?? []));
$validatorBuilder->addMethodCall('addLoader', [new Reference($id)]);
$container->getDefinition($id)->setArgument('$classValidatorRegexp', $regexp);
}
$container->getParameterBag()->remove('validator.auto_mapping');
}
/**
* Builds a regexp to check if a class is auto-mapped.
*/
private function getRegexp(array $patterns): ?string
{
if (!$patterns) {
return null;
}
$regexps = [];
foreach ($patterns as $pattern) {
// Escape namespace
$regex = preg_quote(ltrim($pattern, '\\'));
// Wildcards * and **
$regex = strtr($regex, ['\\*\\*' => '.*?', '\\*' => '[^\\\\]*?']);
// If this class does not end by a slash, anchor the end
if (!str_ends_with($regex, '\\')) {
$regex .= '$';
}
$regexps[] = '^'.$regex;
}
return \sprintf('{%s}', implode('|', $regexps));
}
}

View File

@@ -0,0 +1,47 @@
<?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\DependencyInjection;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\Compiler\ServiceLocatorTagPass;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Reference;
/**
* @author Johannes M. Schmitt <schmittjoh@gmail.com>
* @author Robin Chalas <robin.chalas@gmail.com>
*/
class AddConstraintValidatorsPass implements CompilerPassInterface
{
public function process(ContainerBuilder $container): void
{
if (!$container->hasDefinition('validator.validator_factory')) {
return;
}
$validators = [];
foreach ($container->findTaggedServiceIds('validator.constraint_validator', true) as $id => $attributes) {
$definition = $container->getDefinition($id);
if (isset($attributes[0]['alias'])) {
$validators[$attributes[0]['alias']] = new Reference($id);
}
$validators[$definition->getClass()] = new Reference($id);
}
$container
->getDefinition('validator.validator_factory')
->replaceArgument(0, ServiceLocatorTagPass::register($container, $validators))
;
}
}

View File

@@ -0,0 +1,37 @@
<?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\DependencyInjection;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\Reference;
/**
* @author Fabien Potencier <fabien@symfony.com>
* @author Robin Chalas <robin.chalas@gmail.com>
*/
class AddValidatorInitializersPass implements CompilerPassInterface
{
public function process(ContainerBuilder $container): void
{
if (!$container->hasDefinition('validator.builder')) {
return;
}
$initializers = [];
foreach ($container->findTaggedServiceIds('validator.initializer', true) as $id => $attributes) {
$initializers[] = new Reference($id);
}
$container->getDefinition('validator.builder')->addMethodCall('addObjectInitializers', [$initializers]);
}
}

View File

@@ -0,0 +1,72 @@
<?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\DependencyInjection;
use Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\Validator\Exception\MappingException;
/**
* @author Nicolas Grekas <p@tchwork.com>
*/
final class AttributeMetadataPass implements CompilerPassInterface
{
public function process(ContainerBuilder $container): void
{
if (!$container->hasDefinition('validator.builder')) {
return;
}
$resolve = $container->getParameterBag()->resolveValue(...);
$mappedClasses = [];
foreach ($container->getDefinitions() as $id => $definition) {
if (!$definition->hasTag('validator.attribute_metadata')) {
continue;
}
$class = $resolve($definition->getClass());
foreach ($definition->getTag('validator.attribute_metadata') as $attributes) {
if ($class !== $for = $attributes['for'] ?? $class) {
$this->checkSourceMapsToTarget($container, $class, $for);
}
$mappedClasses[$for][$class] = true;
}
}
if (!$mappedClasses) {
return;
}
ksort($mappedClasses);
$container->getDefinition('validator.builder')
->addMethodCall('addAttributeMappings', [array_map('array_keys', $mappedClasses)]);
}
private function checkSourceMapsToTarget(ContainerBuilder $container, string $source, string $target): void
{
$source = $container->getReflectionClass($source);
$target = $container->getReflectionClass($target);
foreach ($source->getProperties() as $p) {
if ($p->class === $source->name && !($target->hasProperty($p->name) && $target->getProperty($p->name)->class === $target->name)) {
throw new MappingException(\sprintf('The property "%s" on "%s" is not present on "%s".', $p->name, $source->name, $target->name));
}
}
foreach ($source->getMethods() as $m) {
if ($m->class === $source->name && !($target->hasMethod($m->name) && $target->getMethod($m->name)->class === $target->name)) {
throw new MappingException(\sprintf('The method "%s" on "%s" is not present on "%s".', $m->name, $source->name, $target->name));
}
}
}
}