* * 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\UninitializedPropertyException; /** * Converts between objects and arrays by mapping properties. * * The normalization process looks for all the object's properties (public and private). * The result is a map from property names to property values. Property values * are normalized through the serializer. * * The denormalization first looks at the constructor of the given class to see * if any of the parameters have the same name as one of the properties. The * constructor is then called with all parameters or an exception is thrown if * any required parameters were not present as properties. Then the denormalizer * walks through the given map of property names to property values to see if a * property with the corresponding name exists. If found, the property gets the value. * * @author Matthieu Napoli * @author Kévin Dunglas */ class PropertyNormalizer extends AbstractObjectNormalizer { /** * {@inheritdoc} */ public function supportsNormalization(mixed $data, string $format = null): bool { return parent::supportsNormalization($data, $format) && $this->supports(\get_class($data)); } /** * {@inheritdoc} */ public function supportsDenormalization(mixed $data, string $type, string $format = null): bool { return parent::supportsDenormalization($data, $type, $format) && $this->supports($type); } /** * {@inheritdoc} */ public function hasCacheableSupportsMethod(): bool { return __CLASS__ === static::class; } /** * Checks if the given class has any non-static property. */ private function supports(string $class): bool { $class = new \ReflectionClass($class); // We look for at least one non-static property do { foreach ($class->getProperties() as $property) { if (!$property->isStatic()) { return true; } } } while ($class = $class->getParentClass()); return false; } /** * {@inheritdoc} */ protected function isAllowedAttribute(object|string $classOrObject, string $attribute, string $format = null, array $context = []): bool { if (!parent::isAllowedAttribute($classOrObject, $attribute, $format, $context)) { return false; } try { $reflectionProperty = $this->getReflectionProperty($classOrObject, $attribute); if ($reflectionProperty->isStatic()) { return false; } } catch (\ReflectionException $reflectionException) { return false; } return true; } /** * {@inheritdoc} */ protected function extractAttributes(object $object, string $format = null, array $context = []): array { $reflectionObject = new \ReflectionObject($object); $attributes = []; do { foreach ($reflectionObject->getProperties() as $property) { if (!$this->isAllowedAttribute($reflectionObject->getName(), $property->name, $format, $context)) { continue; } $attributes[] = $property->name; } } while ($reflectionObject = $reflectionObject->getParentClass()); return array_unique($attributes); } /** * {@inheritdoc} */ protected function getAttributeValue(object $object, string $attribute, string $format = null, array $context = []): mixed { try { $reflectionProperty = $this->getReflectionProperty($object, $attribute); } catch (\ReflectionException $reflectionException) { return null; } // Override visibility if (!$reflectionProperty->isPublic()) { $reflectionProperty->setAccessible(true); } if ($reflectionProperty->hasType()) { return $reflectionProperty->getValue($object); } if (!method_exists($object, '__get') && !isset($object->$attribute)) { $propertyValues = (array) $object; if (($reflectionProperty->isPublic() && !\array_key_exists($reflectionProperty->name, $propertyValues)) || ($reflectionProperty->isProtected() && !\array_key_exists("\0*\0{$reflectionProperty->name}", $propertyValues)) || ($reflectionProperty->isPrivate() && !\array_key_exists("\0{$reflectionProperty->class}\0{$reflectionProperty->name}", $propertyValues)) ) { throw new UninitializedPropertyException(sprintf('The property "%s::$%s" is not initialized.', \get_class($object), $reflectionProperty->name)); } } return $reflectionProperty->getValue($object); } /** * {@inheritdoc} */ protected function setAttributeValue(object $object, string $attribute, mixed $value, string $format = null, array $context = []) { try { $reflectionProperty = $this->getReflectionProperty($object, $attribute); } catch (\ReflectionException $reflectionException) { return; } if ($reflectionProperty->isStatic()) { return; } // Override visibility if (!$reflectionProperty->isPublic()) { $reflectionProperty->setAccessible(true); } $reflectionProperty->setValue($object, $value); } /** * @throws \ReflectionException */ private function getReflectionProperty(string|object $classOrObject, string $attribute): \ReflectionProperty { $reflectionClass = new \ReflectionClass($classOrObject); while (true) { try { return $reflectionClass->getProperty($attribute); } catch (\ReflectionException $e) { if (!$reflectionClass = $reflectionClass->getParentClass()) { throw $e; } } } } }