225 lines
		
	
	
		
			9.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			225 lines
		
	
	
		
			9.9 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\Mapping\Loader;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\Context;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\DiscriminatorMap;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\Groups;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\Ignore;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\MaxDepth;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\SerializedName;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Attribute\SerializedPath;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\MappingException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\AttributeMetadata;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\AttributeMetadataInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\ClassDiscriminatorMapping;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\ClassMetadataInterface;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Loader for PHP attributes.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Kévin Dunglas <dunglas@gmail.com>
							 | 
						||
| 
								 | 
							
								 * @author Alexander M. Turek <me@derrabus.de>
							 | 
						||
| 
								 | 
							
								 * @author Alexandre Daubois <alex.daubois@gmail.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class AttributeLoader implements LoaderInterface
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    private const KNOWN_ATTRIBUTES = [
							 | 
						||
| 
								 | 
							
								        DiscriminatorMap::class,
							 | 
						||
| 
								 | 
							
								        Groups::class,
							 | 
						||
| 
								 | 
							
								        Ignore::class,
							 | 
						||
| 
								 | 
							
								        MaxDepth::class,
							 | 
						||
| 
								 | 
							
								        SerializedName::class,
							 | 
						||
| 
								 | 
							
								        SerializedPath::class,
							 | 
						||
| 
								 | 
							
								        Context::class,
							 | 
						||
| 
								 | 
							
								    ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function __construct()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function loadClassMetadata(ClassMetadataInterface $classMetadata): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $reflectionClass = $classMetadata->getReflectionClass();
							 | 
						||
| 
								 | 
							
								        $className = $reflectionClass->name;
							 | 
						||
| 
								 | 
							
								        $loaded = false;
							 | 
						||
| 
								 | 
							
								        $classGroups = [];
							 | 
						||
| 
								 | 
							
								        $classContextAttribute = null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $attributesMetadata = $classMetadata->getAttributesMetadata();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($this->loadAttributes($reflectionClass) as $attribute) {
							 | 
						||
| 
								 | 
							
								            match (true) {
							 | 
						||
| 
								 | 
							
								                $attribute instanceof DiscriminatorMap => $classMetadata->setClassDiscriminatorMapping(new ClassDiscriminatorMapping($attribute->getTypeProperty(), $attribute->getMapping(), $attribute->getDefaultType())),
							 | 
						||
| 
								 | 
							
								                $attribute instanceof Groups => $classGroups = $attribute->getGroups(),
							 | 
						||
| 
								 | 
							
								                $attribute instanceof Context => $classContextAttribute = $attribute,
							 | 
						||
| 
								 | 
							
								                default => null,
							 | 
						||
| 
								 | 
							
								            };
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($reflectionClass->getProperties() as $property) {
							 | 
						||
| 
								 | 
							
								            if (!isset($attributesMetadata[$property->name])) {
							 | 
						||
| 
								 | 
							
								                $attributesMetadata[$property->name] = new AttributeMetadata($property->name);
							 | 
						||
| 
								 | 
							
								                $classMetadata->addAttributeMetadata($attributesMetadata[$property->name]);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $attributeMetadata = $attributesMetadata[$property->name];
							 | 
						||
| 
								 | 
							
								            if ($property->getDeclaringClass()->name === $className) {
							 | 
						||
| 
								 | 
							
								                if ($classContextAttribute) {
							 | 
						||
| 
								 | 
							
								                    $this->setAttributeContextsForGroups($classContextAttribute, $attributeMetadata);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                foreach ($classGroups as $group) {
							 | 
						||
| 
								 | 
							
								                    $attributeMetadata->addGroup($group);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                foreach ($this->loadAttributes($property) as $attribute) {
							 | 
						||
| 
								 | 
							
								                    $loaded = true;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    if ($attribute instanceof Groups) {
							 | 
						||
| 
								 | 
							
								                        foreach ($attribute->getGroups() as $group) {
							 | 
						||
| 
								 | 
							
								                            $attributeMetadata->addGroup($group);
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                        continue;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    match (true) {
							 | 
						||
| 
								 | 
							
								                        $attribute instanceof MaxDepth => $attributeMetadata->setMaxDepth($attribute->getMaxDepth()),
							 | 
						||
| 
								 | 
							
								                        $attribute instanceof SerializedName => $attributeMetadata->setSerializedName($attribute->getSerializedName()),
							 | 
						||
| 
								 | 
							
								                        $attribute instanceof SerializedPath => $attributeMetadata->setSerializedPath($attribute->getSerializedPath()),
							 | 
						||
| 
								 | 
							
								                        $attribute instanceof Ignore => $attributeMetadata->setIgnore(true),
							 | 
						||
| 
								 | 
							
								                        $attribute instanceof Context => $this->setAttributeContextsForGroups($attribute, $attributeMetadata),
							 | 
						||
| 
								 | 
							
								                        default => null,
							 | 
						||
| 
								 | 
							
								                    };
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($reflectionClass->getMethods() as $method) {
							 | 
						||
| 
								 | 
							
								            if ($method->getDeclaringClass()->name !== $className) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (0 === stripos($method->name, 'get') && $method->getNumberOfRequiredParameters()) {
							 | 
						||
| 
								 | 
							
								                continue; /*  matches the BC behavior in `Symfony\Component\Serializer\Normalizer\ObjectNormalizer::extractAttributes` */
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $accessorOrMutator = preg_match('/^(get|is|has|set)(.+)$/i', $method->name, $matches);
							 | 
						||
| 
								 | 
							
								            if ($accessorOrMutator && !ctype_lower($matches[2][0])) {
							 | 
						||
| 
								 | 
							
								                $attributeName = lcfirst($matches[2]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (isset($attributesMetadata[$attributeName])) {
							 | 
						||
| 
								 | 
							
								                    $attributeMetadata = $attributesMetadata[$attributeName];
							 | 
						||
| 
								 | 
							
								                } else {
							 | 
						||
| 
								 | 
							
								                    $attributesMetadata[$attributeName] = $attributeMetadata = new AttributeMetadata($attributeName);
							 | 
						||
| 
								 | 
							
								                    $classMetadata->addAttributeMetadata($attributeMetadata);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($this->loadAttributes($method) as $attribute) {
							 | 
						||
| 
								 | 
							
								                if ($attribute instanceof Groups) {
							 | 
						||
| 
								 | 
							
								                    if (!$accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        throw new MappingException(\sprintf('Groups on "%s::%s()" cannot be added. Groups can only be added on methods beginning with "get", "is", "has" or "set".', $className, $method->name));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    foreach ($attribute->getGroups() as $group) {
							 | 
						||
| 
								 | 
							
								                        $attributeMetadata->addGroup($group);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ($attribute instanceof MaxDepth) {
							 | 
						||
| 
								 | 
							
								                    if (!$accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        throw new MappingException(\sprintf('MaxDepth on "%s::%s()" cannot be added. MaxDepth can only be added on methods beginning with "get", "is", "has" or "set".', $className, $method->name));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $attributeMetadata->setMaxDepth($attribute->getMaxDepth());
							 | 
						||
| 
								 | 
							
								                } elseif ($attribute instanceof SerializedName) {
							 | 
						||
| 
								 | 
							
								                    if (!$accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        throw new MappingException(\sprintf('SerializedName on "%s::%s()" cannot be added. SerializedName can only be added on methods beginning with "get", "is", "has" or "set".', $className, $method->name));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $attributeMetadata->setSerializedName($attribute->getSerializedName());
							 | 
						||
| 
								 | 
							
								                } elseif ($attribute instanceof SerializedPath) {
							 | 
						||
| 
								 | 
							
								                    if (!$accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        throw new MappingException(\sprintf('SerializedPath on "%s::%s()" cannot be added. SerializedPath can only be added on methods beginning with "get", "is", "has" or "set".', $className, $method->name));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $attributeMetadata->setSerializedPath($attribute->getSerializedPath());
							 | 
						||
| 
								 | 
							
								                } elseif ($attribute instanceof Ignore) {
							 | 
						||
| 
								 | 
							
								                    if ($accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        $attributeMetadata->setIgnore(true);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ($attribute instanceof Context) {
							 | 
						||
| 
								 | 
							
								                    if (!$accessorOrMutator) {
							 | 
						||
| 
								 | 
							
								                        throw new MappingException(\sprintf('Context on "%s::%s()" cannot be added. Context can only be added on methods beginning with "get", "is", "has" or "set".', $className, $method->name));
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $this->setAttributeContextsForGroups($attribute, $attributeMetadata);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                $loaded = true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $loaded;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function loadAttributes(\ReflectionMethod|\ReflectionClass|\ReflectionProperty $reflector): iterable
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($reflector->getAttributes() as $attribute) {
							 | 
						||
| 
								 | 
							
								            if ($this->isKnownAttribute($attribute->getName())) {
							 | 
						||
| 
								 | 
							
								                try {
							 | 
						||
| 
								 | 
							
								                    yield $attribute->newInstance();
							 | 
						||
| 
								 | 
							
								                } catch (\Error $e) {
							 | 
						||
| 
								 | 
							
								                    if (\Error::class !== $e::class) {
							 | 
						||
| 
								 | 
							
								                        throw $e;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    $on = match (true) {
							 | 
						||
| 
								 | 
							
								                        $reflector instanceof \ReflectionClass => ' on class '.$reflector->name,
							 | 
						||
| 
								 | 
							
								                        $reflector instanceof \ReflectionMethod => \sprintf(' on "%s::%s()"', $reflector->getDeclaringClass()->name, $reflector->name),
							 | 
						||
| 
								 | 
							
								                        $reflector instanceof \ReflectionProperty => \sprintf(' on "%s::$%s"', $reflector->getDeclaringClass()->name, $reflector->name),
							 | 
						||
| 
								 | 
							
								                        default => '',
							 | 
						||
| 
								 | 
							
								                    };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    throw new MappingException(\sprintf('Could not instantiate attribute "%s"%s.', $attribute->getName(), $on), 0, $e);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function setAttributeContextsForGroups(Context $attribute, AttributeMetadataInterface $attributeMetadata): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $context = $attribute->getContext();
							 | 
						||
| 
								 | 
							
								        $groups = $attribute->getGroups();
							 | 
						||
| 
								 | 
							
								        $normalizationContext = $attribute->getNormalizationContext();
							 | 
						||
| 
								 | 
							
								        $denormalizationContext = $attribute->getDenormalizationContext();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($normalizationContext || $context) {
							 | 
						||
| 
								 | 
							
								            $attributeMetadata->setNormalizationContextForGroups($normalizationContext ?: $context, $groups);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($denormalizationContext || $context) {
							 | 
						||
| 
								 | 
							
								            $attributeMetadata->setDenormalizationContextForGroups($denormalizationContext ?: $context, $groups);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    private function isKnownAttribute(string $attributeName): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach (self::KNOWN_ATTRIBUTES as $knownAttribute) {
							 | 
						||
| 
								 | 
							
								            if (is_a($attributeName, $knownAttribute, true)) {
							 | 
						||
| 
								 | 
							
								                return true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return false;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |