diff --git a/lib/Doctrine/ORM/Mapping/Reflection/ReflectionPropertiesGetter.php b/lib/Doctrine/ORM/Mapping/Reflection/ReflectionPropertiesGetter.php new file mode 100644 index 000000000..6ac7f11c3 --- /dev/null +++ b/lib/Doctrine/ORM/Mapping/Reflection/ReflectionPropertiesGetter.php @@ -0,0 +1,163 @@ +. + */ + +namespace Doctrine\ORM\Mapping\Reflection; + +use Doctrine\Common\Persistence\Mapping\ReflectionService; +use ReflectionClass; +use ReflectionProperty; + +/** + * Utility class to retrieve all reflection instance properties of a given class, including + * private inherited properties and transient properties. + * + * @private This API is for internal use only + * + * @author Marco Pivetta + */ +final class ReflectionPropertiesGetter +{ + /** + * @var ReflectionProperty[][] indexed by class name and property internal name + */ + private $properties = []; + + /** + * @var ReflectionService + */ + private $reflectionService; + + /** + * @param ReflectionService $reflectionService + */ + public function __construct(ReflectionService $reflectionService) + { + $this->reflectionService = $reflectionService; + } + + /** + * @param $className + * + * @return ReflectionProperty[] indexed by property internal name + */ + public function getProperties($className) + { + if (isset($this->properties[$className])) { + return $this->properties[$className]; + } + + return $this->properties[$className] = call_user_func_array( + 'array_merge', + // first merge because `array_merge` expects >= 1 params + array_merge( + [[]], + array_map( + [$this, 'getClassProperties'], + $this->getHierarchyClasses($className) + ) + ) + ); + } + + /** + * @param string $className + * + * @return ReflectionClass[] + */ + private function getHierarchyClasses($className) + { + $classes = []; + $parentClassName = $className; + + while ($parentClassName && $currentClass = $this->reflectionService->getClass($parentClassName)) { + $classes[] = $currentClass; + $parentClassName = null; + + if ($parentClass = $currentClass->getParentClass()) { + $parentClassName = $parentClass->getName(); + } + } + + return $classes; + } + + /** + * @param ReflectionClass $reflectionClass + * + * @return ReflectionProperty[] + */ + private function getClassProperties(ReflectionClass $reflectionClass) + { + $properties = $reflectionClass->getProperties(); + + return array_filter( + array_filter(array_map( + [$this, 'getAccessibleProperty'], + array_combine( + array_map([$this, 'getLogicalName'], $properties), + $properties + ) + )), + [$this, 'isInstanceProperty'] + ); + } + + /** + * @param ReflectionProperty $reflectionProperty + * + * @return bool + */ + private function isInstanceProperty(ReflectionProperty $reflectionProperty) + { + return ! $reflectionProperty->isStatic(); + } + + /** + * @param ReflectionProperty $property + * + * @return null|ReflectionProperty + */ + private function getAccessibleProperty(ReflectionProperty $property) + { + return $this->reflectionService->getAccessibleProperty( + $property->getDeclaringClass()->getName(), + $property->getName() + ); + } + + /** + * @param ReflectionProperty $property + * + * @return string + */ + private function getLogicalName(ReflectionProperty $property) + { + $propertyName = $property->getName(); + + if ($property->isPublic()) { + return $propertyName; + } + + if ($property->isProtected()) { + return "\0*\0" . $propertyName; + } + + return "\0" . $property->getDeclaringClass()->getName() . "\0" . $propertyName; + } +} diff --git a/lib/Doctrine/ORM/UnitOfWork.php b/lib/Doctrine/ORM/UnitOfWork.php index cd7c718d6..8f82aa96f 100644 --- a/lib/Doctrine/ORM/UnitOfWork.php +++ b/lib/Doctrine/ORM/UnitOfWork.php @@ -19,8 +19,10 @@ namespace Doctrine\ORM; +use Doctrine\Common\Persistence\Mapping\RuntimeReflectionService; use Doctrine\DBAL\LockMode; use Doctrine\ORM\Internal\HydrationCompleteHandler; +use Doctrine\ORM\Mapping\Reflection\ReflectionPropertiesGetter; use Exception; use InvalidArgumentException; use UnexpectedValueException; @@ -283,6 +285,11 @@ class UnitOfWork implements PropertyChangedListener */ private $hydrationCompleteHandler; + /** + * @var ReflectionPropertiesGetter + */ + private $reflectionPropertiesGetter; + /** * Initializes a new UnitOfWork instance, bound to the given EntityManager. * @@ -290,12 +297,13 @@ class UnitOfWork implements PropertyChangedListener */ public function __construct(EntityManagerInterface $em) { - $this->em = $em; - $this->evm = $em->getEventManager(); - $this->listenersInvoker = new ListenersInvoker($em); - $this->hasCache = $em->getConfiguration()->isSecondLevelCacheEnabled(); - $this->identifierFlattener = new IdentifierFlattener($this, $em->getMetadataFactory()); - $this->hydrationCompleteHandler = new HydrationCompleteHandler($this->listenersInvoker, $em); + $this->em = $em; + $this->evm = $em->getEventManager(); + $this->listenersInvoker = new ListenersInvoker($em); + $this->hasCache = $em->getConfiguration()->isSecondLevelCacheEnabled(); + $this->identifierFlattener = new IdentifierFlattener($this, $em->getMetadataFactory()); + $this->hydrationCompleteHandler = new HydrationCompleteHandler($this->listenersInvoker, $em); + $this->reflectionPropertiesGetter = new ReflectionPropertiesGetter(new RuntimeReflectionService()); } /** @@ -1868,6 +1876,10 @@ class UnitOfWork implements PropertyChangedListener $visited[$oid] = $managedCopy; // mark visited if (!($entity instanceof Proxy && ! $entity->__isInitialized())) { + if ($managedCopy instanceof Proxy && ! $managedCopy->__isInitialized()) { + $managedCopy->__load(); + } + $this->mergeEntityStateIntoManagedCopy($entity, $managedCopy); } @@ -3348,7 +3360,7 @@ class UnitOfWork implements PropertyChangedListener { $class = $this->em->getClassMetadata(get_class($entity)); - foreach ($class->reflClass->getProperties() as $prop) { + foreach ($this->reflectionPropertiesGetter->getProperties($class->name) as $prop) { $name = $prop->name; $prop->setAccessible(true); diff --git a/tests/Doctrine/Tests/Models/DirectoryTree/AbstractContentItem.php b/tests/Doctrine/Tests/Models/DirectoryTree/AbstractContentItem.php index 12b27eed3..d2ba439a0 100644 --- a/tests/Doctrine/Tests/Models/DirectoryTree/AbstractContentItem.php +++ b/tests/Doctrine/Tests/Models/DirectoryTree/AbstractContentItem.php @@ -24,6 +24,8 @@ namespace Doctrine\Tests\Models\DirectoryTree; */ abstract class AbstractContentItem { + const CLASSNAME = __CLASS__; + /** * @Id @Column(type="integer") @GeneratedValue */ @@ -37,6 +39,20 @@ abstract class AbstractContentItem /** @column(type="string") */ protected $name; + /** + * This field is transient and private on purpose + * + * @var bool + */ + private $nodeIsLoaded = false; + + /** + * This field is transient on purpose + * + * @var mixed + */ + public static $fileSystem; + public function __construct(Directory $parentDir = null) { $this->parentDirectory = $parentDir; @@ -61,4 +77,20 @@ abstract class AbstractContentItem { return $this->parentDirectory; } + + /** + * @return bool + */ + public function getNodeIsLoaded() + { + return $this->nodeIsLoaded; + } + + /** + * @param bool $nodeIsLoaded + */ + public function setNodeIsLoaded($nodeIsLoaded) + { + $this->nodeIsLoaded = (bool) $nodeIsLoaded; + } } diff --git a/tests/Doctrine/Tests/Models/DirectoryTree/File.php b/tests/Doctrine/Tests/Models/DirectoryTree/File.php index 35132689a..0b8ed0c80 100644 --- a/tests/Doctrine/Tests/Models/DirectoryTree/File.php +++ b/tests/Doctrine/Tests/Models/DirectoryTree/File.php @@ -26,6 +26,8 @@ namespace Doctrine\Tests\Models\DirectoryTree; */ class File extends AbstractContentItem { + const CLASSNAME = __CLASS__; + /** @Column(type="string") */ protected $extension = "html"; diff --git a/tests/Doctrine/Tests/Models/Reflection/ClassWithMixedProperties.php b/tests/Doctrine/Tests/Models/Reflection/ClassWithMixedProperties.php new file mode 100644 index 000000000..72899959a --- /dev/null +++ b/tests/Doctrine/Tests/Models/Reflection/ClassWithMixedProperties.php @@ -0,0 +1,18 @@ +assertFalse($managed->__isInitialized()); } + /** + * @group DDC-1392 + * @group DDC-1734 + * @group DDC-3368 + * @group #1172 + * + * Bug discovered while working on DDC-2704 - merging towards un-initialized proxies does not initialize them, + * causing merged data to be lost when they are actually initialized + */ + public function testMergeWithExistingUninitializedManagedProxy() + { + $date = new DateTimeModel(); + + $this->_em->persist($date); + $this->_em->flush($date); + $this->_em->clear(); + + $managed = $this->_em->getReference(DateTimeModel::CLASSNAME, $date->id); + + $this->assertInstanceOf('Doctrine\Common\Proxy\Proxy', $managed); + $this->assertFalse($managed->__isInitialized()); + + $date->date = $dateTime = new \DateTime(); + + $this->assertSame($managed, $this->_em->merge($date)); + $this->assertTrue($managed->__isInitialized()); + $this->assertSame($dateTime, $managed->date, 'Data was merged into the proxy after initialization'); + } + /** * @group DDC-1392 * @group DDC-1734 diff --git a/tests/Doctrine/Tests/ORM/Functional/MergeSharedEntitiesTest.php b/tests/Doctrine/Tests/ORM/Functional/MergeSharedEntitiesTest.php index 3dae34b67..28e89c360 100644 --- a/tests/Doctrine/Tests/ORM/Functional/MergeSharedEntitiesTest.php +++ b/tests/Doctrine/Tests/ORM/Functional/MergeSharedEntitiesTest.php @@ -90,6 +90,25 @@ class MergeSharedEntitiesTest extends OrmFunctionalTestCase $this->assertEquals($picture->file, $picture->otherFile, 'Identical entities must remain identical'); } + + /** + * @group DDC-2704 + */ + public function testMergeInheritedTransientPrivateProperties() + { + $admin1 = new MSEAdmin(); + $admin2 = new MSEAdmin(); + + $admin1->id = 123; + $admin2->id = 123; + + $this->_em->persist($admin1); + + $admin2->setSession('zeh current session data'); + + $this->assertSame($admin1, $this->_em->merge($admin2)); + $this->assertSame('zeh current session data', $admin1->getSession()); + } } /** @Entity */ @@ -111,3 +130,26 @@ class MSEFile /** @Column(type="integer") @Id @GeneratedValue(strategy="AUTO") */ public $id; } + +/** @MappedSuperclass */ +abstract class MSEUser +{ + private $session; // intentionally transient property + + public function getSession() + { + return $this->session; + } + + public function setSession($session) + { + $this->session = $session; + } +} + +/** @Entity */ +class MSEAdmin extends MSEUser +{ + /** @Column(type="integer") @Id @GeneratedValue(strategy="NONE") */ + public $id; +} diff --git a/tests/Doctrine/Tests/ORM/Mapping/Reflection/ReflectionPropertiesGetterTest.php b/tests/Doctrine/Tests/ORM/Mapping/Reflection/ReflectionPropertiesGetterTest.php new file mode 100644 index 000000000..7bc83df2e --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Mapping/Reflection/ReflectionPropertiesGetterTest.php @@ -0,0 +1,129 @@ +getProperties(ClassWithMixedProperties::CLASSNAME); + + $this->assertCount(5, $properties); + + foreach ($properties as $property) { + $this->assertInstanceOf('ReflectionProperty', $property); + } + } + + public function testRetrievedInstancesAreNotStatic() + { + $properties = (new ReflectionPropertiesGetter(new RuntimeReflectionService())) + ->getProperties(ClassWithMixedProperties::CLASSNAME); + + foreach ($properties as $property) { + $this->assertFalse($property->isStatic()); + } + } + + public function testExpectedKeys() + { + $properties = (new ReflectionPropertiesGetter(new RuntimeReflectionService())) + ->getProperties(ClassWithMixedProperties::CLASSNAME); + + $this->assertArrayHasKey( + "\0" . ClassWithMixedProperties::CLASSNAME . "\0" . 'privateProperty', + $properties + ); + $this->assertArrayHasKey( + "\0" . ClassWithMixedProperties::CLASSNAME . "\0" . 'privatePropertyOverride', + $properties + ); + $this->assertArrayHasKey( + "\0" . ParentClass::CLASSNAME . "\0" . 'privatePropertyOverride', + $properties + ); + $this->assertArrayHasKey( + "\0*\0protectedProperty", + $properties + ); + $this->assertArrayHasKey( + "publicProperty", + $properties + ); + } + + public function testPropertiesAreAccessible() + { + $object = new ClassWithMixedProperties(); + $properties = (new ReflectionPropertiesGetter(new RuntimeReflectionService())) + ->getProperties(ClassWithMixedProperties::CLASSNAME); + + foreach ($properties as $property) { + $this->assertSame($property->getName(), $property->getValue($object)); + } + } + + public function testPropertyGetterIsIdempotent() + { + $getter = (new ReflectionPropertiesGetter(new RuntimeReflectionService())); + + $this->assertSame( + $getter->getProperties(ClassWithMixedProperties::CLASSNAME), + $getter->getProperties(ClassWithMixedProperties::CLASSNAME) + ); + } + + public function testPropertyGetterWillSkipPropertiesNotRetrievedByTheRuntimeReflectionService() + { + /* @var $reflectionService ReflectionService|\PHPUnit_Framework_MockObject_MockObject */ + $reflectionService = $this->getMock('Doctrine\Common\Persistence\Mapping\ReflectionService'); + + $reflectionService + ->expects($this->exactly(2)) + ->method('getClass') + ->with($this->logicalOr(ClassWithMixedProperties::CLASSNAME, ParentClass::CLASSNAME)) + ->will($this->returnValueMap([ + [ClassWithMixedProperties::CLASSNAME, new ReflectionClass(ClassWithMixedProperties::CLASSNAME)], + [ParentClass::CLASSNAME, new ReflectionClass(ParentClass::CLASSNAME)], + ])); + + $reflectionService + ->expects($this->atLeastOnce()) + ->method('getAccessibleProperty'); + + $getter = (new ReflectionPropertiesGetter($reflectionService)); + + $this->assertEmpty($getter->getProperties(ClassWithMixedProperties::CLASSNAME)); + } + + public function testPropertyGetterWillSkipClassesNotRetrievedByTheRuntimeReflectionService() + { + /* @var $reflectionService ReflectionService|\PHPUnit_Framework_MockObject_MockObject */ + $reflectionService = $this->getMock('Doctrine\Common\Persistence\Mapping\ReflectionService'); + + $reflectionService + ->expects($this->once()) + ->method('getClass') + ->with(ClassWithMixedProperties::CLASSNAME); + + $reflectionService->expects($this->never())->method('getAccessibleProperty'); + + $getter = (new ReflectionPropertiesGetter($reflectionService)); + + $this->assertEmpty($getter->getProperties(ClassWithMixedProperties::CLASSNAME)); + } +}