diff --git a/lib/Doctrine/ORM/UnitOfWork.php b/lib/Doctrine/ORM/UnitOfWork.php index f02f1e71b..828bd19ed 100644 --- a/lib/Doctrine/ORM/UnitOfWork.php +++ b/lib/Doctrine/ORM/UnitOfWork.php @@ -1775,6 +1775,10 @@ class UnitOfWork implements PropertyChangedListener if ($entity instanceof Proxy && ! $entity->__isInitialized__) { $entity->__isInitialized__ = true; $overrideLocalValues = true; + $this->_originalEntityData[$oid] = $data; + if ($entity instanceof NotifyPropertyChanged) { + $entity->addPropertyChangedListener($this); + } } else { $overrideLocalValues = isset($hints[Query::HINT_REFRESH]); } @@ -1844,6 +1848,7 @@ class UnitOfWork implements PropertyChangedListener $this->_entityIdentifiers[$newValueOid] = $associatedId; $this->_identityMap[$targetClass->rootEntityName][$relatedIdHash] = $newValue; $this->_entityStates[$newValueOid] = self::STATE_MANAGED; + // make sure that when an proxy is then finally loaded, $this->_originalEntityData is set also! } } $this->_originalEntityData[$oid][$field] = $newValue; diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC440Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC440Test.php new file mode 100644 index 000000000..e8c0bf085 --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC440Test.php @@ -0,0 +1,207 @@ +_schemaTool->createSchema(array( + $this->_em->getClassMetadata('Doctrine\Tests\ORM\Functional\Ticket\DDC440Phone'), + $this->_em->getClassMetadata('Doctrine\Tests\ORM\Functional\Ticket\DDC440Client') + )); + } catch (\Exception $e) { + // Swallow all exceptions. We do not test the schema tool here. + } + } + + /** + * @group DDC-440 + */ + public function testOriginalEntityDataEmptyWhenProxyLoadedFromTwoAssociations() + { + + + /* The key of the problem is that the first phone is fetched via two association, main_phone and phones. + * + * You will notice that the original_entity_datas are not loaded for the first phone. (They are for the second) + * + * In the Client entity definition, if you define the main_phone relation after the phones relation, both assertions pass. + * (for the sake or this test, I defined the main_phone relation before the phones relation) + * + */ + + //Initialize some data + $client = new DDC440Client; + $client->setName('Client1'); + + $phone = new DDC440Phone; + $phone->setNumber('418 111-1111'); + $phone->setClient($client); + + $phone2 = new DDC440Phone; + $phone2->setNumber('418 222-2222'); + $phone2->setClient($client); + + $client->setMainPhone($phone); + + $this->_em->persist($client); + $this->_em->flush(); + $id = $client->getId(); + $this->_em->clear(); + + $uw = $this->_em->getUnitOfWork(); + $client = $this->_em->find('Doctrine\Tests\ORM\Functional\Ticket\DDC440Client', $id); + $clientPhones = $client->getPhones(); + $p1 = $clientPhones[0]; + $p2 = $clientPhones[1]; + + // Test the first phone. The assertion actually failed because original entity data is not set properly. + // This was because it is also set as MainPhone and that one is created as a proxy, not the + // original object when the find on Client is called. However loading proxies did not work correctly. + $this->assertType('Doctrine\Tests\ORM\Functional\Ticket\DDC440Phone', $p1); + $originalData = $uw->getOriginalEntityData($p1); + $this->assertEquals($phone->getNumber(), $originalData['number']); + + + //If you comment out previous test, this one should pass + $this->assertType('Doctrine\Tests\ORM\Functional\Ticket\DDC440Phone', $p2); + $originalData = $uw->getOriginalEntityData($p2); + $this->assertEquals($phone2->getNumber(), $originalData['number']); + + } +} + +/** +* @Entity +* @Table(name="phone") +*/ +class DDC440Phone { + + /** + * @Column(name="id", type="integer") + * @Id + * @GeneratedValue(strategy="AUTO") + */ + protected $id; + + /** + * @ManyToOne(targetEntity="DDC440Client",inversedBy="phones") + * @JoinColumns({ + * @JoinColumn(name="client_id", referencedColumnName="id") + * }) + */ + protected $client; + + /** + * @Column(name="number", type="string") + */ + protected $number; + + public function setNumber($value){ + $this->number = $value; + } + + public function getNumber(){ + return $this->number; + } + + public function setClient(DDC440Client $value, $update_inverse=true) + { + $this->client = $value; + if($update_inverse){ + $value->addPhone($this); + } + } + + public function getClient() + { + return $this->client; + } + + public function getId() + { + return $this->id; + } + + public function setId($value){ + $this->id = $value; + } +} + +/** + * @Entity + * @Table(name="client") + */ +class DDC440Client { + + /** + * @Column(name="id", type="integer") + * @Id + * @GeneratedValue(strategy="AUTO") + */ + protected $id; + + /** + * @OneToOne(targetEntity="DDC440Phone", fetch="EAGER") + * @JoinColumns({ + * @JoinColumn(name="main_phone_id", referencedColumnName="id",onDelete="SET NULL") + * }) + */ + protected $main_phone; + + /** + * @OneToMany(targetEntity="DDC440Phone", mappedBy="client", cascade={"persist", "remove"}, fetch="EAGER") + */ + protected $phones; + + + /** + * @Column(name="name", type="string") + */ + protected $name; + + public function __construct(){ + + } + + public function setName($value){ + $this->name = $value; + } + + public function getName(){ + return $this->name; + } + + public function addPhone(DDC440Phone $value) + { + $this->phones[] = $value; + $value->setClient($this, false); + } + + public function getPhones() + { + return $this->phones; + } + + public function setMainPhone(DDC440Phone $value){ + $this->main_phone = $value; + } + + public function getMainPhone(){ + return $this->main_phone; + } + + public function getId() + { + return $this->id; + } + + public function setId($value){ + $this->id = $value; + } +}