*/ class IdentityMapTest extends \Doctrine\Tests\OrmFunctionalTestCase { protected function setUp() { $this->useModelSet('cms'); parent::setUp(); } public function testBasicIdentityManagement() { $user = new CmsUser; $user->status = 'dev'; $user->username = 'romanb'; $user->name = 'Roman B.'; $address = new CmsAddress; $address->country = 'de'; $address->zip = 1234; $address->city = 'Berlin'; $user->setAddress($address); $this->_em->persist($user); $this->_em->flush(); $this->_em->clear(); $user2 = $this->_em->find(get_class($user), $user->getId()); $this->assertTrue($user2 !== $user); $user3 = $this->_em->find(get_class($user), $user->getId()); $this->assertTrue($user2 === $user3); $address2 = $this->_em->find(get_class($address), $address->getId()); $this->assertTrue($address2 !== $address); $address3 = $this->_em->find(get_class($address), $address->getId()); $this->assertTrue($address2 === $address3); $this->assertTrue($user2->getAddress() === $address2); // !!! } public function testSingleValuedAssociationIdentityMapBehaviorWithRefresh() { $address = new CmsAddress; $address->country = 'de'; $address->zip = '12345'; $address->city = 'Berlin'; $user1 = new CmsUser; $user1->status = 'dev'; $user1->username = 'romanb'; $user1->name = 'Roman B.'; $user2 = new CmsUser; $user2->status = 'dev'; $user2->username = 'gblanco'; $user2->name = 'Guilherme Blanco'; $address->setUser($user1); $this->_em->persist($address); $this->_em->persist($user1); $this->_em->persist($user2); $this->_em->flush(); $this->assertSame($user1, $address->user); //external update to CmsAddress $this->_em->getConnection()->executeUpdate('update cms_addresses set user_id = ?', array($user2->getId())); // But we want to have this external change! // Solution 1: refresh(), broken atm! $this->_em->refresh($address); // Now the association should be "correct", referencing $user2 $this->assertSame($user2, $address->user); $this->assertSame($user2->address, $address); // check back reference also // Attention! refreshes can result in broken bidirectional associations! this is currently expected! // $user1 still points to $address! $this->assertSame($user1->address, $address); } public function testSingleValuedAssociationIdentityMapBehaviorWithRefreshQuery() { $address = new CmsAddress; $address->country = 'de'; $address->zip = '12345'; $address->city = 'Berlin'; $user1 = new CmsUser; $user1->status = 'dev'; $user1->username = 'romanb'; $user1->name = 'Roman B.'; $user2 = new CmsUser; $user2->status = 'dev'; $user2->username = 'gblanco'; $user2->name = 'Guilherme Blanco'; $address->setUser($user1); $this->_em->persist($address); $this->_em->persist($user1); $this->_em->persist($user2); $this->_em->flush(); $this->assertSame($user1, $address->user); //external update to CmsAddress $this->_em->getConnection()->executeUpdate('update cms_addresses set user_id = ?', array($user2->getId())); //select $q = $this->_em->createQuery('select a, u from Doctrine\Tests\Models\CMS\CmsAddress a join a.user u'); $address2 = $q->getSingleResult(); $this->assertSame($address, $address2); // Should still be $user1 $this->assertSame($user1, $address2->user); $this->assertTrue($user2->address === null); // But we want to have this external change! // Solution 2: Alternatively, a refresh query should work $q = $this->_em->createQuery('select a, u from Doctrine\Tests\Models\CMS\CmsAddress a join a.user u'); $q->setHint(Query::HINT_REFRESH, true); $address3 = $q->getSingleResult(); $this->assertSame($address, $address3); // should still be the same, always from identity map // Now the association should be "correct", referencing $user2 $this->assertSame($user2, $address2->user); $this->assertSame($user2->address, $address2); // check back reference also // Attention! refreshes can result in broken bidirectional associations! this is currently expected! // $user1 still points to $address2! $this->assertSame($user1->address, $address2); } public function testCollectionValuedAssociationIdentityMapBehaviorWithRefreshQuery() { $user = new CmsUser; $user->status = 'dev'; $user->username = 'romanb'; $user->name = 'Roman B.'; $phone1 = new CmsPhonenumber; $phone1->phonenumber = 123; $phone2 = new CmsPhonenumber; $phone2->phonenumber = 234; $phone3 = new CmsPhonenumber; $phone3->phonenumber = 345; $user->addPhonenumber($phone1); $user->addPhonenumber($phone2); $user->addPhonenumber($phone3); $this->_em->persist($user); // cascaded to phone numbers $this->_em->flush(); $this->assertEquals(3, count($user->getPhonenumbers())); $this->assertFalse($user->getPhonenumbers()->isDirty()); //external update to CmsAddress $this->_em->getConnection()->executeUpdate('insert into cms_phonenumbers (phonenumber, user_id) VALUES (?,?)', array(999, $user->getId())); //select $q = $this->_em->createQuery('select u, p from Doctrine\Tests\Models\CMS\CmsUser u join u.phonenumbers p'); $user2 = $q->getSingleResult(); $this->assertSame($user, $user2); // Should still be the same 3 phonenumbers $this->assertEquals(3, count($user2->getPhonenumbers())); // But we want to have this external change! // Solution 1: refresh(). //$this->_em->refresh($user2); broken atm! // Solution 2: Alternatively, a refresh query should work $q = $this->_em->createQuery('select u, p from Doctrine\Tests\Models\CMS\CmsUser u join u.phonenumbers p'); $q->setHint(Query::HINT_REFRESH, true); $user3 = $q->getSingleResult(); $this->assertSame($user, $user3); // should still be the same, always from identity map // Now the collection should be refreshed with correct count $this->assertEquals(4, count($user3->getPhonenumbers())); } public function testCollectionValuedAssociationIdentityMapBehaviorWithRefresh() { $user = new CmsUser; $user->status = 'dev'; $user->username = 'romanb'; $user->name = 'Roman B.'; $phone1 = new CmsPhonenumber; $phone1->phonenumber = 123; $phone2 = new CmsPhonenumber; $phone2->phonenumber = 234; $phone3 = new CmsPhonenumber; $phone3->phonenumber = 345; $user->addPhonenumber($phone1); $user->addPhonenumber($phone2); $user->addPhonenumber($phone3); $this->_em->persist($user); // cascaded to phone numbers $this->_em->flush(); $this->assertEquals(3, count($user->getPhonenumbers())); //external update to CmsAddress $this->_em->getConnection()->executeUpdate('insert into cms_phonenumbers (phonenumber, user_id) VALUES (?,?)', array(999, $user->getId())); //select $q = $this->_em->createQuery('select u, p from Doctrine\Tests\Models\CMS\CmsUser u join u.phonenumbers p'); $user2 = $q->getSingleResult(); $this->assertSame($user, $user2); // Should still be the same 3 phonenumbers $this->assertEquals(3, count($user2->getPhonenumbers())); // But we want to have this external change! // Solution 1: refresh(). $this->_em->refresh($user2); $this->assertSame($user, $user2); // should still be the same, always from identity map // Now the collection should be refreshed with correct count $this->assertEquals(4, count($user2->getPhonenumbers())); } public function testReusedSplObjectHashDoesNotConfuseUnitOfWork() { $hash1 = $this->subRoutine($this->_em); if (!defined('HHVM_VERSION')) { // See comment below about PersistentCollection gc_collect_cycles(); } $user1 = new CmsUser; $user1->status = 'dev'; $user1->username = 'jwage'; $user1->name = 'Jonathan W.'; $hash2 = spl_object_hash($user1); $this->assertEquals($hash1, $hash2); // Hash reused! $this->_em->persist($user1); $this->_em->flush(); } private function subRoutine($em) { $user = new CmsUser; $user->status = 'dev'; $user->username = 'romanb'; $user->name = 'Roman B.'; $em->persist($user); $em->flush(); // The PersistentCollection references its owner, hence without breaking // the cycle the object will not (yet) be garbage collected and thus // the object hash is not reused. This is not a memory leak! if (defined('HHVM_VERSION')) { $ed = $this->_em->getUnitOfWork()->getOriginalEntityData($user); $ed['phonenumbers']->setOwner(null, array('inversedBy' => 1)); $ed['articles']->setOwner(null, array('inversedBy' => 1)); $ed['groups']->setOwner(null, array('inversedBy' => 1)); } $em->remove($user); $em->flush(); return spl_object_hash($user); } }