[DDC-119] Fixed.
This commit is contained in:
parent
d288e99a34
commit
e7ac35ed95
5 changed files with 250 additions and 32 deletions
|
@ -279,14 +279,15 @@ final class PersistentCollection implements Collection
|
||||||
{
|
{
|
||||||
if ( ! $this->isDirty) {
|
if ( ! $this->isDirty) {
|
||||||
$this->isDirty = true;
|
$this->isDirty = true;
|
||||||
//if ($this->isNotifyRequired) {
|
if ($this->association !== null && $this->association->isOwningSide && $this->association->isManyToMany() &&
|
||||||
//$this->em->getUnitOfWork()->scheduleCollectionUpdate($this);
|
$this->em->getClassMetadata(get_class($this->owner))->isChangeTrackingNotify()) {
|
||||||
//}
|
$this->em->getUnitOfWork()->scheduleForDirtyCheck($this->owner);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets a boolean flag indicating whether this colleciton is dirty which means
|
* Gets a boolean flag indicating whether this collection is dirty which means
|
||||||
* its state needs to be synchronized with the database.
|
* its state needs to be synchronized with the database.
|
||||||
*
|
*
|
||||||
* @return boolean TRUE if the collection is dirty, FALSE otherwise.
|
* @return boolean TRUE if the collection is dirty, FALSE otherwise.
|
||||||
|
|
|
@ -1656,6 +1656,7 @@ class Parser
|
||||||
$peek = $this->_lexer->glimpse();
|
$peek = $this->_lexer->glimpse();
|
||||||
|
|
||||||
$supportsAlias = true;
|
$supportsAlias = true;
|
||||||
|
|
||||||
if ($peek['value'] != '(' && $this->_lexer->lookahead['type'] === Lexer::T_IDENTIFIER) {
|
if ($peek['value'] != '(' && $this->_lexer->lookahead['type'] === Lexer::T_IDENTIFIER) {
|
||||||
if ($peek['value'] == '.') {
|
if ($peek['value'] == '.') {
|
||||||
// ScalarExpression
|
// ScalarExpression
|
||||||
|
|
|
@ -405,8 +405,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
}
|
}
|
||||||
|
|
||||||
$oid = spl_object_hash($entity);
|
$oid = spl_object_hash($entity);
|
||||||
|
|
||||||
//TODO: Skip this step if changetracking = NOTIFY
|
|
||||||
$actualData = array();
|
$actualData = array();
|
||||||
foreach ($class->reflFields as $name => $refProp) {
|
foreach ($class->reflFields as $name => $refProp) {
|
||||||
$value = $refProp->getValue($entity);
|
$value = $refProp->getValue($entity);
|
||||||
|
@ -455,7 +453,8 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
// Entity is "fully" MANAGED: it was already fully persisted before
|
// Entity is "fully" MANAGED: it was already fully persisted before
|
||||||
// and we have a copy of the original data
|
// and we have a copy of the original data
|
||||||
$originalData = $this->originalEntityData[$oid];
|
$originalData = $this->originalEntityData[$oid];
|
||||||
$changeSet = array();
|
$isChangeTrackingNotify = isset($this->entityChangeSets[$oid]);
|
||||||
|
$changeSet = $isChangeTrackingNotify ? $this->entityChangeSets[$oid] : array();
|
||||||
|
|
||||||
foreach ($actualData as $propName => $actualValue) {
|
foreach ($actualData as $propName => $actualValue) {
|
||||||
$orgValue = isset($originalData[$propName]) ? $originalData[$propName] : null;
|
$orgValue = isset($originalData[$propName]) ? $originalData[$propName] : null;
|
||||||
|
@ -474,6 +473,8 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
$this->collectionDeletions[] = $orgValue;
|
$this->collectionDeletions[] = $orgValue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else if ($isChangeTrackingNotify) {
|
||||||
|
continue;
|
||||||
} else if (is_object($orgValue) && $orgValue !== $actualValue) {
|
} else if (is_object($orgValue) && $orgValue !== $actualValue) {
|
||||||
$changeSet[$propName] = array($orgValue, $actualValue);
|
$changeSet[$propName] = array($orgValue, $actualValue);
|
||||||
} else if ($orgValue != $actualValue || ($orgValue === null ^ $actualValue === null)) {
|
} else if ($orgValue != $actualValue || ($orgValue === null ^ $actualValue === null)) {
|
||||||
|
@ -513,13 +514,14 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
foreach ($this->identityMap as $className => $entities) {
|
foreach ($this->identityMap as $className => $entities) {
|
||||||
$class = $this->em->getClassMetadata($className);
|
$class = $this->em->getClassMetadata($className);
|
||||||
|
|
||||||
// Skip class if change tracking happens through notification
|
// Skip class if instances are read-only
|
||||||
if ($class->isChangeTrackingNotify() /* || $class->isReadOnly*/) {
|
//if ($class->isReadOnly) {
|
||||||
continue;
|
// continue;
|
||||||
}
|
//}
|
||||||
|
|
||||||
// If change tracking is explicit, then only compute changes on explicitly persisted entities
|
// If change tracking is explicit or happens through notification, then only compute
|
||||||
$entitiesToProcess = $class->isChangeTrackingDeferredExplicit() ?
|
// changes on entities of that type that are explicitly marked for synchronization.
|
||||||
|
$entitiesToProcess = ! $class->isChangeTrackingDeferredImplicit() ?
|
||||||
(isset($this->scheduledForDirtyCheck[$className]) ?
|
(isset($this->scheduledForDirtyCheck[$className]) ?
|
||||||
$this->scheduledForDirtyCheck[$className] : array())
|
$this->scheduledForDirtyCheck[$className] : array())
|
||||||
: $entities;
|
: $entities;
|
||||||
|
@ -958,6 +960,12 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
return isset($this->entityUpdates[spl_object_hash($entity)]);
|
return isset($this->entityUpdates[spl_object_hash($entity)]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function isScheduledForDirtyCheck($entity)
|
||||||
|
{
|
||||||
|
$rootEntityName = $this->em->getClassMetadata(get_class($entity))->rootEntityName;
|
||||||
|
return isset($this->scheduledForDirtyCheck[$rootEntityName][spl_object_hash($entity)]);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* INTERNAL:
|
* INTERNAL:
|
||||||
* Schedules an entity for deletion.
|
* Schedules an entity for deletion.
|
||||||
|
@ -2010,7 +2018,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
public function scheduleForDirtyCheck($entity)
|
public function scheduleForDirtyCheck($entity)
|
||||||
{
|
{
|
||||||
$rootClassName = $this->em->getClassMetadata(get_class($entity))->rootEntityName;
|
$rootClassName = $this->em->getClassMetadata(get_class($entity))->rootEntityName;
|
||||||
$this->scheduledForDirtyCheck[$rootClassName][] = $entity;
|
$this->scheduledForDirtyCheck[$rootClassName][spl_object_hash($entity)] = $entity;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -2119,7 +2127,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
* @param string $propertyName The name of the property that changed.
|
* @param string $propertyName The name of the property that changed.
|
||||||
* @param mixed $oldValue The old value of the property.
|
* @param mixed $oldValue The old value of the property.
|
||||||
* @param mixed $newValue The new value of the property.
|
* @param mixed $newValue The new value of the property.
|
||||||
* @todo Simply use _scheduledForDirtyCheck, otherwise a lot of behavior is potentially inconsistent.
|
|
||||||
*/
|
*/
|
||||||
public function propertyChanged($entity, $propertyName, $oldValue, $newValue)
|
public function propertyChanged($entity, $propertyName, $oldValue, $newValue)
|
||||||
{
|
{
|
||||||
|
@ -2132,24 +2139,13 @@ class UnitOfWork implements PropertyChangedListener
|
||||||
return; // ignore non-persistent fields
|
return; // ignore non-persistent fields
|
||||||
}
|
}
|
||||||
|
|
||||||
//$this->scheduleForDirtyCheck($entity);
|
// Update changeset and mark entity for synchronization
|
||||||
$this->entityChangeSets[$oid][$propertyName] = array($oldValue, $newValue);
|
$this->entityChangeSets[$oid][$propertyName] = array($oldValue, $newValue);
|
||||||
|
if ( ! isset($this->scheduledForDirtyCheck[$class->rootEntityName][$oid])) {
|
||||||
if ($isAssocField) {
|
$this->scheduleForDirtyCheck($entity);
|
||||||
$assoc = $class->associationMappings[$propertyName];
|
|
||||||
if ($assoc->isOneToOne() && $assoc->isOwningSide) {
|
|
||||||
$this->entityUpdates[$oid] = $entity;
|
|
||||||
} else if ($oldValue instanceof PersistentCollection) {
|
|
||||||
// A PersistentCollection was de-referenced, so delete it.
|
|
||||||
if ( ! in_array($oldValue, $this->collectionDeletions, true)) {
|
|
||||||
$this->collectionDeletions[] = $oldValue;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
$this->entityUpdates[$oid] = $entity;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the currently scheduled entity insertions in this UnitOfWork.
|
* Gets the currently scheduled entity insertions in this UnitOfWork.
|
||||||
*
|
*
|
||||||
|
|
160
tests/Doctrine/Tests/ORM/Functional/NotifyPolicyTest.php
Normal file
160
tests/Doctrine/Tests/ORM/Functional/NotifyPolicyTest.php
Normal file
|
@ -0,0 +1,160 @@
|
||||||
|
<?php
|
||||||
|
namespace Doctrine\Tests\ORM\Functional;
|
||||||
|
|
||||||
|
use Doctrine\Common\Collections\ArrayCollection,
|
||||||
|
Doctrine\Common\NotifyPropertyChanged,
|
||||||
|
Doctrine\Common\PropertyChangedListener;
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../../TestInit.php';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* NativeQueryTest
|
||||||
|
*
|
||||||
|
* @author robo
|
||||||
|
*/
|
||||||
|
class NotifyPolicyTest extends \Doctrine\Tests\OrmFunctionalTestCase
|
||||||
|
{
|
||||||
|
protected function setUp() {
|
||||||
|
parent::setUp();
|
||||||
|
try {
|
||||||
|
$this->_schemaTool->createSchema(array(
|
||||||
|
$this->_em->getClassMetadata('Doctrine\Tests\ORM\Functional\NotifyUser'),
|
||||||
|
$this->_em->getClassMetadata('Doctrine\Tests\ORM\Functional\NotifyGroup')
|
||||||
|
));
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
// Swallow all exceptions. We do not test the schema tool here.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public function testChangeTracking()
|
||||||
|
{
|
||||||
|
//$this->_em->getConnection()->getConfiguration()->setSQLLogger(new \Doctrine\DBAL\Logging\EchoSQLLogger);
|
||||||
|
|
||||||
|
$user = new NotifyUser();
|
||||||
|
$group = new NotifyGroup();
|
||||||
|
$user->setName('roman');
|
||||||
|
$group->setName('dev');
|
||||||
|
|
||||||
|
$user->getGroups()->add($group);
|
||||||
|
$group->getUsers()->add($user);
|
||||||
|
|
||||||
|
$this->_em->persist($user);
|
||||||
|
$this->_em->persist($group);
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$userId = $user->getId();
|
||||||
|
$groupId = $group->getId();
|
||||||
|
unset($user, $group);
|
||||||
|
|
||||||
|
$user = $this->_em->find(__NAMESPACE__.'\NotifyUser', $userId);
|
||||||
|
$this->assertEquals(1, $user->getGroups()->count());
|
||||||
|
$group = $this->_em->find(__NAMESPACE__.'\NotifyGroup', $groupId);
|
||||||
|
$this->assertEquals(1, $group->getUsers()->count());
|
||||||
|
|
||||||
|
$group2 = new NotifyGroup();
|
||||||
|
$group2->setName('nerds');
|
||||||
|
$this->_em->persist($group2);
|
||||||
|
$user->getGroups()->add($group2);
|
||||||
|
$group2->getUsers()->add($user);
|
||||||
|
|
||||||
|
$group->setName('geeks');
|
||||||
|
|
||||||
|
$this->_em->flush();
|
||||||
|
$this->_em->clear();
|
||||||
|
|
||||||
|
$group2Id = $group2->getId();
|
||||||
|
unset($group2, $user);
|
||||||
|
|
||||||
|
$user = $this->_em->find(__NAMESPACE__.'\NotifyUser', $userId);
|
||||||
|
$this->assertEquals(2, $user->getGroups()->count());
|
||||||
|
$group2 = $this->_em->find(__NAMESPACE__.'\NotifyGroup', $group2Id);
|
||||||
|
$this->assertEquals(1, $group2->getUsers()->count());
|
||||||
|
$group = $this->_em->find(__NAMESPACE__.'\NotifyGroup', $groupId);
|
||||||
|
$this->assertEquals(1, $group->getUsers()->count());
|
||||||
|
$this->assertEquals('geeks', $group->getName());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class NotifyBaseEntity implements NotifyPropertyChanged {
|
||||||
|
private $listeners = array();
|
||||||
|
|
||||||
|
public function addPropertyChangedListener(PropertyChangedListener $listener) {
|
||||||
|
$this->listeners[] = $listener;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function onPropertyChanged($propName, $oldValue, $newValue) {
|
||||||
|
if ($this->listeners) {
|
||||||
|
foreach ($this->listeners as $listener) {
|
||||||
|
$listener->propertyChanged($this, $propName, $oldValue, $newValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @Entity @ChangeTrackingPolicy("NOTIFY") */
|
||||||
|
class NotifyUser extends NotifyBaseEntity {
|
||||||
|
/** @Id @Column(type="integer") @GeneratedValue */
|
||||||
|
private $id;
|
||||||
|
|
||||||
|
/** @Column */
|
||||||
|
private $name;
|
||||||
|
|
||||||
|
/** @ManyToMany(targetEntity="NotifyGroup") */
|
||||||
|
private $groups;
|
||||||
|
|
||||||
|
function __construct() {
|
||||||
|
$this->groups = new ArrayCollection;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getId() {
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getName() {
|
||||||
|
return $this->name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setName($name) {
|
||||||
|
$this->onPropertyChanged('name', $this->name, $name);
|
||||||
|
$this->name = $name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getGroups() {
|
||||||
|
return $this->groups;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @Entity */
|
||||||
|
class NotifyGroup extends NotifyBaseEntity {
|
||||||
|
/** @Id @Column(type="integer") @GeneratedValue */
|
||||||
|
private $id;
|
||||||
|
|
||||||
|
/** @Column */
|
||||||
|
private $name;
|
||||||
|
|
||||||
|
/** @ManyToMany(targetEntity="NotifyUser", mappedBy="groups") */
|
||||||
|
private $users;
|
||||||
|
|
||||||
|
function __construct() {
|
||||||
|
$this->users = new ArrayCollection;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getId() {
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getName() {
|
||||||
|
return $this->name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setName($name) {
|
||||||
|
$this->onPropertyChanged('name', $this->name, $name);
|
||||||
|
$this->name = $name;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getUsers() {
|
||||||
|
return $this->users;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -131,21 +131,44 @@ class UnitOfWorkTest extends \Doctrine\Tests\OrmTestCase
|
||||||
{
|
{
|
||||||
$persister = new EntityPersisterMock($this->_emMock, $this->_emMock->getClassMetadata("Doctrine\Tests\ORM\NotifyChangedEntity"));
|
$persister = new EntityPersisterMock($this->_emMock, $this->_emMock->getClassMetadata("Doctrine\Tests\ORM\NotifyChangedEntity"));
|
||||||
$this->_unitOfWork->setEntityPersister('Doctrine\Tests\ORM\NotifyChangedEntity', $persister);
|
$this->_unitOfWork->setEntityPersister('Doctrine\Tests\ORM\NotifyChangedEntity', $persister);
|
||||||
|
$itemPersister = new EntityPersisterMock($this->_emMock, $this->_emMock->getClassMetadata("Doctrine\Tests\ORM\NotifyChangedRelatedItem"));
|
||||||
|
$this->_unitOfWork->setEntityPersister('Doctrine\Tests\ORM\NotifyChangedRelatedItem', $itemPersister);
|
||||||
|
|
||||||
$entity = new NotifyChangedEntity;
|
$entity = new NotifyChangedEntity;
|
||||||
$entity->setData('thedata');
|
$entity->setData('thedata');
|
||||||
$this->_unitOfWork->persist($entity);
|
$this->_unitOfWork->persist($entity);
|
||||||
|
|
||||||
$this->_unitOfWork->commit();
|
$this->_unitOfWork->commit();
|
||||||
|
$this->assertEquals(1, count($persister->getInserts()));
|
||||||
|
$persister->reset();
|
||||||
|
|
||||||
$this->assertTrue($this->_unitOfWork->isInIdentityMap($entity));
|
$this->assertTrue($this->_unitOfWork->isInIdentityMap($entity));
|
||||||
|
|
||||||
$entity->setData('newdata');
|
$entity->setData('newdata');
|
||||||
$entity->setTransient('newtransientvalue');
|
$entity->setTransient('newtransientvalue');
|
||||||
|
|
||||||
$this->assertTrue($this->_unitOfWork->isScheduledForUpdate($entity));
|
$this->assertTrue($this->_unitOfWork->isScheduledForDirtyCheck($entity));
|
||||||
|
|
||||||
$this->assertEquals(array('data' => array('thedata', 'newdata')), $this->_unitOfWork->getEntityChangeSet($entity));
|
$this->assertEquals(array('data' => array('thedata', 'newdata')), $this->_unitOfWork->getEntityChangeSet($entity));
|
||||||
|
|
||||||
|
$item = new NotifyChangedRelatedItem();
|
||||||
|
$entity->getItems()->add($item);
|
||||||
|
$item->setOwner($entity);
|
||||||
|
$this->_unitOfWork->persist($item);
|
||||||
|
|
||||||
|
$this->_unitOfWork->commit();
|
||||||
|
$this->assertEquals(1, count($itemPersister->getInserts()));
|
||||||
|
$persister->reset();
|
||||||
|
$itemPersister->reset();
|
||||||
|
|
||||||
|
|
||||||
|
$entity->getItems()->removeElement($item);
|
||||||
|
$item->setOwner(null);
|
||||||
|
$this->assertTrue($entity->getItems()->isDirty());
|
||||||
|
$this->_unitOfWork->commit();
|
||||||
|
$updates = $itemPersister->getUpdates();
|
||||||
|
$this->assertEquals(1, count($updates));
|
||||||
|
$this->assertTrue($updates[0] === $item);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function testGetEntityStateOnVersionedEntityWithAssignedIdentifier()
|
public function testGetEntityStateOnVersionedEntityWithAssignedIdentifier()
|
||||||
|
@ -192,7 +215,7 @@ class NotifyChangedEntity implements \Doctrine\Common\NotifyPropertyChanged
|
||||||
/**
|
/**
|
||||||
* @Id
|
* @Id
|
||||||
* @Column(type="integer")
|
* @Column(type="integer")
|
||||||
* @GeneratedValue(strategy="AUTO")
|
* @GeneratedValue
|
||||||
*/
|
*/
|
||||||
private $id;
|
private $id;
|
||||||
/**
|
/**
|
||||||
|
@ -202,10 +225,21 @@ class NotifyChangedEntity implements \Doctrine\Common\NotifyPropertyChanged
|
||||||
|
|
||||||
private $transient; // not persisted
|
private $transient; // not persisted
|
||||||
|
|
||||||
|
/** @OneToMany(targetEntity="NotifyChangedRelatedItem", mappedBy="owner") */
|
||||||
|
private $items;
|
||||||
|
|
||||||
|
public function __construct() {
|
||||||
|
$this->items = new \Doctrine\Common\Collections\ArrayCollection;
|
||||||
|
}
|
||||||
|
|
||||||
public function getId() {
|
public function getId() {
|
||||||
return $this->id;
|
return $this->id;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function getItems() {
|
||||||
|
return $this->items;
|
||||||
|
}
|
||||||
|
|
||||||
public function setTransient($value) {
|
public function setTransient($value) {
|
||||||
if ($value != $this->transient) {
|
if ($value != $this->transient) {
|
||||||
$this->_onPropertyChanged('transient', $this->transient, $value);
|
$this->_onPropertyChanged('transient', $this->transient, $value);
|
||||||
|
@ -238,6 +272,32 @@ class NotifyChangedEntity implements \Doctrine\Common\NotifyPropertyChanged
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @Entity */
|
||||||
|
class NotifyChangedRelatedItem
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* @Id
|
||||||
|
* @Column(type="integer")
|
||||||
|
* @GeneratedValue
|
||||||
|
*/
|
||||||
|
private $id;
|
||||||
|
|
||||||
|
/** @ManyToOne(targetEntity="NotifyChangedEntity", inversedBy="items") */
|
||||||
|
private $owner;
|
||||||
|
|
||||||
|
public function getId() {
|
||||||
|
return $this->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getOwner() {
|
||||||
|
return $this->owner;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function setOwner($owner) {
|
||||||
|
$this->owner = $owner;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/** @Entity */
|
/** @Entity */
|
||||||
class VersionedAssignedIdentifierEntity
|
class VersionedAssignedIdentifierEntity
|
||||||
{
|
{
|
||||||
|
|
Loading…
Add table
Reference in a new issue