Skip to content

[AutoMapper] New component to automatically map a source object to a target object #30248

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Closed
wants to merge 38 commits into from
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
38 commits
Select commit Hold shift + click to select a range
b5592ee
Initial commit and implementation of symfony/automapper
joelwurtz Feb 14, 2019
936bc2f
Add normalizer bridge
joelwurtz Feb 14, 2019
2ea1951
Add missing licence
joelwurtz Feb 14, 2019
d272d9c
Fix typo in interface
joelwurtz Feb 14, 2019
beb0f49
Add attribute checking option
joelwurtz Feb 15, 2019
e4c1385
Allow attribute checking to be set in factory
joelwurtz Feb 15, 2019
cf1fdf2
Add empty test classes
joelwurtz Feb 15, 2019
fc1d87d
Add automapper tests
joelwurtz Feb 19, 2019
bd387f5
Apply suggestions from @dunglas code review
dunglas Mar 25, 2019
20b7735
Add expiremental annotation where needed, add final to class that sho…
joelwurtz Mar 25, 2019
12e5d1f
Fix cs and typo
joelwurtz Mar 25, 2019
26e0c8a
Avoid too many function calls
joelwurtz Mar 25, 2019
678537a
Add interface for generator metadatas
joelwurtz Mar 25, 2019
ceeeaf1
Use array for context and provide helper class
joelwurtz Mar 26, 2019
ec97f74
Use new context construction in normalizer bridge
joelwurtz Mar 26, 2019
92784b1
Fix test case class test
joelwurtz Mar 26, 2019
eb6f638
Remove useless class
joelwurtz Mar 26, 2019
fab7135
expiremental > expiremental in 4.3
joelwurtz Mar 26, 2019
5154440
Fixing tests
Korbeil Dec 31, 2019
c688855
Added AutoMapperNormalizerTest
Korbeil Jan 3, 2020
9f16b43
Context tests
Korbeil Jan 5, 2020
ed00ec3
Add MapperGeneratorMetadataFactory tests
Korbeil Jan 5, 2020
ab82b9f
Add FromSourceMappingExtractor tests
Korbeil Jan 5, 2020
ee908a6
Add FromTargetMappingExtractor tests
Korbeil Jan 5, 2020
cd2d414
WIP PrivateReflectionExtractor tests
Korbeil Jan 6, 2020
cd85e2d
Remove internal for generated mapper
joelwurtz Jan 31, 2020
f74e22e
Fix context rebase
joelwurtz Jan 31, 2020
ee671dd
Add missing deps in dev
joelwurtz Jan 31, 2020
78ef805
Use property read / write info extractor
joelwurtz Jan 31, 2020
1e4bd5e
Fix cs
joelwurtz Jan 31, 2020
20a37df
Add tests and date time mutable / immutable transformations
joelwurtz Feb 2, 2020
af1fdf2
Fix header, expiremental in 5.1
joelwurtz Feb 2, 2020
c07edd2
Fix php version
joelwurtz Feb 2, 2020
ad190a2
Fix createFromImmutable not available in php 7.2
joelwurtz Feb 2, 2020
9c3cb23
Fix test
joelwurtz Feb 2, 2020
fecf949
Better conditions on automapper
joelwurtz Feb 3, 2020
ad0f487
Remove bad deps on rebase
joelwurtz Feb 3, 2020
5ba8171
Fix class exists
joelwurtz Feb 3, 2020
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
Add FromTargetMappingExtractor tests
  • Loading branch information
