167 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			167 lines
		
	
	
		
			6.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\PropertyAccess\Exception\NoSuchPropertyException;
 | |
| use Symfony\Component\PropertyAccess\PropertyAccess;
 | |
| use Symfony\Component\PropertyAccess\PropertyAccessorInterface;
 | |
| use Symfony\Component\PropertyInfo\PropertyTypeExtractorInterface;
 | |
| use Symfony\Component\Serializer\Exception\LogicException;
 | |
| use Symfony\Component\Serializer\Mapping\AttributeMetadata;
 | |
| use Symfony\Component\Serializer\Mapping\ClassDiscriminatorResolverInterface;
 | |
| use Symfony\Component\Serializer\Mapping\Factory\ClassMetadataFactoryInterface;
 | |
| use Symfony\Component\Serializer\NameConverter\NameConverterInterface;
 | |
| 
 | |
| /**
 | |
|  * Converts between objects and arrays using the PropertyAccess component.
 | |
|  *
 | |
|  * @author Kévin Dunglas <dunglas@gmail.com>
 | |
|  */
 | |
| class ObjectNormalizer extends AbstractObjectNormalizer
 | |
| {
 | |
|     protected $propertyAccessor;
 | |
| 
 | |
|     private $discriminatorCache = [];
 | |
| 
 | |
|     private $objectClassResolver;
 | |
| 
 | |
|     public function __construct(ClassMetadataFactoryInterface $classMetadataFactory = null, NameConverterInterface $nameConverter = null, PropertyAccessorInterface $propertyAccessor = null, PropertyTypeExtractorInterface $propertyTypeExtractor = null, ClassDiscriminatorResolverInterface $classDiscriminatorResolver = null, callable $objectClassResolver = null, array $defaultContext = [])
 | |
|     {
 | |
|         if (!class_exists(PropertyAccess::class)) {
 | |
|             throw new LogicException('The ObjectNormalizer class requires the "PropertyAccess" component. Install "symfony/property-access" to use it.');
 | |
|         }
 | |
| 
 | |
|         parent::__construct($classMetadataFactory, $nameConverter, $propertyTypeExtractor, $classDiscriminatorResolver, $objectClassResolver, $defaultContext);
 | |
| 
 | |
|         $this->propertyAccessor = $propertyAccessor ?: PropertyAccess::createPropertyAccessor();
 | |
| 
 | |
|         $this->objectClassResolver = $objectClassResolver ?? function ($class) {
 | |
|             return \is_object($class) ? $class::class : $class;
 | |
|         };
 | |
|     }
 | |
| 
 | |
|     public function hasCacheableSupportsMethod(): bool
 | |
|     {
 | |
|         return __CLASS__ === static::class;
 | |
|     }
 | |
| 
 | |
|     protected function extractAttributes(object $object, string $format = null, array $context = []): array
 | |
|     {
 | |
|         if (\stdClass::class === $object::class) {
 | |
|             return array_keys((array) $object);
 | |
|         }
 | |
| 
 | |
|         // If not using groups, detect manually
 | |
|         $attributes = [];
 | |
| 
 | |
|         // methods
 | |
|         $class = ($this->objectClassResolver)($object);
 | |
|         $reflClass = new \ReflectionClass($class);
 | |
| 
 | |
|         foreach ($reflClass->getMethods(\ReflectionMethod::IS_PUBLIC) as $reflMethod) {
 | |
|             if (
 | |
|                 0 !== $reflMethod->getNumberOfRequiredParameters() ||
 | |
|                 $reflMethod->isStatic() ||
 | |
|                 $reflMethod->isConstructor() ||
 | |
|                 $reflMethod->isDestructor()
 | |
|             ) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             $name = $reflMethod->name;
 | |
|             $attributeName = null;
 | |
| 
 | |
|             if (str_starts_with($name, 'get') || str_starts_with($name, 'has') || str_starts_with($name, 'can')) {
 | |
|                 // getters, hassers and canners
 | |
|                 $attributeName = substr($name, 3);
 | |
| 
 | |
|                 if (!$reflClass->hasProperty($attributeName)) {
 | |
|                     $attributeName = lcfirst($attributeName);
 | |
|                 }
 | |
|             } elseif (str_starts_with($name, 'is')) {
 | |
|                 // issers
 | |
|                 $attributeName = substr($name, 2);
 | |
| 
 | |
|                 if (!$reflClass->hasProperty($attributeName)) {
 | |
|                     $attributeName = lcfirst($attributeName);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (null !== $attributeName && $this->isAllowedAttribute($object, $attributeName, $format, $context)) {
 | |
|                 $attributes[$attributeName] = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // properties
 | |
|         foreach ($reflClass->getProperties() as $reflProperty) {
 | |
|             if (!$reflProperty->isPublic()) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             if ($reflProperty->isStatic() || !$this->isAllowedAttribute($object, $reflProperty->name, $format, $context)) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             $attributes[$reflProperty->name] = true;
 | |
|         }
 | |
| 
 | |
|         return array_keys($attributes);
 | |
|     }
 | |
| 
 | |
|     protected function getAttributeValue(object $object, string $attribute, string $format = null, array $context = []): mixed
 | |
|     {
 | |
|         $cacheKey = $object::class;
 | |
|         if (!\array_key_exists($cacheKey, $this->discriminatorCache)) {
 | |
|             $this->discriminatorCache[$cacheKey] = null;
 | |
|             if (null !== $this->classDiscriminatorResolver) {
 | |
|                 $mapping = $this->classDiscriminatorResolver->getMappingForMappedObject($object);
 | |
|                 $this->discriminatorCache[$cacheKey] = $mapping?->getTypeProperty();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $attribute === $this->discriminatorCache[$cacheKey] ? $this->classDiscriminatorResolver->getTypeForMappedObject($object) : $this->propertyAccessor->getValue($object, $attribute);
 | |
|     }
 | |
| 
 | |
|     protected function setAttributeValue(object $object, string $attribute, mixed $value, string $format = null, array $context = [])
 | |
|     {
 | |
|         try {
 | |
|             $this->propertyAccessor->setValue($object, $attribute, $value);
 | |
|         } catch (NoSuchPropertyException) {
 | |
|             // Properties not found are ignored
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     protected function getAllowedAttributes(string|object $classOrObject, array $context, bool $attributesAsString = false): array|bool
 | |
|     {
 | |
|         if (false === $allowedAttributes = parent::getAllowedAttributes($classOrObject, $context, $attributesAsString)) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         if (null !== $this->classDiscriminatorResolver) {
 | |
|             $class = \is_object($classOrObject) ? $classOrObject::class : $classOrObject;
 | |
|             if (null !== $discriminatorMapping = $this->classDiscriminatorResolver->getMappingForMappedObject($classOrObject)) {
 | |
|                 $allowedAttributes[] = $attributesAsString ? $discriminatorMapping->getTypeProperty() : new AttributeMetadata($discriminatorMapping->getTypeProperty());
 | |
|             }
 | |
| 
 | |
|             if (null !== $discriminatorMapping = $this->classDiscriminatorResolver->getMappingForClass($class)) {
 | |
|                 $attributes = [];
 | |
|                 foreach ($discriminatorMapping->getTypesMapping() as $mappedClass) {
 | |
|                     $attributes[] = parent::getAllowedAttributes($mappedClass, $context, $attributesAsString);
 | |
|                 }
 | |
|                 $allowedAttributes = array_merge($allowedAttributes, ...$attributes);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $allowedAttributes;
 | |
|     }
 | |
| }
 |