From 71c1fe5221564f0f9ac702c5acbde36ca7a68b7c Mon Sep 17 00:00:00 2001 From: romanb <romanb@625475ce-881a-0410-a577-b389adb331d8> Date: Tue, 6 Oct 2009 10:04:32 +0000 Subject: [PATCH] [2.0][DDC-2] Fixed. --- .../ORM/Internal/Hydration/ObjectHydrator.php | 13 ++++-- lib/Doctrine/ORM/UnitOfWork.php | 5 ++- .../OneToOneBidirectionalAssociationTest.php | 42 ++++++++++++++++++- .../Doctrine/Tests/OrmFunctionalTestCase.php | 2 + 4 files changed, 57 insertions(+), 5 deletions(-) diff --git a/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php b/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php index 253adef0d..0665df3ce 100644 --- a/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php +++ b/lib/Doctrine/ORM/Internal/Hydration/ObjectHydrator.php @@ -49,6 +49,7 @@ class ObjectHydrator extends AbstractHydrator private $_fetchedAssociations; private $_rootAliases = array(); private $_initializedCollections = array(); + private $_proxyFactory; /** @override */ protected function _prepare() @@ -56,6 +57,10 @@ class ObjectHydrator extends AbstractHydrator $this->_allowPartialObjects = $this->_em->getConfiguration()->getAllowPartialObjects() || isset($this->_hints[Query::HINT_FORCE_PARTIAL_LOAD]); + if ( ! $this->_allowPartialObjects) { + $this->_proxyFactory = $this->_em->getProxyFactory(); + } + $this->_identifierMap = $this->_resultPointers = $this->_idTemplate = @@ -184,6 +189,7 @@ class ObjectHydrator extends AbstractHydrator // Properly initialize any unfetched associations, if partial objects are not allowed. if ( ! $this->_allowPartialObjects) { + $oid = spl_object_hash($entity); foreach ($this->_getClassMetadata($className)->associationMappings as $field => $assoc) { // Check if the association is not among the fetch-joined associatons already. if ( ! isset($this->_fetchedAssociations[$className][$field])) { @@ -194,9 +200,9 @@ class ObjectHydrator extends AbstractHydrator } if ($assoc->isLazilyFetched()) { // Inject proxy - $this->_ce[$className]->reflFields[$field]->setValue($entity, - $this->_em->getProxyFactory()->getAssociationProxy($entity, $assoc, $joinColumns) - ); + $proxy = $this->_proxyFactory->getAssociationProxy($entity, $assoc, $joinColumns); + $this->_uow->setOriginalEntityProperty($oid, $field, $proxy); + $this->_ce[$className]->reflFields[$field]->setValue($entity, $proxy); } else { // Eager load //TODO: Allow more efficient and configurable batching of these loads @@ -217,6 +223,7 @@ class ObjectHydrator extends AbstractHydrator //TODO: Allow more efficient and configurable batching of these loads $assoc->load($entity, $pColl, $this->_em); } + $this->_uow->setOriginalEntityProperty($oid, $field, $pColl); } } } diff --git a/lib/Doctrine/ORM/UnitOfWork.php b/lib/Doctrine/ORM/UnitOfWork.php index 334b86814..0613e0c96 100644 --- a/lib/Doctrine/ORM/UnitOfWork.php +++ b/lib/Doctrine/ORM/UnitOfWork.php @@ -598,6 +598,9 @@ class UnitOfWork implements PropertyChangedListener * Computes the changeset of an individual entity, independently of the * computeChangeSets() routine that is used at the beginning of a UnitOfWork#commit(). * + * The passed entity must be a managed entity. + * + * @ignore * @param $class * @param $entity */ @@ -1381,7 +1384,7 @@ class UnitOfWork implements PropertyChangedListener $this->removeFromIdentityMap($entity); unset($this->_entityInsertions[$oid], $this->_entityUpdates[$oid], $this->_entityDeletions[$oid], $this->_entityIdentifiers[$oid], - $this->_entityStates[$oid]); + $this->_entityStates[$oid], $this->_originalEntityData[$oid]); break; case self::STATE_NEW: case self::STATE_DETACHED: diff --git a/tests/Doctrine/Tests/ORM/Functional/OneToOneBidirectionalAssociationTest.php b/tests/Doctrine/Tests/ORM/Functional/OneToOneBidirectionalAssociationTest.php index 4d3338cf6..727bbb2ea 100644 --- a/tests/Doctrine/Tests/ORM/Functional/OneToOneBidirectionalAssociationTest.php +++ b/tests/Doctrine/Tests/ORM/Functional/OneToOneBidirectionalAssociationTest.php @@ -79,7 +79,8 @@ class OneToOneBidirectionalAssociationTest extends \Doctrine\Tests\OrmFunctional $this->assertEquals('Giorgio', $cart->getCustomer()->getName()); } - public function testLazyLoadsObjectsOnTheInverseSide() { + public function testLazyLoadsObjectsOnTheInverseSide() + { $this->_createFixture(); $this->_em->getConfiguration()->setAllowPartialObjects(false); $metadata = $this->_em->getClassMetadata('Doctrine\Tests\Models\ECommerce\ECommerceCustomer'); @@ -93,6 +94,45 @@ class OneToOneBidirectionalAssociationTest extends \Doctrine\Tests\OrmFunctional $this->assertTrue($customer->getCart() instanceof ECommerceCart); $this->assertEquals('paypal', $customer->getCart()->getPayment()); } + + public function testUpdateWithProxyObject() + { + $this->_em->getConfiguration()->setAllowPartialObjects(false); + + $cust = new ECommerceCustomer; + $cust->setName('Roman'); + $cart = new ECommerceCart; + $cart->setPayment('CARD'); + $cust->setCart($cart); + + $this->_em->persist($cust); + $this->_em->flush(); + $this->_em->clear(); + + $this->assertTrue($cust->getCart() instanceof ECommerceCart); + $this->assertEquals('Roman', $cust->getName()); + $this->assertSame($cust, $cart->getCustomer()); + + $query = $this->_em->createQuery('select ca from Doctrine\Tests\Models\ECommerce\ECommerceCart ca where ca.id =?1'); + $query->setParameter(1, $cart->getId()); + + $cart2 = $query->getSingleResult(); + + $cart2->setPayment('CHEQUE'); + + $this->_em->flush(); + $this->_em->clear(); + + $query2 = $this->_em->createQuery('select ca, c from Doctrine\Tests\Models\ECommerce\ECommerceCart ca left join ca.customer c where ca.id =?1'); + $query2->setParameter(1, $cart->getId()); + + $cart3 = $query2->getSingleResult(); + + $this->assertTrue($cart3->getCustomer() instanceof ECommerceCustomer); + $this->assertEquals('Roman', $cart3->getCustomer()->getName()); + + $this->_em->getConfiguration()->setAllowPartialObjects(true); + } protected function _createFixture() { diff --git a/tests/Doctrine/Tests/OrmFunctionalTestCase.php b/tests/Doctrine/Tests/OrmFunctionalTestCase.php index 95dfb31a0..bd35e5fb7 100644 --- a/tests/Doctrine/Tests/OrmFunctionalTestCase.php +++ b/tests/Doctrine/Tests/OrmFunctionalTestCase.php @@ -162,6 +162,8 @@ class OrmFunctionalTestCase extends OrmTestCase if (is_null(self::$_queryCacheImpl)) { self::$_queryCacheImpl = new \Doctrine\Common\Cache\ArrayCache; } + //FIXME: two different configs! $conn and the created entity manager have + // different configs. $config = new \Doctrine\ORM\Configuration(); $config->setMetadataCacheImpl(self::$_metadataCacheImpl); $config->setQueryCacheImpl(self::$_queryCacheImpl);