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

Improved entity state detection.

This commit is contained in:
Roman S. Borschel 2010-07-04 18:55:49 +02:00
parent 26d8b4dafd
commit 88b0813536
9 changed files with 224 additions and 123 deletions

View file

@ -143,7 +143,11 @@ class ClassMetadata extends ClassMetadataInfo
} }
return $id; return $id;
} else { } else {
return array($this->identifier[0] => $this->reflFields[$this->identifier[0]]->getValue($entity)); $value = $this->reflFields[$this->identifier[0]]->getValue($entity);
if ($value !== null) {
return array($this->identifier[0] => $value);
}
return array();
} }
} }

View file

@ -32,12 +32,11 @@ use Doctrine\ORM\Mapping\AssociationMapping,
* Similarly, if you remove entities from a collection that is part of a one-many * Similarly, if you remove entities from a collection that is part of a one-many
* mapping this will only result in the nulling out of the foreign keys on flush. * mapping this will only result in the nulling out of the foreign keys on flush.
* *
* @license http://www.opensource.org/licenses/lgpl-license.php LGPL
* @since 2.0 * @since 2.0
* @version $Revision: 4930 $
* @author Konsta Vesterinen <kvesteri@cc.hut.fi> * @author Konsta Vesterinen <kvesteri@cc.hut.fi>
* @author Roman Borschel <roman@code-factory.org> * @author Roman Borschel <roman@code-factory.org>
* @author Giorgio Sironi <piccoloprincipeazzurro@gmail.com> * @author Giorgio Sironi <piccoloprincipeazzurro@gmail.com>
* @todo Design for inheritance to allow custom implementations?
*/ */
final class PersistentCollection implements Collection final class PersistentCollection implements Collection
{ {

View file

@ -57,9 +57,9 @@ use PDO,
* *
* - {@link load} : Loads (the state of) a single, managed entity. * - {@link load} : Loads (the state of) a single, managed entity.
* - {@link loadAll} : Loads multiple, managed entities. * - {@link loadAll} : Loads multiple, managed entities.
* - {@link loadOneToOneEntity} : Loads a one/many-to-one association (lazy-loading). * - {@link loadOneToOneEntity} : Loads a one/many-to-one entity association (lazy-loading).
* - {@link loadOneToManyCollection} : Loads a one-to-many association (lazy-loading). * - {@link loadOneToManyCollection} : Loads a one-to-many entity association (lazy-loading).
* - {@link loadManyToManyCollection} : Loads a many-to-many association (lazy-loading). * - {@link loadManyToManyCollection} : Loads a many-to-many entity association (lazy-loading).
* *
* The BasicEntityPersister implementation provides the default behavior for * The BasicEntityPersister implementation provides the default behavior for
* persisting and querying entities that are mapped to a single database table. * persisting and querying entities that are mapped to a single database table.
@ -712,7 +712,7 @@ class BasicEntityPersister
* Creates or fills a single entity object from an SQL result. * Creates or fills a single entity object from an SQL result.
* *
* @param $result The SQL result. * @param $result The SQL result.
* @param object $entity The entity object to fill. * @param object $entity The entity object to fill, if any.
* @param array $hints Hints for entity creation. * @param array $hints Hints for entity creation.
* @return object The filled and managed entity object or NULL, if the SQL result is empty. * @return object The filled and managed entity object or NULL, if the SQL result is empty.
*/ */
@ -1119,6 +1119,22 @@ class BasicEntityPersister
$stmt->closeCursor(); $stmt->closeCursor();
} }
/**
* Checks whether the given managed entity exists in the database.
*
* @param object $entity
* @return boolean TRUE if the entity exists in the database, FALSE otherwise.
*/
public function exists($entity)
{
$criteria = $this->_class->getIdentifierValues($entity);
$sql = 'SELECT 1 FROM ' . $this->_class->getQuotedTableName($this->_platform)
. ' ' . $this->_getSQLTableAlias($this->_class->name)
. ' WHERE ' . $this->_getSelectConditionSQL($criteria);
return (bool) $this->_conn->fetchColumn($sql, array_values($criteria));
}
//TODO //TODO
/*protected function _getOneToOneEagerFetchSQL() /*protected function _getOneToOneEagerFetchSQL()
{ {

View file

@ -497,7 +497,6 @@ class SqlWalker implements TreeWalker
$sql .= reset($assoc->targetToSourceKeyColumns); $sql .= reset($assoc->targetToSourceKeyColumns);
} else { } else {
// 2- Inverse side: NOT (YET?) SUPPORTED
throw QueryException::associationPathInverseSideNotSupported(); throw QueryException::associationPathInverseSideNotSupported();
} }
break; break;

View file

@ -556,7 +556,7 @@ class UnitOfWork implements PropertyChangedListener
return; // "Persistence by reachability" only if persist cascade specified return; // "Persistence by reachability" only if persist cascade specified
} }
// Look through the entities, and in any of their associations, for transient // Look through the entities, and in any of their associations, for transient (new)
// enities, recursively. ("Persistence by reachability") // enities, recursively. ("Persistence by reachability")
if ($assoc->isOneToOne()) { if ($assoc->isOneToOne()) {
if ($value instanceof Proxy && ! $value->__isInitialized__) { if ($value instanceof Proxy && ! $value->__isInitialized__) {
@ -569,42 +569,45 @@ class UnitOfWork implements PropertyChangedListener
$targetClass = $this->_em->getClassMetadata($assoc->targetEntityName); $targetClass = $this->_em->getClassMetadata($assoc->targetEntityName);
foreach ($value as $entry) { foreach ($value as $entry) {
$state = $this->getEntityState($entry, self::STATE_NEW); $state = $this->getEntityState($entry);
$oid = spl_object_hash($entry); $oid = spl_object_hash($entry);
if ($state == self::STATE_NEW) { if ($state == self::STATE_NEW) {
if (isset($targetClass->lifecycleCallbacks[Events::prePersist])) { $this->persistNew($targetClass, $entry);
$targetClass->invokeLifecycleCallbacks(Events::prePersist, $entry);
}
if ($this->_evm->hasListeners(Events::prePersist)) {
$this->_evm->dispatchEvent(Events::prePersist, new LifecycleEventArgs($entry, $this->_em));
}
// Get identifier, if possible (not post-insert)
$idGen = $targetClass->idGenerator;
if ( ! $idGen->isPostInsertGenerator()) {
$idValue = $idGen->generate($this->_em, $entry);
if ( ! $idGen instanceof \Doctrine\ORM\Id\AssignedGenerator) {
$this->_entityIdentifiers[$oid] = array($targetClass->identifier[0] => $idValue);
$targetClass->getSingleIdReflectionProperty()->setValue($entry, $idValue);
} else {
$this->_entityIdentifiers[$oid] = $idValue;
}
$this->addToIdentityMap($entry);
}
$this->_entityStates[$oid] = self::STATE_MANAGED;
// NEW entities are INSERTed within the current unit of work.
$this->_entityInsertions[$oid] = $entry;
$this->computeChangeSet($targetClass, $entry); $this->computeChangeSet($targetClass, $entry);
} else if ($state == self::STATE_REMOVED) { } else if ($state == self::STATE_REMOVED) {
throw ORMException::removedEntityInCollectionDetected($entity, $assoc); throw ORMException::removedEntityInCollectionDetected($entry, $assoc);
} else if ($state == self::STATE_DETACHED) {
throw new \InvalidArgumentException("Detached entity in association during cascading a persist operation.");
} }
// MANAGED associated entities are already taken into account // MANAGED associated entities are already taken into account
// during changeset calculation anyway, since they are in the identity map. // during changeset calculation anyway, since they are in the identity map.
} }
} }
private function persistNew($class, $entity)
{
$oid = spl_object_hash($entity);
if (isset($class->lifecycleCallbacks[Events::prePersist])) {
$class->invokeLifecycleCallbacks(Events::prePersist, $entity);
}
if ($this->_evm->hasListeners(Events::prePersist)) {
$this->_evm->dispatchEvent(Events::prePersist, new LifecycleEventArgs($entity, $this->_em));
}
$idGen = $class->idGenerator;
if ( ! $idGen->isPostInsertGenerator()) {
$idValue = $idGen->generate($this->_em, $entity);
if ( ! $idGen instanceof \Doctrine\ORM\Id\AssignedGenerator) {
$this->_entityIdentifiers[$oid] = array($class->identifier[0] => $idValue);
$class->setIdentifierValues($entity, $idValue);
} else {
$this->_entityIdentifiers[$oid] = $idValue;
}
}
$this->_entityStates[$oid] = self::STATE_MANAGED;
$this->scheduleForInsert($entity);
}
/** /**
* INTERNAL: * INTERNAL:
@ -1031,35 +1034,54 @@ class UnitOfWork implements PropertyChangedListener
} }
/** /**
* Gets the state of an entity within the current unit of work. * Gets the state of an entity with regard to the current unit of work.
*
* NOTE: This method sees entities that are not MANAGED or REMOVED and have a
* populated identifier, whether it is generated or manually assigned, as
* DETACHED. This can be incorrect for manually assigned identifiers.
* *
* @param object $entity * @param object $entity
* @param integer $assume The state to assume if the state is not yet known. This is usually * @param integer $assume The state to assume if the state is not yet known (not MANAGED or REMOVED).
* used to avoid costly state lookups, in the worst case with a database * This parameter can be set to improve performance of entity state detection
* lookup. * by potentially avoiding a database lookup if the distinction between NEW and DETACHED
* is either known or does not matter for the caller of the method.
* @return int The entity state. * @return int The entity state.
*/ */
public function getEntityState($entity, $assume = null) public function getEntityState($entity, $assume = null)
{ {
$oid = spl_object_hash($entity); $oid = spl_object_hash($entity);
if ( ! isset($this->_entityStates[$oid])) { if ( ! isset($this->_entityStates[$oid])) {
// State can only be NEW or DETACHED, because MANAGED/REMOVED states are immediately // State can only be NEW or DETACHED, because MANAGED/REMOVED states are known.
// set by the UnitOfWork directly. We treat all entities that have a populated // Note that you can not remember the NEW or DETACHED state in _entityStates since
// identifier as DETACHED and all others as NEW. This is not really correct for // the UoW does not hold references to such objects and the object hash can be reused.
// manually assigned identifiers but in that case we would need to hit the database // More generally because the state may "change" between NEW/DETACHED without the UoW being aware of it.
// and we would like to avoid that.
if ($assume === null) { if ($assume === null) {
if ($this->_em->getClassMetadata(get_class($entity))->getIdentifierValues($entity)) { $class = $this->_em->getClassMetadata(get_class($entity));
$this->_entityStates[$oid] = self::STATE_DETACHED; $id = $class->getIdentifierValues($entity);
if ( ! $id) {
return self::STATE_NEW;
} else if ($class->isIdentifierNatural()) {
// Check for a version field, if available, to avoid a db lookup.
if ($class->isVersioned) {
if ($class->getFieldValue($entity, $class->versionField)) {
return self::STATE_DETACHED;
} else {
return self::STATE_NEW;
}
} else {
// Last try before db lookup: check the identity map.
if ($this->tryGetById($id, $class->rootEntityName)) {
return self::STATE_DETACHED;
} else {
// db lookup
if ($this->getEntityPersister(get_class($entity))->exists($entity)) {
return self::STATE_DETACHED;
} else {
return self::STATE_NEW;
}
}
}
} else { } else {
$this->_entityStates[$oid] = self::STATE_NEW; return self::STATE_DETACHED;
} }
} else { } else {
$this->_entityStates[$oid] = $assume; return $assume;
} }
} }
return $this->_entityStates[$oid]; return $this->_entityStates[$oid];
@ -1169,12 +1191,10 @@ class UnitOfWork implements PropertyChangedListener
} }
/** /**
* Saves an entity as part of the current unit of work. * Persists an entity as part of the current unit of work.
* This method is internally called during save() cascades as it tracks
* the already visited entities to prevent infinite recursions.
* *
* NOTE: This method always considers entities that are not yet known to * This method is internally called during persist() cascades as it tracks
* this UnitOfWork as NEW. * the already visited entities to prevent infinite recursions.
* *
* @param object $entity The entity to persist. * @param object $entity The entity to persist.
* @param array $visited The already visited entities. * @param array $visited The already visited entities.
@ -1189,8 +1209,8 @@ class UnitOfWork implements PropertyChangedListener
$visited[$oid] = $entity; // Mark visited $visited[$oid] = $entity; // Mark visited
$class = $this->_em->getClassMetadata(get_class($entity)); $class = $this->_em->getClassMetadata(get_class($entity));
$entityState = $this->getEntityState($entity, self::STATE_NEW); $entityState = $this->getEntityState($entity);
switch ($entityState) { switch ($entityState) {
case self::STATE_MANAGED: case self::STATE_MANAGED:
// Nothing to do, except if policy is "deferred explicit" // Nothing to do, except if policy is "deferred explicit"
@ -1199,30 +1219,10 @@ class UnitOfWork implements PropertyChangedListener
} }
break; break;
case self::STATE_NEW: case self::STATE_NEW:
if (isset($class->lifecycleCallbacks[Events::prePersist])) { $this->persistNew($class, $entity);
$class->invokeLifecycleCallbacks(Events::prePersist, $entity);
}
if ($this->_evm->hasListeners(Events::prePersist)) {
$this->_evm->dispatchEvent(Events::prePersist, new LifecycleEventArgs($entity, $this->_em));
}
$idGen = $class->idGenerator;
if ( ! $idGen->isPostInsertGenerator()) {
$idValue = $idGen->generate($this->_em, $entity);
if ( ! $idGen instanceof \Doctrine\ORM\Id\AssignedGenerator) {
$this->_entityIdentifiers[$oid] = array($class->identifier[0] => $idValue);
$class->setIdentifierValues($entity, $idValue);
} else {
$this->_entityIdentifiers[$oid] = $idValue;
}
}
$this->_entityStates[$oid] = self::STATE_MANAGED;
$this->scheduleForInsert($entity);
break; break;
case self::STATE_DETACHED: case self::STATE_DETACHED:
throw new \InvalidArgumentException( throw new \InvalidArgumentException("Detached entity passed to persist().");
"Behavior of persist() for a detached entity is not yet defined.");
case self::STATE_REMOVED: case self::STATE_REMOVED:
// Entity becomes managed again // Entity becomes managed again
if ($this->isScheduledForDelete($entity)) { if ($this->isScheduledForDelete($entity)) {
@ -1235,7 +1235,7 @@ class UnitOfWork implements PropertyChangedListener
default: default:
throw ORMException::invalidEntityState($entityState); throw ORMException::invalidEntityState($entityState);
} }
$this->_cascadePersist($entity, $visited); $this->_cascadePersist($entity, $visited);
} }
@ -1440,7 +1440,6 @@ class UnitOfWork implements PropertyChangedListener
* *
* @param object $entity * @param object $entity
* @param array $visited * @param array $visited
* @internal This method always considers entities with an assigned identifier as DETACHED.
*/ */
private function _doDetach($entity, array &$visited) private function _doDetach($entity, array &$visited)
{ {
@ -1657,7 +1656,7 @@ class UnitOfWork implements PropertyChangedListener
*/ */
public function lock($entity, $lockMode, $lockVersion = null) public function lock($entity, $lockMode, $lockVersion = null)
{ {
if ($this->getEntityState($entity) != self::STATE_MANAGED) { if ($this->getEntityState($entity, self::STATE_NEW) != self::STATE_MANAGED) {
throw new \InvalidArgumentException("Entity is not MANAGED."); throw new \InvalidArgumentException("Entity is not MANAGED.");
} }
@ -2183,4 +2182,47 @@ class UnitOfWork implements PropertyChangedListener
{ {
return $this->_collectionUpdates; return $this->_collectionUpdates;
} }
/**
* Specifically tells this UnitOfWork that the given entity should be treated
* as NEW. This can be useful in the case of manually assigned identifiers to
* avoid a database lookup that is used to tell NEW and DETACHED entities apart.
* Use setNew($entity) prior to persist($entity) to avoid the database lookup.
*
* Note that the UnitOfWork uses the spl_object_hash() of the object to associate the
* state with the object. Thus the UnitOfWork does not prevent the object from being
* garbage collected which can result in the object hash being reused for other objects.
*
* @param object $entity The entity to mark as NEW.
* @throws InvalidArgumentException If the entity is already known to this UnitOfWork.
*/
public function setNew($entity)
{
$oid = spl_object_hash($entity);
if (isset($this->_entityStates[$oid])) {
throw new \InvalidArgumentException("The passed entity must not be already known to this UnitOfWork.");
}
$this->_entityStates[$oid] = self::STATE_NEW;
}
/**
* Specifically tells this UnitOfWork that the given entity should be treated
* as DETACHED. This can be useful in the case of manually assigned identifiers to
* avoid a database lookup that is used to tell NEW and DETACHED entities apart.
*
* Note that the UnitOfWork uses the spl_object_hash() of the object to associate the
* state with the object. Thus the UnitOfWork does not prevent the object from being
* garbage collected which can result in the object hash being reused for other objects.
*
* @param object $entity The entity to mark as DETACHED.
* @throws InvalidArgumentException If the entity is already known to this UnitOfWork.
*/
public function setDetached($entity)
{
$oid = spl_object_hash($entity);
if (isset($this->_entityStates[$oid])) {
throw new \InvalidArgumentException("The passed entity must not be already known to this UnitOfWork.");
}
$this->_entityStates[$oid] = self::STATE_DETACHED;
}
} }

View file

@ -13,6 +13,7 @@ class EntityPersisterMock extends \Doctrine\ORM\Persisters\BasicEntityPersister
private $_identityColumnValueCounter = 0; private $_identityColumnValueCounter = 0;
private $_mockIdGeneratorType; private $_mockIdGeneratorType;
private $_postInsertIds = array(); private $_postInsertIds = array();
private $existsCalled = false;
/** /**
* @param <type> $entity * @param <type> $entity
@ -57,6 +58,11 @@ class EntityPersisterMock extends \Doctrine\ORM\Persisters\BasicEntityPersister
{ {
$this->_updates[] = $entity; $this->_updates[] = $entity;
} }
public function exists($entity)
{
$this->existsCalled = true;
}
public function delete($entity) public function delete($entity)
{ {
@ -80,9 +86,15 @@ class EntityPersisterMock extends \Doctrine\ORM\Persisters\BasicEntityPersister
public function reset() public function reset()
{ {
$this->existsCalled = false;
$this->_identityColumnValueCounter = 0; $this->_identityColumnValueCounter = 0;
$this->_inserts = array(); $this->_inserts = array();
$this->_updates = array(); $this->_updates = array();
$this->_deletes = array(); $this->_deletes = array();
} }
public function isExistsCalled()
{
return $this->existsCalled;
}
} }

View file

@ -86,6 +86,20 @@ class DetachedEntityTest extends \Doctrine\Tests\OrmFunctionalTestCase
$this->assertTrue($this->_em->contains($phonenumbers[1])); $this->assertTrue($this->_em->contains($phonenumbers[1]));
} }
/**
* @expectedException InvalidArgumentException
* @group DDC-203
*/
public function testDetachedEntityWithAssignedIdentityThrowsExceptionOnPersist()
{
$ph = new CmsPhonenumber();
$ph->phonenumber = '12345';
$this->_em->persist($ph);
$this->_em->flush();
$this->_em->clear();
$this->_em->persist($ph);
}
/** /**
* @group DDC-518 * @group DDC-518
*/ */

View file

@ -185,7 +185,9 @@ class ManyToManyBasicAssociationTest extends \Doctrine\Tests\OrmFunctionalTestCa
/* @var $freshUser CmsUser */ /* @var $freshUser CmsUser */
$freshUser = $this->_em->find('Doctrine\Tests\Models\CMS\CmsUser', $user->getId()); $freshUser = $this->_em->find('Doctrine\Tests\Models\CMS\CmsUser', $user->getId());
$freshUser->addGroup($group); $newGroup = new CmsGroup();
$newGroup->setName('12Monkeys');
$freshUser->addGroup($newGroup);
$this->assertFalse($freshUser->groups->isInitialized(), "CmsUser::groups Collection has to be uninitialized for this test."); $this->assertFalse($freshUser->groups->isInitialized(), "CmsUser::groups Collection has to be uninitialized for this test.");

View file

@ -2,6 +2,7 @@
namespace Doctrine\Tests\ORM; namespace Doctrine\Tests\ORM;
use Doctrine\ORM\UnitOfWork;
use Doctrine\Tests\Mocks\ConnectionMock; use Doctrine\Tests\Mocks\ConnectionMock;
use Doctrine\Tests\Mocks\EntityManagerMock; use Doctrine\Tests\Mocks\EntityManagerMock;
use Doctrine\Tests\Mocks\UnitOfWorkMock; use Doctrine\Tests\Mocks\UnitOfWorkMock;
@ -147,47 +148,46 @@ class UnitOfWorkTest extends \Doctrine\Tests\OrmTestCase
$this->assertEquals(array('data' => array('thedata', 'newdata')), $this->_unitOfWork->getEntityChangeSet($entity)); $this->assertEquals(array('data' => array('thedata', 'newdata')), $this->_unitOfWork->getEntityChangeSet($entity));
} }
/* public function testGetEntityStateOnVersionedEntityWithAssignedIdentifier()
public function testSavingSingleEntityWithSequenceIdGeneratorSchedulesInsert()
{ {
//... $persister = new EntityPersisterMock($this->_emMock, $this->_emMock->getClassMetadata("Doctrine\Tests\ORM\VersionedAssignedIdentifierEntity"));
$this->_unitOfWork->setEntityPersister('Doctrine\Tests\ORM\VersionedAssignedIdentifierEntity', $persister);
$e = new VersionedAssignedIdentifierEntity();
$e->id = 42;
$this->assertEquals(UnitOfWork::STATE_NEW, $this->_unitOfWork->getEntityState($e));
$this->assertFalse($persister->isExistsCalled());
} }
public function testSavingSingleEntityWithTableIdGeneratorSchedulesInsert() public function testGetEntityStateWithAssignedIdentity()
{ {
//... $persister = new EntityPersisterMock($this->_emMock, $this->_emMock->getClassMetadata("Doctrine\Tests\Models\CMS\CmsPhonenumber"));
$this->_unitOfWork->setEntityPersister('Doctrine\Tests\Models\CMS\CmsPhonenumber', $persister);
$ph = new \Doctrine\Tests\Models\CMS\CmsPhonenumber();
$ph->phonenumber = '12345';
$this->assertEquals(UnitOfWork::STATE_NEW, $this->_unitOfWork->getEntityState($ph));
$this->assertTrue($persister->isExistsCalled());
$persister->reset();
// setNew should avoid exists() check
$this->_unitOfWork->setNew($ph);
$this->assertEquals(UnitOfWork::STATE_NEW, $this->_unitOfWork->getEntityState($ph));
$this->assertFalse($persister->isExistsCalled());
$persister->reset();
// if the entity is already managed the exists() check should also be skipped
$this->_unitOfWork->registerManaged($ph, array('phonenumber' => '12345'), array());
$this->assertEquals(UnitOfWork::STATE_MANAGED, $this->_unitOfWork->getEntityState($ph));
$this->assertFalse($persister->isExistsCalled());
$ph2 = new \Doctrine\Tests\Models\CMS\CmsPhonenumber();
$ph2->phonenumber = '12345';
$this->assertEquals(UnitOfWork::STATE_DETACHED, $this->_unitOfWork->getEntityState($ph2));
$this->assertFalse($persister->isExistsCalled());
} }
public function testSavingSingleEntityWithSingleNaturalIdForcesInsert()
{
//...
}
public function testSavingSingleEntityWithCompositeIdForcesInsert()
{
//...
}
public function testSavingEntityGraphWithIdentityColumnsForcesInserts()
{
//...
}
public function testSavingEntityGraphWithSequencesDelaysInserts()
{
//...
}
public function testSavingEntityGraphWithNaturalIdsForcesInserts()
{
//...
}
public function testSavingEntityGraphWithMixedIdGenerationStrategies()
{
//...
}
*/
} }
/** /**
@ -243,4 +243,17 @@ class NotifyChangedEntity implements \Doctrine\Common\NotifyPropertyChanged
} }
} }
} }
}
/** @Entity */
class VersionedAssignedIdentifierEntity
{
/**
* @Id @Column(type="integer")
*/
public $id;
/**
* @Version @Column(type="integer")
*/
public $version;
} }