Add way to keep track of read only objects in the UnitOfWork which are never updated during flush.
Changed the behavior of EntityManager#getPartialReference to be read-only. No changes are ever done to this entities. Changed UnitOfWork#computeChangeSet to never create a changeset for fields that are partially omitted from a DQL or NativeQuery. To check if an entity is read only use the new API: if ($entityManager->getUnitOfWork()->isReadOnly($entity))
This commit is contained in:
parent
2166a21511
commit
3801e0c230
4 changed files with 81 additions and 6 deletions
|
@ -413,6 +413,7 @@ class EntityManager implements ObjectManager
|
||||||
$entity = $class->newInstance();
|
$entity = $class->newInstance();
|
||||||
$class->setIdentifierValues($entity, $identifier);
|
$class->setIdentifierValues($entity, $identifier);
|
||||||
$this->unitOfWork->registerManaged($entity, $identifier, array());
|
$this->unitOfWork->registerManaged($entity, $identifier, array());
|
||||||
|
$this->unitOfWork->markReadOnly($entity);
|
||||||
|
|
||||||
return $entity;
|
return $entity;
|
||||||
}
|
}
|
||||||
|
|
|
@ -215,8 +215,13 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
private $orphanRemovals = array();
|
private $orphanRemovals = array();
|
||||||
|
|
||||||
//private $_readOnlyObjects = array();
|
/**
|
||||||
|
* Read-Only objects are never evaluated
|
||||||
|
*
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
private $readOnlyObjects = array();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Map of Entity Class-Names and corresponding IDs that should eager loaded when requested.
|
* Map of Entity Class-Names and corresponding IDs that should eager loaded when requested.
|
||||||
|
@ -403,6 +408,11 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
}
|
}
|
||||||
|
|
||||||
$oid = spl_object_hash($entity);
|
$oid = spl_object_hash($entity);
|
||||||
|
|
||||||
|
if (isset($this->readOnlyObjects[$oid])) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$actualData = array();
|
$actualData = array();
|
||||||
foreach ($class->reflFields as $name => $refProp) {
|
foreach ($class->reflFields as $name => $refProp) {
|
||||||
$value = $refProp->getValue($entity);
|
$value = $refProp->getValue($entity);
|
||||||
|
@ -459,6 +469,15 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
|
|
||||||
foreach ($actualData as $propName => $actualValue) {
|
foreach ($actualData as $propName => $actualValue) {
|
||||||
$orgValue = isset($originalData[$propName]) ? $originalData[$propName] : null;
|
$orgValue = isset($originalData[$propName]) ? $originalData[$propName] : null;
|
||||||
|
if (isset($originalData[$propName])) {
|
||||||
|
$orgValue = $originalData[$propName];
|
||||||
|
} else if (array_key_exists($propName, $originalData)) {
|
||||||
|
$orgValue = null;
|
||||||
|
} else {
|
||||||
|
// skip field, its a partially omitted one!
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (isset($class->associationMappings[$propName])) {
|
if (isset($class->associationMappings[$propName])) {
|
||||||
$assoc = $class->associationMappings[$propName];
|
$assoc = $class->associationMappings[$propName];
|
||||||
if ($assoc['type'] & ClassMetadata::TO_ONE && $orgValue !== $actualValue) {
|
if ($assoc['type'] & ClassMetadata::TO_ONE && $orgValue !== $actualValue) {
|
||||||
|
@ -528,7 +547,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
|
|
||||||
foreach ($entitiesToProcess as $entity) {
|
foreach ($entitiesToProcess as $entity) {
|
||||||
// Ignore uninitialized proxy objects
|
// Ignore uninitialized proxy objects
|
||||||
if (/* $entity is readOnly || */ $entity instanceof Proxy && ! $entity->__isInitialized__) {
|
if ($entity instanceof Proxy && ! $entity->__isInitialized__) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
// Only MANAGED entities that are NOT SCHEDULED FOR INSERTION are processed here.
|
// Only MANAGED entities that are NOT SCHEDULED FOR INSERTION are processed here.
|
||||||
|
@ -2407,4 +2426,37 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
{
|
{
|
||||||
return method_exists($obj, '__toString') ? (string)$obj : get_class($obj).'@'.spl_object_hash($obj);
|
return method_exists($obj, '__toString') ? (string)$obj : get_class($obj).'@'.spl_object_hash($obj);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Marks an entity as read-only so that it will not be considered for updates during UnitOfWork#commit().
|
||||||
|
*
|
||||||
|
* This operation cannot be undone as some parts of the UnitOfWork now keep gathering information
|
||||||
|
* on this object that might be necessary to perform a correct udpate.
|
||||||
|
*
|
||||||
|
* @throws \InvalidArgumentException
|
||||||
|
* @param $object
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function markReadOnly($object)
|
||||||
|
{
|
||||||
|
if ( ! is_object($object) || ! $this->isInIdentityMap($object)) {
|
||||||
|
throw new InvalidArgumentException("Managed entity required");
|
||||||
|
}
|
||||||
|
$this->readOnlyObjects[spl_object_hash($object)] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is this entity read only?
|
||||||
|
*
|
||||||
|
* @throws \InvalidArgumentException
|
||||||
|
* @param $object
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function isReadOnly($object)
|
||||||
|
{
|
||||||
|
if ( ! is_object($object) ) {
|
||||||
|
throw new InvalidArgumentException("Managed entity required");
|
||||||
|
}
|
||||||
|
return $this->readOnlyObjects[spl_object_hash($object)];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -863,7 +863,6 @@ class BasicFunctionalTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
|
|
||||||
public function testGetPartialReferenceToUpdateObjectWithoutLoadingIt()
|
public function testGetPartialReferenceToUpdateObjectWithoutLoadingIt()
|
||||||
{
|
{
|
||||||
//$this->_em->getConnection()->getConfiguration()->setSQLLogger(new \Doctrine\DBAL\Logging\EchoSQLLogger);
|
|
||||||
$user = new CmsUser();
|
$user = new CmsUser();
|
||||||
$user->username = "beberlei";
|
$user->username = "beberlei";
|
||||||
$user->name = "Benjamin E.";
|
$user->name = "Benjamin E.";
|
||||||
|
@ -882,7 +881,7 @@ class BasicFunctionalTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$this->_em->flush();
|
$this->_em->flush();
|
||||||
$this->_em->clear();
|
$this->_em->clear();
|
||||||
|
|
||||||
$this->assertEquals('Stephan', $this->_em->find(get_class($user), $userId)->name);
|
$this->assertEquals('Benjamin E.', $this->_em->find(get_class($user), $userId)->name);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testMergePersistsNewEntities()
|
public function testMergePersistsNewEntities()
|
||||||
|
|
|
@ -54,7 +54,30 @@ class DefaultValuesTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
$this->assertEquals($userId, $a2->getUser()->getId());
|
$this->assertEquals($userId, $a2->getUser()->getId());
|
||||||
$this->assertEquals('Poweruser', $a2->getUser()->type);
|
$this->assertEquals('Poweruser', $a2->getUser()->type);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group DDC-1386
|
||||||
|
*/
|
||||||
|
public function testGetPartialReferenceWithDefaultValueNotEvalutedInFlush()
|
||||||
|
{
|
||||||
|
$user = new DefaultValueUser;
|
||||||
|
$user->name = 'romanb';
|
||||||
|
$user->type = 'Normaluser';
|
||||||
|
|
||||||
|
$this->_em->persist($user);
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$user = $this->_em->getPartialReference('Doctrine\Tests\ORM\Functional\DefaultValueUser', $user->id);
|
||||||
|
$this->assertTrue($this->_em->getUnitOfWork()->isReadOnly($user));
|
||||||
|
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$user = $this->_em->find('Doctrine\Tests\ORM\Functional\DefaultValueUser', $user->id);
|
||||||
|
|
||||||
|
$this->assertEquals('Normaluser', $user->type);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue