DDC-1193 - Fix bug with cascade remove and proxy classes.
This commit is contained in:
parent
875912bffd
commit
acaf08d4b7
3 changed files with 111 additions and 6 deletions
|
@ -563,7 +563,7 @@ class BasicEntityPersister
|
||||||
* @todo Check identity map? loadById method? Try to guess whether $criteria is the id?
|
* @todo Check identity map? loadById method? Try to guess whether $criteria is the id?
|
||||||
*/
|
*/
|
||||||
public function load(array $criteria, $entity = null, $assoc = null, array $hints = array(), $lockMode = 0)
|
public function load(array $criteria, $entity = null, $assoc = null, array $hints = array(), $lockMode = 0)
|
||||||
{
|
{
|
||||||
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, $lockMode);
|
$sql = $this->_getSelectEntitiesSQL($criteria, $assoc, $lockMode);
|
||||||
list($params, $types) = $this->expandParameters($criteria);
|
list($params, $types) = $this->expandParameters($criteria);
|
||||||
$stmt = $this->_conn->executeQuery($sql, $params, $types);
|
$stmt = $this->_conn->executeQuery($sql, $params, $types);
|
||||||
|
@ -577,7 +577,7 @@ class BasicEntityPersister
|
||||||
} else {
|
} else {
|
||||||
$hydrator = $this->_em->newHydrator(Query::HYDRATE_SIMPLEOBJECT);
|
$hydrator = $this->_em->newHydrator(Query::HYDRATE_SIMPLEOBJECT);
|
||||||
}
|
}
|
||||||
$entities = $hydrator->hydrateAll($stmt, $this->_rsm, $hints);
|
$entities = $hydrator->hydrateAll($stmt, $this->_rsm, $hints);
|
||||||
return $entities ? $entities[0] : null;
|
return $entities ? $entities[0] : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1290,6 +1290,10 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
}
|
}
|
||||||
|
|
||||||
$visited[$oid] = $entity; // mark visited
|
$visited[$oid] = $entity; // mark visited
|
||||||
|
|
||||||
|
// Cascade first, because scheduleForDelete() removes the entity from the identity map, which
|
||||||
|
// can cause problems when a lazy proxy has to be initialized for the cascade operation.
|
||||||
|
$this->cascadeRemove($entity, $visited);
|
||||||
|
|
||||||
$class = $this->em->getClassMetadata(get_class($entity));
|
$class = $this->em->getClassMetadata(get_class($entity));
|
||||||
$entityState = $this->getEntityState($entity);
|
$entityState = $this->getEntityState($entity);
|
||||||
|
@ -1313,7 +1317,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
throw new UnexpectedValueException("Unexpected entity state: $entityState.");
|
throw new UnexpectedValueException("Unexpected entity state: $entityState.");
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->cascadeRemove($entity, $visited);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -1674,6 +1677,11 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
if ( ! $assoc['isCascadePersist']) {
|
if ( ! $assoc['isCascadePersist']) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($entity instanceof Proxy && !$entity->__isInitialized__) {
|
||||||
|
$entity->__load();
|
||||||
|
}
|
||||||
|
|
||||||
$relatedEntities = $class->reflFields[$assoc['fieldName']]->getValue($entity);
|
$relatedEntities = $class->reflFields[$assoc['fieldName']]->getValue($entity);
|
||||||
if (($relatedEntities instanceof Collection || is_array($relatedEntities))) {
|
if (($relatedEntities instanceof Collection || is_array($relatedEntities))) {
|
||||||
if ($relatedEntities instanceof PersistentCollection) {
|
if ($relatedEntities instanceof PersistentCollection) {
|
||||||
|
@ -1702,7 +1710,11 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
if ( ! $assoc['isCascadeRemove']) {
|
if ( ! $assoc['isCascadeRemove']) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
//TODO: If $entity instanceof Proxy => Initialize ?
|
|
||||||
|
if ($entity instanceof Proxy) {
|
||||||
|
$entity->__load();
|
||||||
|
}
|
||||||
|
|
||||||
$relatedEntities = $class->reflFields[$assoc['fieldName']]->getValue($entity);
|
$relatedEntities = $class->reflFields[$assoc['fieldName']]->getValue($entity);
|
||||||
if ($relatedEntities instanceof Collection || is_array($relatedEntities)) {
|
if ($relatedEntities instanceof Collection || is_array($relatedEntities)) {
|
||||||
// If its a PersistentCollection initialization is intended! No unwrap!
|
// If its a PersistentCollection initialization is intended! No unwrap!
|
||||||
|
@ -1865,8 +1877,8 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
}
|
}
|
||||||
$id = array($class->identifier[0] => $idHash);
|
$id = array($class->identifier[0] => $idHash);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isset($this->identityMap[$class->rootEntityName][$idHash])) {
|
if (isset($this->identityMap[$class->rootEntityName][$idHash])) {
|
||||||
$entity = $this->identityMap[$class->rootEntityName][$idHash];
|
$entity = $this->identityMap[$class->rootEntityName][$idHash];
|
||||||
$oid = spl_object_hash($entity);
|
$oid = spl_object_hash($entity);
|
||||||
if ($entity instanceof Proxy && ! $entity->__isInitialized__) {
|
if ($entity instanceof Proxy && ! $entity->__isInitialized__) {
|
||||||
|
|
93
tests/Doctrine/Tests/ORM/Functional/Ticket/DDC1193Test.php
Normal file
93
tests/Doctrine/Tests/ORM/Functional/Ticket/DDC1193Test.php
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
<?php
|
||||||
|
namespace Doctrine\Tests\ORM\Functional\Ticket;
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../../../TestInit.php';
|
||||||
|
|
||||||
|
use DateTime, Doctrine\DBAL\Types\Type;
|
||||||
|
|
||||||
|
class DDC1193Test extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
|
{
|
||||||
|
protected function setUp()
|
||||||
|
{
|
||||||
|
parent::setUp();
|
||||||
|
//$this->_em->getConnection()->getConfiguration()->setSQLLogger(new \Doctrine\DBAL\Logging\EchoSQLLogger);
|
||||||
|
$this->_schemaTool->createSchema(array(
|
||||||
|
$this->_em->getClassMetadata(__NAMESPACE__ . '\DDC1193Company'),
|
||||||
|
$this->_em->getClassMetadata(__NAMESPACE__ . '\DDC1193Person'),
|
||||||
|
$this->_em->getClassMetadata(__NAMESPACE__ . '\DDC1193Account')
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group DDC-1193
|
||||||
|
*/
|
||||||
|
public function testIssue()
|
||||||
|
{
|
||||||
|
$company = new DDC1193Company();
|
||||||
|
$person = new DDC1193Person();
|
||||||
|
$account = new DDC1193Account();
|
||||||
|
|
||||||
|
$person->account = $account;
|
||||||
|
$person->company = $company;
|
||||||
|
|
||||||
|
$company->member = $person;
|
||||||
|
|
||||||
|
$this->_em->persist($company);
|
||||||
|
|
||||||
|
$this->_em->flush();
|
||||||
|
|
||||||
|
$companyId = $company->id;
|
||||||
|
$accountId = $account->id;
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$company = $this->_em->find(get_class($company), $companyId);
|
||||||
|
|
||||||
|
$this->assertTrue($this->_em->getUnitOfWork()->isInIdentityMap($company), "Company is in identity map.");
|
||||||
|
$this->assertFalse($company->member->__isInitialized__, "Pre-Condition");
|
||||||
|
$this->assertTrue($this->_em->getUnitOfWork()->isInIdentityMap($company->member), "Member is in identity map.");
|
||||||
|
|
||||||
|
$this->_em->remove($company);
|
||||||
|
$this->_em->flush();
|
||||||
|
|
||||||
|
$this->assertEquals(count($this->_em->getRepository(get_class($account))->findAll()), 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @Entity */
|
||||||
|
class DDC1193Company {
|
||||||
|
/**
|
||||||
|
* @Id @Column(type="integer")
|
||||||
|
* @GeneratedValue
|
||||||
|
*/
|
||||||
|
public $id;
|
||||||
|
|
||||||
|
/** @OneToOne(targetEntity="DDC1193Person", cascade={"persist", "remove"}) */
|
||||||
|
public $member;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @Entity */
|
||||||
|
class DDC1193Person {
|
||||||
|
/**
|
||||||
|
* @Id @Column(type="integer")
|
||||||
|
* @GeneratedValue
|
||||||
|
*/
|
||||||
|
public $id;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @OneToOne(targetEntity="DDC1193Account", cascade={"persist", "remove"})
|
||||||
|
*/
|
||||||
|
public $account;
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @Entity */
|
||||||
|
class DDC1193Account {
|
||||||
|
/**
|
||||||
|
* @Id @Column(type="integer")
|
||||||
|
* @GeneratedValue
|
||||||
|
*/
|
||||||
|
public $id;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue