From b91b89758abe81fef33b71b986f5cf27a5f0b1df Mon Sep 17 00:00:00 2001 From: Priyadi Iman Nurcahyo <1102197+priyadi@users.noreply.github.com> Date: Wed, 13 Nov 2024 01:24:34 +0700 Subject: [PATCH] fix: infinite loop in `MapperFactory` (#254) * fix: infinite loop in `MapperFactory` * fix --- CHANGELOG.md | 4 +++ src/MapperFactory.php | 29 ++++++++++++++-- .../rekalogika-mapper/generated-mappings.php | 1 + tests/src/UnitTest/MapperFactoryTest.php | 34 +++++++++++++++++++ 4 files changed, 66 insertions(+), 2 deletions(-) create mode 100644 tests/src/UnitTest/MapperFactoryTest.php diff --git a/CHANGELOG.md b/CHANGELOG.md index d041b37..6f45813 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,9 @@ # CHANGELOG +## 1.13.5 + +* fix: infinite loop in `MapperFactory` + ## 1.13.4 * fix: fix list handling diff --git a/src/MapperFactory.php b/src/MapperFactory.php index 0bcbf5a..320928b 100644 --- a/src/MapperFactory.php +++ b/src/MapperFactory.php @@ -179,10 +179,14 @@ class MapperFactory private ?TransformerRegistryInterface $transformerRegistry = null; + private ?ContainerInterface $propertyMapperLocator = null; + private ?PropertyMapperResolverInterface $propertyMapperResolver = null; private ?ObjectMapperTableFactoryInterface $objectMapperTableFactory = null; + private ?ContainerInterface $objectMapperLocator = null; + private ?ObjectMapperResolverInterface $objectMapperResolver = null; private ?PropertyReadInfoExtractorInterface $propertyReadInfoExtractor = null; @@ -443,7 +447,7 @@ protected function getObjectMapperTransformer(): TransformerInterface if (null === $this->objectMapperTransformer) { $this->objectMapperTransformer = new ObjectMapperTransformer( $this->getSubMapperFactory(), - $this->getTransformersLocator(), + $this->getObjectMapperLocator(), $this->getObjectMapperTableFactory(), $this->getObjectMapperResolver(), ); @@ -800,6 +804,23 @@ protected function getObjectMapperTableFactory(): ObjectMapperTableFactoryInterf return $this->objectMapperTableFactory; } + protected function getObjectMapperLocator(): ContainerInterface + { + if ($this->objectMapperLocator !== null) { + return $this->objectMapperLocator; + } + + $services = []; + + foreach ($this->objectMappers as $objectMapper) { + $service = $objectMapper['service']; + $class = $service::class; + $services[$class] = $service; + } + + return $this->objectMapperLocator = new ServiceLocator($services); + } + protected function getObjectMapperResolver(): ObjectMapperResolverInterface { if (null === $this->objectMapperResolver) { @@ -813,6 +834,10 @@ protected function getObjectMapperResolver(): ObjectMapperResolverInterface protected function getPropertyMapperLocator(): ContainerInterface { + if ($this->propertyMapperLocator !== null) { + return $this->propertyMapperLocator; + } + $services = []; foreach ($this->propertyMappers as $propertyMapper) { @@ -821,7 +846,7 @@ protected function getPropertyMapperLocator(): ContainerInterface $services[$class] = $service; } - return new ServiceLocator($services); + return $this->propertyMapperLocator = new ServiceLocator($services); } protected function getPropertyReadInfoExtractor(): PropertyReadInfoExtractorInterface diff --git a/tests/config/rekalogika-mapper/generated-mappings.php b/tests/config/rekalogika-mapper/generated-mappings.php index 310f22f..357a355 100644 --- a/tests/config/rekalogika-mapper/generated-mappings.php +++ b/tests/config/rekalogika-mapper/generated-mappings.php @@ -281,6 +281,7 @@ $mappingCollection->addObjectMapping( // tests/src/IntegrationTest/BasicMappingTest.php on line 33 // tests/src/IntegrationTest/IterableMapperTest.php on line 24 + // tests/src/UnitTest/MapperFactoryTest.php on line 29 source: \Rekalogika\Mapper\Tests\Fixtures\Basic\Person::class, target: \Rekalogika\Mapper\Tests\Fixtures\Basic\PersonDto::class ); diff --git a/tests/src/UnitTest/MapperFactoryTest.php b/tests/src/UnitTest/MapperFactoryTest.php new file mode 100644 index 0000000..3813bfe --- /dev/null +++ b/tests/src/UnitTest/MapperFactoryTest.php @@ -0,0 +1,34 @@ + + * + * For the full copyright and license information, please view the LICENSE file + * that was distributed with this source code. + */ + +namespace Rekalogika\Mapper\Tests\UnitTest; + +use PHPUnit\Framework\TestCase; +use Rekalogika\Mapper\MapperFactory; +use Rekalogika\Mapper\Tests\Fixtures\Basic\Person; +use Rekalogika\Mapper\Tests\Fixtures\Basic\PersonDto; + +class MapperFactoryTest extends TestCase +{ + public function testMapperFactory(): void + { + $mapperFactory = new MapperFactory(); + $mapper = $mapperFactory->getMapper(); + + $source = new Person('John Doe', 30); + $target = $mapper->map($source, PersonDto::class); + + $this->assertSame('John Doe', $target->name); + $this->assertSame(30, $target->age); + } +}