598 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
		
		
			
		
	
	
			598 lines
		
	
	
		
			25 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\Serializer\Exception\CircularReferenceException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\InvalidArgumentException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\LogicException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\MissingConstructorArgumentsException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\NotNormalizableValueException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Exception\RuntimeException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\AttributeMetadataInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\Mapping\Factory\ClassMetadataFactoryInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\NameConverter\NameConverterInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\SerializerAwareInterface;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\Serializer\SerializerAwareTrait;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * Normalizer implementation.
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @author Kévin Dunglas <dunglas@gmail.com>
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								abstract class AbstractNormalizer implements NormalizerInterface, DenormalizerInterface, SerializerAwareInterface
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    use ObjectToPopulateTrait;
							 | 
						||
| 
								 | 
							
								    use SerializerAwareTrait;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /* constants to configure the context */
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * How many loops of circular reference to allow while normalizing.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * The default value of 1 means that when we encounter the same object a
							 | 
						||
| 
								 | 
							
								     * second time, we consider that a circular reference.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * You can raise this value for special cases, e.g. in combination with the
							 | 
						||
| 
								 | 
							
								     * max depth setting of the object normalizer.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const CIRCULAR_REFERENCE_LIMIT = 'circular_reference_limit';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Instead of creating a new instance of an object, update the specified object.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * If you have a nested structure, child objects will be overwritten with
							 | 
						||
| 
								 | 
							
								     * new instances unless you set DEEP_OBJECT_TO_POPULATE to true.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const OBJECT_TO_POPULATE = 'object_to_populate';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Only (de)normalize attributes that are in the specified groups.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const GROUPS = 'groups';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Limit (de)normalize to the specified names.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * For nested structures, this list needs to reflect the object tree.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const ATTRIBUTES = 'attributes';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * If ATTRIBUTES are specified, and the source has fields that are not part of that list,
							 | 
						||
| 
								 | 
							
								     * either ignore those attributes (true) or throw an ExtraAttributesException (false).
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const ALLOW_EXTRA_ATTRIBUTES = 'allow_extra_attributes';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Hashmap of default values for constructor arguments.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * The names need to match the parameter names in the constructor arguments.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const DEFAULT_CONSTRUCTOR_ARGUMENTS = 'default_constructor_arguments';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Hashmap of field name => callable to (de)normalize this field.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * The callable is called if the field is encountered with the arguments:
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * - mixed         $attributeValue value of this field
							 | 
						||
| 
								 | 
							
								     * - object|string $object         the whole object being normalized or the object's class being denormalized
							 | 
						||
| 
								 | 
							
								     * - string        $attributeName  name of the attribute being (de)normalized
							 | 
						||
| 
								 | 
							
								     * - string        $format         the requested format
							 | 
						||
| 
								 | 
							
								     * - array         $context        the serialization context
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const CALLBACKS = 'callbacks';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Handler to call when a circular reference has been detected.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * If you specify no handler, a CircularReferenceException is thrown.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * The method will be called with ($object, $format, $context) and its
							 | 
						||
| 
								 | 
							
								     * return value is returned as the result of the normalize call.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const CIRCULAR_REFERENCE_HANDLER = 'circular_reference_handler';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Skip the specified attributes when normalizing an object tree.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * This list is applied to each element of nested structures.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * Note: The behaviour for nested structures is different from ATTRIBUTES
							 | 
						||
| 
								 | 
							
								     * for historical reason. Aligning the behaviour would be a BC break.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const IGNORED_ATTRIBUTES = 'ignored_attributes';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Require all properties to be listed in the input instead of falling
							 | 
						||
| 
								 | 
							
								     * back to null for nullable ones.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const REQUIRE_ALL_PROPERTIES = 'require_all_properties';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Flag to control whether a non-boolean value should be filtered using the
							 | 
						||