Korbeil authored and joelwurtz committed Jun 20, 2020
commit ee908a6cc90f8963d38bde149e223a2e3f2dec0c
Original file line number Diff line number Diff line change
Expand Up @@ -46,7 +46,7 @@ public function __construct(PropertyInfoExtractorInterface $propertyInfoExtracto
*/
public function getPropertiesMapping(MapperMetadataInterface $mapperMetadata): array
{
$targetProperties = array_unique($this->propertyInfoExtractor->getProperties($mapperMetadata->getTarget()));
$targetProperties = array_unique($this->propertyInfoExtractor->getProperties($mapperMetadata->getTarget()) ?? []);

if (!\in_array($mapperMetadata->getSource(), self::ALLOWED_SOURCES, true)) {
throw new InvalidMappingException('Only array or stdClass are accepted as a source');
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@
use Symfony\Component\AutoMapper\Extractor\ReflectionExtractor;
use Symfony\Component\AutoMapper\MapperMetadata;
use Symfony\Component\AutoMapper\Tests\AutoMapperBaseTest;
use Symfony\Component\AutoMapper\Tests\Fixtures;
use Symfony\Component\AutoMapper\Transformer\ArrayTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\BuiltinTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\ChainTransformerFactory;
Expand All @@ -31,7 +32,6 @@
use Symfony\Component\PropertyInfo\PropertyInfoExtractor;
use Symfony\Component\Serializer\Mapping\Factory\ClassMetadataFactory;
use Symfony\Component\Serializer\Mapping\Loader\AnnotationLoader;
use Symfony\Component\AutoMapper\Tests\Fixtures;

/**
* @author Baptiste Leduc <baptiste.leduc@gmail.com>
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
<?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\AutoMapper\Tests\Extractor;

use Doctrine\Common\Annotations\AnnotationReader;
use Symfony\Component\AutoMapper\Exception\InvalidMappingException;
use Symfony\Component\AutoMapper\Extractor\FromTargetMappingExtractor;
use Symfony\Component\AutoMapper\Extractor\PrivateReflectionExtractor;
use Symfony\Component\AutoMapper\Extractor\PropertyMapping;
use Symfony\Component\AutoMapper\Extractor\ReflectionExtractor;
use Symfony\Component\AutoMapper\MapperMetadata;
use Symfony\Component\AutoMapper\Tests\AutoMapperBaseTest;
use Symfony\Component\AutoMapper\Tests\Fixtures;
use Symfony\Component\AutoMapper\Transformer\ArrayTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\BuiltinTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\ChainTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\DateTimeTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\MultipleTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\NullableTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\ObjectTransformerFactory;
use Symfony\Component\AutoMapper\Transformer\UniqueTypeTransformerFactory;
use Symfony\Component\PropertyInfo\Extractor\PhpDocExtractor;
use Symfony\Component\PropertyInfo\PropertyInfoExtractor;
use Symfony\Component\Serializer\Mapping\Factory\ClassMetadataFactory;
use Symfony\Component\Serializer\Mapping\Loader\AnnotationLoader;

/**
* @author Baptiste Leduc <baptiste.leduc@gmail.com>
*/
class FromTargetMappingExtractorTest extends AutoMapperBaseTest
{
/** @var FromTargetMappingExtractor */
protected $fromTargetMappingExtractor;

public function setUp(): void
{
parent::setUp();
$this->fromTargetMappingExtractorBootstrap();
}

private function fromTargetMappingExtractorBootstrap(bool $private = true): void
{
$classMetadataFactory = new ClassMetadataFactory(new AnnotationLoader(new AnnotationReader()));
$reflectionExtractor = $private ? new PrivateReflectionExtractor() : new \Symfony\Component\PropertyInfo\Extractor\ReflectionExtractor();

$phpDocExtractor = new PhpDocExtractor();
$propertyInfoExtractor = new PropertyInfoExtractor(
[$reflectionExtractor],
[$phpDocExtractor, $reflectionExtractor],
[$reflectionExtractor],
[$reflectionExtractor]
);

$accessorExtractor = new ReflectionExtractor($private);
$transformerFactory = new ChainTransformerFactory();

$this->fromTargetMappingExtractor = new FromTargetMappingExtractor(
$propertyInfoExtractor,
$accessorExtractor,
$transformerFactory,
$classMetadataFactory
);

$transformerFactory->addTransformerFactory(new MultipleTransformerFactory($transformerFactory));
$transformerFactory->addTransformerFactory(new NullableTransformerFactory($transformerFactory));
$transformerFactory->addTransformerFactory(new UniqueTypeTransformerFactory($transformerFactory));
$transformerFactory->addTransformerFactory(new DateTimeTransformerFactory());
$transformerFactory->addTransformerFactory(new BuiltinTransformerFactory());
$transformerFactory->addTransformerFactory(new ArrayTransformerFactory($transformerFactory));
$transformerFactory->addTransformerFactory(new ObjectTransformerFactory($this->autoMapper));
}

public function testWithSourceAsArray(): void
{
$userReflection = new \ReflectionClass(Fixtures\User::class);
$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, 'array', Fixtures\User::class);
$targetPropertiesMapping = $this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);

self::assertCount(\count($userReflection->getProperties()), $targetPropertiesMapping);
/** @var PropertyMapping $propertyMapping */
foreach ($targetPropertiesMapping as $propertyMapping) {
self::assertTrue($userReflection->hasProperty($propertyMapping->getProperty()));
}
}

public function testWithSourceAsStdClass(): void
{
$userReflection = new \ReflectionClass(Fixtures\User::class);
$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, 'stdClass', Fixtures\User::class);
$targetPropertiesMapping = $this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);

self::assertCount(\count($userReflection->getProperties()), $targetPropertiesMapping);
/** @var PropertyMapping $propertyMapping */
foreach ($targetPropertiesMapping as $propertyMapping) {
self::assertTrue($userReflection->hasProperty($propertyMapping->getProperty()));
}
}

public function testWithTargetAsEmpty(): void
{
$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, 'array', Fixtures\Empty_::class);
$targetPropertiesMapping = $this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);

self::assertCount(0, $targetPropertiesMapping);
}

public function testWithTargetAsPrivate(): void
{
$privateReflection = new \ReflectionClass(Fixtures\Private_::class);
$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, 'array', Fixtures\Private_::class);
$targetPropertiesMapping = $this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);
self::assertCount(\count($privateReflection->getProperties()), $targetPropertiesMapping);

$this->fromTargetMappingExtractorBootstrap(false);
$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, 'array', Fixtures\Private_::class);
$targetPropertiesMapping = $this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);
self::assertCount(0, $targetPropertiesMapping);
}

public function testWithTargetAsArray(): void
{
self::expectException(InvalidMappingException::class);
self::expectExceptionMessage('Only array or stdClass are accepted as a source');

$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, Fixtures\User::class, 'array');
$this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);
}

public function testWithTargetAsStdClass(): void
{
self::expectException(InvalidMappingException::class);
self::expectExceptionMessage('Only array or stdClass are accepted as a source');

$mapperMetadata = new MapperMetadata($this->autoMapper, $this->fromTargetMappingExtractor, Fixtures\User::class, 'stdClass');
$this->fromTargetMappingExtractor->getPropertiesMapping($mapperMetadata);
}
}