1
0
Fork 0
mirror of synced 2025-04-03 13:23:37 +03:00

Updated tests.

This commit is contained in:
Roman S. Borschel 2010-07-29 14:08:36 +02:00
parent e69c7c7c60
commit 954a8c3935
3 changed files with 117 additions and 59 deletions

View file

@ -1344,34 +1344,51 @@ class UnitOfWork implements PropertyChangedListener
*/ */
private function doMerge($entity, array &$visited, $prevManagedCopy = null, $assoc = null) private function doMerge($entity, array &$visited, $prevManagedCopy = null, $assoc = null)
{ {
$class = $this->em->getClassMetadata(get_class($entity)); $oid = spl_object_hash($entity);
$id = $class->getIdentifierValues($entity); if (isset($visited[$oid])) {
return; // Prevent infinite recursion
if ( ! $id) {
throw new InvalidArgumentException('New entity detected during merge.'
. ' Persist the new entity before merging.');
} }
// MANAGED entities are ignored by the merge operation $class = $this->em->getClassMetadata(get_class($entity));
// First we assume DETACHED, although it can still be NEW but we can avoid
// an extra db-roundtrip this way. If it is DETACHED or NEW, we need to fetch
// it from the db anyway in order to merge.
// MANAGED entities are ignored by the merge operation.
if ($this->getEntityState($entity, self::STATE_DETACHED) == self::STATE_MANAGED) { if ($this->getEntityState($entity, self::STATE_DETACHED) == self::STATE_MANAGED) {
$managedCopy = $entity; $managedCopy = $entity;
} else { } else {
// Try to look the entity up in the identity map. // Try to look the entity up in the identity map.
$managedCopy = $this->tryGetById($id, $class->rootEntityName); $id = $class->getIdentifierValues($entity);
if ($managedCopy) {
// We have the entity in-memory already, just make sure its not removed.
if ($this->getEntityState($managedCopy) == self::STATE_REMOVED) {
throw new InvalidArgumentException('Removed entity detected during merge.'
. ' Can not merge with a removed entity.');
}
} else {
// We need to fetch the managed copy in order to merge.
$managedCopy = $this->em->find($class->name, $id);
}
if ($managedCopy === null) { // If there is no ID, it is actually NEW.
throw new InvalidArgumentException('New entity detected during merge.' if ( ! $id) {
. ' Persist the new entity before merging.'); $managedCopy = $class->newInstance();
$this->persistNew($class, $managedCopy);
} else {
$managedCopy = $this->tryGetById($id, $class->rootEntityName);
if ($managedCopy) {
// We have the entity in-memory already, just make sure its not removed.
if ($this->getEntityState($managedCopy) == self::STATE_REMOVED) {
throw new InvalidArgumentException('Removed entity detected during merge.'
. ' Can not merge with a removed entity.');
}
} else {
// We need to fetch the managed copy in order to merge.
$managedCopy = $this->em->find($class->name, $id);
}
if ($managedCopy === null) {
// If the identifier is ASSIGNED, it is NEW, otherwise an error
// since the managed entity was not found.
if ($class->isIdentifierNatural()) {
$managedCopy = $class->newInstance();
$class->setIdentifierValues($managedCopy, $id);
$this->persistNew($class, $managedCopy);
} else {
throw new EntityNotFoundException;
}
}
} }
if ($class->isVersioned) { if ($class->isVersioned) {
@ -1388,17 +1405,20 @@ class UnitOfWork implements PropertyChangedListener
if ( ! isset($class->associationMappings[$name])) { if ( ! isset($class->associationMappings[$name])) {
$prop->setValue($managedCopy, $prop->getValue($entity)); $prop->setValue($managedCopy, $prop->getValue($entity));
} else { } else {
// why $assoc2? See the method signature, there is $assoc already!
$assoc2 = $class->associationMappings[$name]; $assoc2 = $class->associationMappings[$name];
if ($assoc2->isOneToOne()) { if ($assoc2->isOneToOne()) {
if ( ! $assoc2->isCascadeMerge) { if ( ! $assoc2->isCascadeMerge) {
$other = $class->reflFields[$name]->getValue($entity); //TODO: Just $prop->getValue($entity)? $other = $prop->getValue($entity);
if ($other !== null) { if ($other !== null) {
$targetClass = $this->em->getClassMetadata($assoc2->targetEntityName); if ($this->getEntityState($other, self::STATE_DETACHED) == self::STATE_MANAGED) {
$id = $targetClass->getIdentifierValues($other); $prop->setValue($managedCopy, $other);
$proxy = $this->em->getProxyFactory()->getProxy($assoc2->targetEntityName, $id); } else {
$prop->setValue($managedCopy, $proxy); $targetClass = $this->em->getClassMetadata($assoc2->targetEntityName);
$this->registerManaged($proxy, $id, array()); $id = $targetClass->getIdentifierValues($other);
$proxy = $this->em->getProxyFactory()->getProxy($assoc2->targetEntityName, $id);
$prop->setValue($managedCopy, $proxy);
$this->registerManaged($proxy, $id, array());
}
} }
} }
} else { } else {
@ -1421,8 +1441,8 @@ class UnitOfWork implements PropertyChangedListener
//TODO: put changed fields in changeset...? //TODO: put changed fields in changeset...?
} }
} }
if ($class->isChangeTrackingDeferredExplicit()) { if ( ! $class->isChangeTrackingDeferredImplicit()) {
//TODO: Mark $managedCopy for dirty check...? ($this->scheduledForDirtyCheck) $this->scheduleForDirtyCheck($entity);
} }
} }

View file

@ -768,37 +768,39 @@ class BasicFunctionalTest extends \Doctrine\Tests\OrmFunctionalTestCase
$this->assertEquals('Stephan', $this->_em->find(get_class($user), $userId)->name); $this->assertEquals('Stephan', $this->_em->find(get_class($user), $userId)->name);
} }
//DRAFT OF EXPECTED/DESIRED BEHAVIOR public function testMergePersistsNewEntities()
/*public function testPersistentCollectionContainsDoesNeverInitialize()
{ {
$user = new CmsUser; $user = new CmsUser();
$user->name = 'Guilherme'; $user->username = "beberlei";
$user->username = 'gblanco'; $user->name = "Benjamin E.";
$user->status = 'developer'; $user->status = 'active';
$group = new CmsGroup; $managedUser = $this->_em->merge($user);
$group->name = 'Developers'; $this->assertEquals('beberlei', $managedUser->username);
$this->assertEquals('Benjamin E.', $managedUser->name);
$user->addGroup($group); $this->assertEquals('active', $managedUser->status);
$this->_em->persist($user); $this->assertTrue($user !== $managedUser);
$this->assertTrue($this->_em->contains($managedUser));
$this->_em->flush(); $this->_em->flush();
$userId = $managedUser->id;
$this->_em->clear(); $this->_em->clear();
$group = $this->_em->find(get_class($group), $group->getId()); $this->assertTrue($this->_em->find(get_class($managedUser), $userId) instanceof CmsUser);
}
public function testMergeThrowsExceptionIfEntityWithGeneratedIdentifierDoesNotExist()
$user2 = new CmsUser; {
$user2->id = $user->getId(); $user = new CmsUser();
$this->assertFalse($group->getUsers()->contains($user2)); $user->username = "beberlei";
$this->assertFalse($group->getUsers()->isInitialized()); $user->name = "Benjamin E.";
$user->status = 'active';
$user2 = $this->_em->getReference(get_class($user), $user->getId()); $user->id = 42;
$this->assertTrue($group->getUsers()->contains($user2)); try {
$this->assertFalse($group->getUsers()->isInitialized()); $this->_em->merge($user);
$this->fail();
} catch (\Doctrine\ORM\EntityNotFoundException $enfe) {}
} }
*/
} }

View file

@ -0,0 +1,36 @@
<?php
namespace Doctrine\Tests\ORM\Functional\Ticket;
require_once __DIR__ . '/../../../TestInit.php';
class DDC518Test extends \Doctrine\Tests\OrmFunctionalTestCase
{
public function setUp()
{
$this->useModelSet('cms');
parent::setUp();
}
public function testMergeWithRelatedNew()
{
$article = new \Doctrine\Tests\Models\CMS\CmsArticle();
$article->text = "foo";
$article->topic = "bar";
$this->_em->persist($article);
$this->_em->flush();
$this->_em->detach($article);
$this->_em->clear();
$user = new \Doctrine\Tests\Models\CMS\CmsUser();
$user->username = "beberlei";
$user->name = "Benjamin Eberlei";
$user->status = "active";
$article->user = $user;
$this->_em->persist($user);
$managedArticle = $this->_em->merge($article);
$this->assertSame($article->user, $managedArticle->user);
}
}