| 
								 | 
							
								     * filter_var function with the {@see https://www.php.net/manual/fr/filter.filters.validate.php}
							 | 
						||
| 
								 | 
							
								     * \FILTER_VALIDATE_BOOL filter before casting it to a boolean.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * "0", "false", "off", "no" and "" will be cast to false.
							 | 
						||
| 
								 | 
							
								     * "1", "true", "on" and "yes" will be cast to true.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public const FILTER_BOOL = 'filter_bool';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected const CIRCULAR_REFERENCE_LIMIT_COUNTERS = 'circular_reference_limit_counters';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    protected array $defaultContext = [
							 | 
						||
| 
								 | 
							
								        self::ALLOW_EXTRA_ATTRIBUTES => true,
							 | 
						||
| 
								 | 
							
								        self::CIRCULAR_REFERENCE_HANDLER => null,
							 | 
						||
| 
								 | 
							
								        self::CIRCULAR_REFERENCE_LIMIT => 1,
							 | 
						||
| 
								 | 
							
								        self::IGNORED_ATTRIBUTES => [],
							 | 
						||
| 
								 | 
							
								    ];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Sets the {@link ClassMetadataFactoryInterface} to use.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    public function __construct(
							 | 
						||
| 
								 | 
							
								        protected ?ClassMetadataFactoryInterface $classMetadataFactory = null,
							 | 
						||
| 
								 | 
							
								        protected ?NameConverterInterface $nameConverter = null,
							 | 
						||
| 
								 | 
							
								        array $defaultContext = [],
							 | 
						||
| 
								 | 
							
								    ) {
							 | 
						||
| 
								 | 
							
								        $this->defaultContext = array_merge($this->defaultContext, $defaultContext);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $this->validateCallbackContext($this->defaultContext, 'default');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (isset($this->defaultContext[self::CIRCULAR_REFERENCE_HANDLER]) && !\is_callable($this->defaultContext[self::CIRCULAR_REFERENCE_HANDLER])) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException(\sprintf('Invalid callback found in the "%s" default context option.', self::CIRCULAR_REFERENCE_HANDLER));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Detects if the configured circular reference limit is reached.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @throws CircularReferenceException
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function isCircularReference(object $object, array &$context): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $objectId = spl_object_id($object);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $circularReferenceLimit = $context[self::CIRCULAR_REFERENCE_LIMIT] ?? $this->defaultContext[self::CIRCULAR_REFERENCE_LIMIT];
							 | 
						||
| 
								 | 
							
								        if (isset($context[self::CIRCULAR_REFERENCE_LIMIT_COUNTERS][$objectId])) {
							 | 
						||
| 
								 | 
							
								            if ($context[self::CIRCULAR_REFERENCE_LIMIT_COUNTERS][$objectId] >= $circularReferenceLimit) {
							 | 
						||
| 
								 | 
							
								                unset($context[self::CIRCULAR_REFERENCE_LIMIT_COUNTERS][$objectId]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            ++$context[self::CIRCULAR_REFERENCE_LIMIT_COUNTERS][$objectId];
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            $context[self::CIRCULAR_REFERENCE_LIMIT_COUNTERS][$objectId] = 1;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return false;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Handles a circular reference.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * If a circular reference handler is set, it will be called. Otherwise, a
							 | 
						||
| 
								 | 
							
								     * {@class CircularReferenceException} will be thrown.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @final
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @throws CircularReferenceException
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function handleCircularReference(object $object, ?string $format = null, array $context = []): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $circularReferenceHandler = $context[self::CIRCULAR_REFERENCE_HANDLER] ?? $this->defaultContext[self::CIRCULAR_REFERENCE_HANDLER];
							 | 
						||
| 
								 | 
							
								        if ($circularReferenceHandler) {
							 | 
						||
| 
								 | 
							
								            return $circularReferenceHandler($object, $format, $context);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        throw new CircularReferenceException(\sprintf('A circular reference has been detected when serializing the object of class "%s" (configured limit: %d).', get_debug_type($object), $context[self::CIRCULAR_REFERENCE_LIMIT] ?? $this->defaultContext[self::CIRCULAR_REFERENCE_LIMIT]));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Gets attributes to normalize using groups.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @param bool $attributesAsString If false, return an array of {@link AttributeMetadataInterface}
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @return string[]|AttributeMetadataInterface[]|bool
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @throws LogicException if the 'allow_extra_attributes' context variable is false and no class metadata factory is provided
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function getAllowedAttributes(string|object $classOrObject, array $context, bool $attributesAsString = false): array|bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $allowExtraAttributes = $context[self::ALLOW_EXTRA_ATTRIBUTES] ?? $this->defaultContext[self::ALLOW_EXTRA_ATTRIBUTES];
							 | 
						||
| 
								 | 
							
								        if (!$this->classMetadataFactory) {
							 | 
						||
| 
								 | 
							
								            if (!$allowExtraAttributes) {
							 | 
						||
| 
								 | 
							
								                throw new LogicException(\sprintf('A class metadata factory must be provided in the constructor when setting "%s" to false.', self::ALLOW_EXTRA_ATTRIBUTES));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $groups = $this->getGroups($context);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $allowedAttributes = [];
							 | 
						||
| 
								 | 
							
								        $ignoreUsed = false;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($this->classMetadataFactory->getMetadataFor($classOrObject)->getAttributesMetadata() as $attributeMetadata) {
							 | 
						||
| 
								 | 
							
								            if ($ignore = $attributeMetadata->isIgnored()) {
							 | 
						||
| 
								 | 
							
								                $ignoreUsed = true;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            // If you update this check, update accordingly the one in Symfony\Component\PropertyInfo\Extractor\SerializerExtractor::getProperties()
							 | 
						||
| 
								 | 
							
								            if (
							 | 
						||
| 
								 | 
							
								                !$ignore
							 | 
						||
| 
								 | 
							
								                && ([] === $groups || \in_array('*', $groups, true) || array_intersect($attributeMetadata->getGroups(), $groups))
							 | 
						||
| 
								 | 
							
								                && $this->isAllowedAttribute($classOrObject, $name = $attributeMetadata->getName(), null, $context)
							 | 
						||
| 
								 | 
							
								            ) {
							 | 
						||
| 
								 | 
							
								                $allowedAttributes[] = $attributesAsString ? $name : $attributeMetadata;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$ignoreUsed && [] === $groups && $allowExtraAttributes) {
							 | 
						||
| 
								 | 
							
								            // Backward Compatibility with the code using this method written before the introduction of @Ignore
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $allowedAttributes;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    protected function getGroups(array $context): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $groups = $context[self::GROUPS] ?? $this->defaultContext[self::GROUPS] ?? [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return \is_scalar($groups) ? (array) $groups : $groups;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Is this attribute allowed?
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function isAllowedAttribute(object|string $classOrObject, string $attribute, ?string $format = null, array $context = []): bool
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $ignoredAttributes = $context[self::IGNORED_ATTRIBUTES] ?? $this->defaultContext[self::IGNORED_ATTRIBUTES];
							 | 
						||
| 
								 | 
							
								        if (\in_array($attribute, $ignoredAttributes, true)) {
							 | 
						||
| 
								 | 
							
								            return false;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $attributes = $context[self::ATTRIBUTES] ?? $this->defaultContext[self::ATTRIBUTES] ?? null;
							 | 
						||
| 
								 | 
							
								        if (isset($attributes[$attribute])) {
							 | 
						||
| 
								 | 
							
								            // Nested attributes
							 | 
						||
| 
								 | 
							
								            return true;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (\is_array($attributes)) {
							 | 
						||
| 
								 | 
							
								            return \in_array($attribute, $attributes, true);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return true;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Normalizes the given data to an array. It's particularly useful during
							 | 
						||
| 
								 | 
							
								     * the denormalization process.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function prepareForDenormalization(mixed $data): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return (array) $data;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Returns the method to use to construct an object. This method must be either
							 | 
						||
| 
								 | 
							
								     * the object constructor or static.
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function getConstructor(array &$data, string $class, array &$context, \ReflectionClass $reflectionClass, array|bool $allowedAttributes): ?\ReflectionMethod
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        return $reflectionClass->getConstructor();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Instantiates an object using constructor parameters when needed.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * This method also allows to denormalize data into an existing object if
							 | 
						||
| 
								 | 
							
								     * it is present in the context with the object_to_populate. This object
							 | 
						||
| 
								 | 
							
								     * is removed from the context before being returned to avoid side effects
							 | 
						||
| 
								 | 
							
								     * when recursively normalizing an object graph.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @throws RuntimeException
							 | 
						||
| 
								 | 
							
								     * @throws MissingConstructorArgumentsException
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function instantiateObject(array &$data, string $class, array &$context, \ReflectionClass $reflectionClass, array|bool $allowedAttributes, ?string $format = null): object
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (null !== $object = $this->extractObjectToPopulate($class, $context, self::OBJECT_TO_POPULATE)) {
							 | 
						||
| 
								 | 
							
								            unset($context[self::OBJECT_TO_POPULATE]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return $object;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        // clean up even if no match
							 | 
						||
| 
								 | 
							
								        unset($context[static::OBJECT_TO_POPULATE]);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $constructor = $this->getConstructor($data, $class, $context, $reflectionClass, $allowedAttributes);
							 | 
						||
| 
								 | 
							
								        if ($constructor) {
							 | 
						||
| 
								 | 
							
								            $context['has_constructor'] = true;
							 | 
						||
| 
								 | 
							
								            if (true !== $constructor->isPublic()) {
							 | 
						||
| 
								 | 
							
								                return $reflectionClass->newInstanceWithoutConstructor();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $constructorParameters = $constructor->getParameters();
							 | 
						||
| 
								 | 
							
								            $missingConstructorArguments = [];
							 | 
						||
| 
								 | 
							
								            $params = [];
							 | 
						||
| 
								 | 
							
								            $unsetKeys = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            foreach ($constructorParameters as $constructorParameter) {
							 | 
						||
| 
								 | 
							
								                $paramName = $constructorParameter->name;
							 | 
						||
| 
								 | 
							
								                $attributeContext = $this->getAttributeDenormalizationContext($class, $paramName, $context);
							 | 
						||
| 
								 | 
							
								                $key = $this->nameConverter ? $this->nameConverter->normalize($paramName, $class, $format, $context) : $paramName;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                $allowed = false === $allowedAttributes || \in_array($paramName, $allowedAttributes, true);
							 | 
						||
| 
								 | 
							
								                $ignored = !$this->isAllowedAttribute($class, $paramName, $format, $context);
							 | 
						||
| 
								 | 
							
								                if ($constructorParameter->isVariadic()) {
							 | 
						||
| 
								 | 
							
								                    if ($allowed && !$ignored && (isset($data[$key]) || \array_key_exists($key, $data))) {
							 | 
						||
| 
								 | 
							
								                        if (!\is_array($data[$key])) {
							 | 
						||
| 
								 | 
							
								                            throw new RuntimeException(\sprintf('Cannot create an instance of "%s" from serialized data because the variadic parameter "%s" can only accept an array.', $class, $constructorParameter->name));
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                        $variadicParameters = [];
							 | 
						||
| 
								 | 
							
								                        foreach ($data[$key] as $parameterKey => $parameterData) {
							 | 
						||
| 
								 | 
							
								                            try {
							 | 
						||
| 
								 | 
							
								                                $variadicParameters[$parameterKey] = $this->denormalizeParameter($reflectionClass, $constructorParameter, $paramName, $parameterData, $attributeContext, $format);
							 | 
						||
| 
								 | 
							
								                            } catch (NotNormalizableValueException $exception) {
							 | 
						||
| 
								 | 
							
								                                if (!isset($context['not_normalizable_value_exceptions'])) {
							 | 
						||
| 
								 | 
							
								                                    throw $exception;
							 | 
						||
| 
								 | 
							
								                                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                                $context['not_normalizable_value_exceptions'][] = $exception;
							 | 
						||
| 
								 | 
							
								                                $params[$paramName] = $parameterData;
							 | 
						||
| 
								 | 
							
								                            }
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                        $params = array_merge(array_values($params), $variadicParameters);
							 | 
						||
| 
								 | 
							
								                        $unsetKeys[] = $key;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                } elseif ($allowed && !$ignored && (isset($data[$key]) || \array_key_exists($key, $data))) {
							 | 
						||
| 
								 | 
							
								                    $parameterData = $data[$key];
							 | 
						||
| 
								 | 
							
								                    if (null === $parameterData && $constructorParameter->allowsNull()) {
							 | 
						||
| 
								 | 
							
								                        $params[$paramName] = null;
							 | 
						||
| 
								 | 
							
								                        $unsetKeys[] = $key;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                        continue;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    try {
							 | 
						||
| 
								 | 
							
								                        $params[$paramName] = $this->denormalizeParameter($reflectionClass, $constructorParameter, $paramName, $parameterData, $attributeContext, $format);
							 | 
						||
| 
								 | 
							
								                    } catch (NotNormalizableValueException $exception) {
							 | 
						||
| 
								 | 
							
								                        if (!isset($context['not_normalizable_value_exceptions'])) {
							 | 
						||
| 
								 | 
							
								                            throw $exception;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                        $context['not_normalizable_value_exceptions'][] = $exception;
							 | 
						||
| 
								 | 
							
								                        $params[$paramName] = $parameterData;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $unsetKeys[] = $key;
							 | 
						||
| 
								 | 
							
								                } elseif (\array_key_exists($key, $context[static::DEFAULT_CONSTRUCTOR_ARGUMENTS][$class] ?? [])) {
							 | 
						||
| 
								 | 
							
								                    $params[$paramName] = $context[static::DEFAULT_CONSTRUCTOR_ARGUMENTS][$class][$key];
							 | 
						||
| 
								 | 
							
								                } elseif (\array_key_exists($key, $this->defaultContext[self::DEFAULT_CONSTRUCTOR_ARGUMENTS][$class] ?? [])) {
							 | 
						||
| 
								 | 
							
								                    $params[$paramName] = $this->defaultContext[self::DEFAULT_CONSTRUCTOR_ARGUMENTS][$class][$key];
							 | 
						||
| 
								 | 
							
								                } elseif ($constructorParameter->isDefaultValueAvailable()) {
							 | 
						||
| 
								 | 
							
								                    $params[$paramName] = $constructorParameter->getDefaultValue();
							 | 
						||
| 
								 | 
							
								                } elseif (!($context[self::REQUIRE_ALL_PROPERTIES] ?? $this->defaultContext[self::REQUIRE_ALL_PROPERTIES] ?? false) && $constructorParameter->hasType() && $constructorParameter->getType()->allowsNull()) {
							 | 
						||
| 
								 | 
							
								                    $params[$paramName] = null;
							 | 
						||
| 
								 | 
							
								                } else {
							 | 
						||
| 
								 | 
							
								                    if (!isset($context['not_normalizable_value_exceptions'])) {
							 | 
						||
| 
								 | 
							
								                        $missingConstructorArguments[] = $constructorParameter->name;
							 | 
						||
| 
								 | 
							
								                        continue;
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $constructorParameterTypes = [];
							 | 
						||
| 
								 | 
							
								                    $reflectionType = $constructorParameter->getType();
							 | 
						||
| 
								 | 
							
								                    if ($reflectionType instanceof \ReflectionUnionType) {
							 | 
						||
| 
								 | 
							
								                        foreach ($reflectionType->getTypes() as $reflectionType) {
							 | 
						||
| 
								 | 
							
								                            $constructorParameterTypes[] = (string) $reflectionType;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                    } elseif ($reflectionType instanceof \ReflectionType) {
							 | 
						||
| 
								 | 
							
								                        $constructorParameterTypes[] = (string) $reflectionType;
							 | 
						||
| 
								 | 
							
								                    } else {
							 | 
						||
| 
								 | 
							
								                        $constructorParameterTypes[] = 'unknown';
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                    $exception = NotNormalizableValueException::createForUnexpectedDataType(
							 | 
						||
| 
								 | 
							
								                        \sprintf('Failed to create object because the class misses the "%s" property.', $constructorParameter->name),
							 | 
						||
| 
								 | 
							
								                        null,
							 | 
						||
| 
								 | 
							
								                        $constructorParameterTypes,
							 | 
						||
| 
								 | 
							
								                        $attributeContext['deserialization_path'] ?? null,
							 | 
						||
| 
								 | 
							
								                        true
							 | 
						||
| 
								 | 
							
								                    );
							 | 
						||
| 
								 | 
							
								                    $context['not_normalizable_value_exceptions'][] = $exception;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if ($missingConstructorArguments) {
							 | 
						||
| 
								 | 
							
								                throw new MissingConstructorArgumentsException(\sprintf('Cannot create an instance of "%s" from serialized data because its constructor requires the following parameters to be present : "$%s".', $class, implode('", "$', $missingConstructorArguments)), 0, null, $missingConstructorArguments, $class);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (!$constructor->isConstructor()) {
							 | 
						||
| 
								 | 
							
								                $instance = $constructor->invokeArgs(null, $params);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // do not set a parameter that has been set in the constructor
							 | 
						||
| 
								 | 
							
								                foreach ($unsetKeys as $key) {
							 | 
						||
| 
								 | 
							
								                    unset($data[$key]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return $instance;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                $instance = $reflectionClass->newInstanceArgs($params);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                // do not set a parameter that has been set in the constructor
							 | 
						||
| 
								 | 
							
								                foreach ($unsetKeys as $key) {
							 | 
						||
| 
								 | 
							
								                    unset($data[$key]);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return $instance;
							 | 
						||
| 
								 | 
							
								            } catch (\TypeError $e) {
							 | 
						||
| 
								 | 
							
								                if (!isset($context['not_normalizable_value_exceptions'])) {
							 | 
						||
| 
								 | 
							
								                    throw $e;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return $reflectionClass->newInstanceWithoutConstructor();
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        unset($context['has_constructor']);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!$reflectionClass->isInstantiable()) {
							 | 
						||
| 
								 | 
							
								            throw NotNormalizableValueException::createForUnexpectedDataType(\sprintf('Failed to create object because the class "%s" is not instantiable.', $class), $data, ['unknown'], $context['deserialization_path'] ?? null);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return new $class();
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function denormalizeParameter(\ReflectionClass $class, \ReflectionParameter $parameter, string $parameterName, mixed $parameterData, array $context, ?string $format = null): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            if (($parameterType = $parameter->getType()) instanceof \ReflectionNamedType && !$parameterType->isBuiltin()) {
							 | 
						||
| 
								 | 
							
								                $parameterClass = $parameterType->getName();
							 | 
						||
| 
								 | 
							
								                new \ReflectionClass($parameterClass); // throws a \ReflectionException if the class doesn't exist
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                if (!$this->serializer instanceof DenormalizerInterface) {
							 | 
						||
| 
								 | 
							
								                    throw new LogicException(\sprintf('Cannot create an instance of "%s" from serialized data because the serializer inject in "%s" is not a denormalizer.', $parameterClass, static::class));
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                $parameterData = $this->serializer->denormalize($parameterData, $parameterClass, $format, $this->createChildContext($context, $parameterName, $format));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        } catch (\ReflectionException $e) {
							 | 
						||
| 
								 | 
							
								            throw new RuntimeException(\sprintf('Could not determine the class of the parameter "%s".', $parameterName), 0, $e);
							 | 
						||
| 
								 | 
							
								        } catch (MissingConstructorArgumentsException $e) {
							 | 
						||
| 
								 | 
							
								            if (!$parameter->getType()->allowsNull()) {
							 | 
						||
| 
								 | 
							
								                throw $e;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return null;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        $parameterData = $this->applyCallbacks($parameterData, $class->getName(), $parameterName, $format, $context);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $this->applyFilterBool($parameter, $parameterData, $context);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function createChildContext(array $parentContext, string $attribute, ?string $format): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (isset($parentContext[self::ATTRIBUTES][$attribute])) {
							 | 
						||
| 
								 | 
							
								            $parentContext[self::ATTRIBUTES] = $parentContext[self::ATTRIBUTES][$attribute];
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            unset($parentContext[self::ATTRIBUTES]);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $parentContext;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Validate callbacks set in context.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @param string $contextType Used to specify which context is invalid in exceptions
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @throws InvalidArgumentException
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    final protected function validateCallbackContext(array $context, string $contextType = ''): void
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!isset($context[self::CALLBACKS])) {
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!\is_array($context[self::CALLBACKS])) {
							 | 
						||
| 
								 | 
							
								            throw new InvalidArgumentException(\sprintf('The "%s"%s context option must be an array of callables.', self::CALLBACKS, '' !== $contextType ? " $contextType" : ''));
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        foreach ($context[self::CALLBACKS] as $attribute => $callback) {
							 | 
						||
| 
								 | 
							
								            if (!\is_callable($callback)) {
							 | 
						||
| 
								 | 
							
								                throw new InvalidArgumentException(\sprintf('Invalid callback found for attribute "%s" in the "%s"%s context option.', $attribute, self::CALLBACKS, '' !== $contextType ? " $contextType" : ''));
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    final protected function applyCallbacks(mixed $value, object|string $object, string $attribute, ?string $format, array $context): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        /**
							 | 
						||
| 
								 | 
							
								         * @var callable|null
							 | 
						||
| 
								 | 
							
								         */
							 | 
						||
| 
								 | 
							
								        $callback = $context[self::CALLBACKS][$attribute] ?? $this->defaultContext[self::CALLBACKS][$attribute] ?? null;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $callback ? $callback($value, $object, $attribute, $format, $context) : $value;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    final protected function applyFilterBool(\ReflectionParameter $parameter, mixed $value, array $context): mixed
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!($context[self::FILTER_BOOL] ?? false)) {
							 | 
						||
| 
								 | 
							
								            return $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (!($parameterType = $parameter->getType()) instanceof \ReflectionNamedType || 'bool' !== $parameterType->getName()) {
							 | 
						||
| 
								 | 
							
								            return $value;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return filter_var($value, \FILTER_VALIDATE_BOOL, \FILTER_NULL_ON_FAILURE) ?? $value;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Computes the normalization context merged with current one. Metadata always wins over global context, as more specific.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function getAttributeNormalizationContext(object $object, string $attribute, array $context): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (null === $metadata = $this->getAttributeMetadata($object, $attribute)) {
							 | 
						||
| 
								 | 
							
								            return $context;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return array_merge($context, $metadata->getNormalizationContextForGroups($this->getGroups($context)));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * Computes the denormalization context merged with current one. Metadata always wins over global context, as more specific.
							 | 
						||
| 
								 | 
							
								     *
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function getAttributeDenormalizationContext(string $class, string $attribute, array $context): array
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $context['deserialization_path'] = ($context['deserialization_path'] ?? false) ? $context['deserialization_path'].'.'.$attribute : $attribute;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null === $metadata = $this->getAttributeMetadata($class, $attribute)) {
							 | 
						||
| 
								 | 
							
								            return $context;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return array_merge($context, $metadata->getDenormalizationContextForGroups($this->getGroups($context)));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    /**
							 | 
						||
| 
								 | 
							
								     * @internal
							 | 
						||
| 
								 | 
							
								     */
							 | 
						||
| 
								 | 
							
								    protected function getAttributeMetadata(object|string $objectOrClass, string $attribute): ?AttributeMetadataInterface
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!$this->classMetadataFactory) {
							 | 
						||
| 
								 | 
							
								            return null;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $this->classMetadataFactory->getMetadataFor($objectOrClass)->getAttributesMetadata()[$attribute] ?? null;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |