diff --git a/src/Mapping/ClassMetadata.php b/src/Mapping/ClassMetadata.php index ecf2fef5cb2..e524b81df53 100644 --- a/src/Mapping/ClassMetadata.php +++ b/src/Mapping/ClassMetadata.php @@ -18,6 +18,7 @@ use Doctrine\ORM\Mapping\PropertyAccessors\EnumPropertyAccessor; use Doctrine\ORM\Mapping\PropertyAccessors\ObjectCastPropertyAccessor; use Doctrine\ORM\Mapping\PropertyAccessors\PropertyAccessor; +use Doctrine\ORM\Mapping\PropertyAccessors\RawValuePropertyAccessor; use Doctrine\ORM\Mapping\PropertyAccessors\ReadonlyAccessor; use Doctrine\ORM\Mapping\PropertyAccessors\TypedNoDefaultPropertyAccessor; use Doctrine\Persistence\Mapping\ClassMetadata as PersistenceClassMetadata; @@ -62,6 +63,8 @@ use function trait_exists; use function trim; +use const PHP_VERSION_ID; + /** * A ClassMetadata instance holds all the object-relational mapping metadata * of an entity and its associations. @@ -2689,7 +2692,10 @@ public function getSequencePrefix(AbstractPlatform $platform): string private function createPropertyAccessor(string $className, string $propertyName): PropertyAccessor { $reflectionProperty = new ReflectionProperty($className, $propertyName); - $accessor = ObjectCastPropertyAccessor::fromReflectionProperty($reflectionProperty); + + $accessor = PHP_VERSION_ID >= 80400 + ? RawValuePropertyAccessor::fromReflectionProperty($reflectionProperty) + : ObjectCastPropertyAccessor::fromReflectionProperty($reflectionProperty); if ($reflectionProperty->hasType() && ! $reflectionProperty->getType()->allowsNull()) { $accessor = new TypedNoDefaultPropertyAccessor($accessor, $reflectionProperty); diff --git a/src/Mapping/PropertyAccessors/RawValuePropertyAccessor.php b/src/Mapping/PropertyAccessors/RawValuePropertyAccessor.php new file mode 100644 index 00000000000..27d3cf6ab87 --- /dev/null +++ b/src/Mapping/PropertyAccessors/RawValuePropertyAccessor.php @@ -0,0 +1,52 @@ +getName(); + $key = $reflectionProperty->isPrivate() ? "\0" . ltrim($reflectionProperty->getDeclaringClass()->getName(), '\\') . "\0" . $name : ($reflectionProperty->isProtected() ? "\0*\0" . $name : $name); + + return new self($reflectionProperty, $key); + } + + private function __construct(private ReflectionProperty $reflectionProperty, private string $key) + { + } + + public function setValue(object $object, mixed $value): void + { + if (! ($object instanceof InternalProxy && ! $object->__isInitialized())) { + $this->reflectionProperty->setRawValue($object, $value); + + return; + } + + $object->__setInitialized(true); + + $this->reflectionProperty->setRawValue($object, $value); + + $object->__setInitialized(false); + } + + public function getValue(object $object): mixed + { + return ((array) $object)[$this->key] ?? null; + } +}