From 55faa9c34e933c25c4569c4c4403fa0f0955592d Mon Sep 17 00:00:00 2001 From: Sergei Predvoditelev Date: Mon, 16 Sep 2024 14:13:26 +0300 Subject: [PATCH 1/2] Fix populating readonly properties from parent classes --- CHANGELOG.md | 2 +- src/Hydrator.php | 22 +++++++++- tests/HydratorTest.php | 40 +++++++++++++++++++ .../Classes/Inheritance/Figure/Circle.php | 10 +++++ .../Classes/Inheritance/Figure/Figure.php | 22 ++++++++++ .../Inheritance/ReadOnly/ImageSlideDto.php | 10 +++++ .../Classes/Inheritance/ReadOnly/SlideDto.php | 11 +++++ 7 files changed, 115 insertions(+), 2 deletions(-) create mode 100644 tests/Support/Classes/Inheritance/Figure/Circle.php create mode 100644 tests/Support/Classes/Inheritance/Figure/Figure.php create mode 100644 tests/Support/Classes/Inheritance/ReadOnly/ImageSlideDto.php create mode 100644 tests/Support/Classes/Inheritance/ReadOnly/SlideDto.php diff --git a/CHANGELOG.md b/CHANGELOG.md index 5c39c82..00ad293 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,7 +2,7 @@ ## 1.4.1 under development -- no changes in this release. +- Bug #95: Fix populating readonly properties from parent classes (@vjik) ## 1.4.0 August 23, 2024 diff --git a/src/Hydrator.php b/src/Hydrator.php index ce73c78..eeb58b7 100644 --- a/src/Hydrator.php +++ b/src/Hydrator.php @@ -22,6 +22,7 @@ use Yiisoft\Hydrator\TypeCaster\TypeCasterInterface; use function is_array; +use function var_dump; /** * Creates or hydrate objects from a set of raw data. @@ -79,6 +80,7 @@ public function hydrate(object $object, array|DataInterface $data = []): void $this->hydrateInternal( $object, + $reflectionClass, ReflectionFilter::filterProperties($object, $reflectionClass), $data ); @@ -108,6 +110,7 @@ public function create(string $class, array|DataInterface $data = []): object $this->hydrateInternal( $object, + $reflectionClass, ReflectionFilter::filterProperties($object, $reflectionClass, $excludeProperties), $data ); @@ -120,6 +123,7 @@ public function create(string $class, array|DataInterface $data = []): object */ private function hydrateInternal( object $object, + ReflectionClass $reflectionClass, array $reflectionProperties, DataInterface $data, ): void { @@ -141,9 +145,25 @@ private function hydrateInternal( new TypeCastContext($this, $property), ); if ($result->isResolved()) { - $property->setValue($object, $result->getValue()); + $this + ->preparePropertyToSetValue($reflectionClass, $property, $propertyName) + ->setValue($object, $result->getValue()); } } } } + + private function preparePropertyToSetValue( + ReflectionClass $class, + ReflectionProperty $property, + string $propertyName, + ): ReflectionProperty { + if ($property->isReadOnly()) { + $declaringClass = $property->getDeclaringClass(); + if ($declaringClass !== $class) { + return $declaringClass->getProperty($propertyName); + } + } + return $property; + } } diff --git a/tests/HydratorTest.php b/tests/HydratorTest.php index 2722ea7..189ba68 100644 --- a/tests/HydratorTest.php +++ b/tests/HydratorTest.php @@ -30,6 +30,8 @@ use Yiisoft\Hydrator\Tests\Support\Classes\ConstructorTypeClass; use Yiisoft\Hydrator\Tests\Support\Classes\CounterClass; use Yiisoft\Hydrator\Tests\Support\Classes\FromPredefinedArrayClass; +use Yiisoft\Hydrator\Tests\Support\Classes\Inheritance\ReadOnly\ImageSlideDto; +use Yiisoft\Hydrator\Tests\Support\Classes\Inheritance\Figure\Circle; use Yiisoft\Hydrator\Tests\Support\Classes\InvalidDataResolverClass; use Yiisoft\Hydrator\Tests\Support\Classes\NestedModel\UserModel; use Yiisoft\Hydrator\Tests\Support\Classes\NonInitializedReadonlyProperties; @@ -769,4 +771,42 @@ public function testHydrateNonInitializedReadonlyProperties(): void $this->assertSame('1', $object->a); $this->assertSame('3', $object->b); } + + public function testBaseInheritance(): void + { + $hydrator = new Hydrator(); + + $object = $hydrator->create( + Circle::class, + [ + 'name' => 'Wheel', + 'color' => 'Red', + 'id' => 'x7', + 'radius' => 17 + ] + ); + + $this->assertSame(17, $object->radius); + $this->assertSame('Wheel', $object->name); + $this->assertSame('Red', $object->getColor()); + $this->assertNull($object->getId()); + } + + public function testReadOnlyInheritance(): void + { + $hydrator = new Hydrator(); + + $object = $hydrator->create( + ImageSlideDto::class, + [ + 'src' => '/images/slide.jpg', + 'width' => 200, + 'height' => 300, + ] + ); + + $this->assertSame('/images/slide.jpg', $object->src); + $this->assertSame(200, $object->width); + $this->assertSame(300, $object->height); + } } diff --git a/tests/Support/Classes/Inheritance/Figure/Circle.php b/tests/Support/Classes/Inheritance/Figure/Circle.php new file mode 100644 index 0000000..a244b33 --- /dev/null +++ b/tests/Support/Classes/Inheritance/Figure/Circle.php @@ -0,0 +1,10 @@ +color; + } + + public function getId(): ?int + { + return $this->id; + } +} diff --git a/tests/Support/Classes/Inheritance/ReadOnly/ImageSlideDto.php b/tests/Support/Classes/Inheritance/ReadOnly/ImageSlideDto.php new file mode 100644 index 0000000..ede189a --- /dev/null +++ b/tests/Support/Classes/Inheritance/ReadOnly/ImageSlideDto.php @@ -0,0 +1,10 @@ + Date: Mon, 16 Sep 2024 11:14:14 +0000 Subject: [PATCH 2/2] Apply fixes from StyleCI --- src/Hydrator.php | 1 - tests/HydratorTest.php | 2 +- 2 files changed, 1 insertion(+), 2 deletions(-) diff --git a/src/Hydrator.php b/src/Hydrator.php index eeb58b7..ee214c1 100644 --- a/src/Hydrator.php +++ b/src/Hydrator.php @@ -22,7 +22,6 @@ use Yiisoft\Hydrator\TypeCaster\TypeCasterInterface; use function is_array; -use function var_dump; /** * Creates or hydrate objects from a set of raw data. diff --git a/tests/HydratorTest.php b/tests/HydratorTest.php index 189ba68..a27e5e3 100644 --- a/tests/HydratorTest.php +++ b/tests/HydratorTest.php @@ -782,7 +782,7 @@ public function testBaseInheritance(): void 'name' => 'Wheel', 'color' => 'Red', 'id' => 'x7', - 'radius' => 17 + 'radius' => 17, ] );