Skip to content

Commit 17bc1ad

Browse files
Zacharias Luitenzluiten
Zacharias Luiten
authored andcommitted
Fix #4252 Cascade first before refreshing the entity itself so toMany associations are not reset to empty collections
1 parent 676a1bb commit 17bc1ad

File tree

2 files changed

+44
-28
lines changed

2 files changed

+44
-28
lines changed

lib/Doctrine/ORM/UnitOfWork.php

+2-2
Original file line numberDiff line numberDiff line change
@@ -2216,12 +2216,12 @@ private function doRefresh($entity, array &$visited): void
22162216
throw ORMInvalidArgumentException::entityNotManaged($entity);
22172217
}
22182218

2219+
$this->cascadeRefresh($entity, $visited);
2220+
22192221
$this->getEntityPersister($class->name)->refresh(
22202222
array_combine($class->getIdentifierFieldNames(), $this->entityIdentifiers[$oid]),
22212223
$entity
22222224
);
2223-
2224-
$this->cascadeRefresh($entity, $visited);
22252225
}
22262226

22272227
/**

tests/Doctrine/Tests/ORM/Functional/Ticket/GH4252Test.php

+42-26
Original file line numberDiff line numberDiff line change
@@ -1,48 +1,60 @@
11
<?php
22

3+
declare(strict_types=1);
4+
35
namespace Doctrine\Tests\ORM\Functional\Ticket;
46

57
use Doctrine\Common\Collections\ArrayCollection;
68
use Doctrine\Common\Collections\Collection;
9+
use Doctrine\ORM\Mapping\Column;
10+
use Doctrine\ORM\Mapping\Entity;
11+
use Doctrine\ORM\Mapping\GeneratedValue;
12+
use Doctrine\ORM\Mapping\Id;
13+
use Doctrine\ORM\Mapping\ManyToMany;
14+
use Doctrine\ORM\Mapping\ManyToOne;
15+
use Doctrine\ORM\Mapping\OneToMany;
16+
use Doctrine\Tests\OrmFunctionalTestCase;
17+
18+
use function assert;
719

820
/**
921
* @group GH-4252
1022
*/
11-
class GH4252Test extends \Doctrine\Tests\OrmFunctionalTestCase
23+
class GH4252Test extends OrmFunctionalTestCase
1224
{
13-
protected function setUp()
25+
protected function setUp(): void
1426
{
1527
parent::setUp();
1628

17-
$this->_schemaTool->createSchema(array(
29+
$this->_schemaTool->createSchema([
1830
$this->_em->getClassMetadata(GH4252City::class),
1931
$this->_em->getClassMetadata(GH4252Resident::class),
2032
$this->_em->getClassMetadata(GH4252Address::class),
21-
));
33+
]);
2234
}
2335

24-
public function testIssue()
36+
public function testIssue(): void
2537
{
2638
$city = new GH4252City([new GH4252Resident([new GH4252Address()])]);
2739

2840
$this->_em->persist($city);
2941
$this->_em->flush();
3042
$this->_em->clear();
3143

32-
/** @var GH4252City $city */
3344
$city = $this->_em->find(GH4252City::class, $city->getId());
45+
assert($city instanceof GH4252City);
3446
$city->setFlag(false);
35-
/** @var GH4252Resident $resident */
3647
$resident = $city->getResidents()->first();
48+
assert($resident instanceof GH4252Resident);
3749
$resident->setFlag(false);
38-
/** @var GH4252Address $address */
3950
$address = $resident->getAddresses()->first();
51+
assert($address instanceof GH4252Address);
4052
$address->setFlag(false);
4153

4254
$this->_em->refresh($city);
4355

4456
$resident = $city->getResidents()->first();
45-
$address = $resident->getAddresses()->first();
57+
$address = $resident->getAddresses()->first();
4658

4759
$this->assertTrue($city->getFlag());
4860
$this->assertTrue($resident->getFlag());
@@ -57,7 +69,9 @@ class GH4252City
5769
{
5870
/**
5971
* @var int
60-
* @Id @Column(type="integer") @GeneratedValue
72+
* @Id
73+
* @Column(type="integer")
74+
* @GeneratedValue
6175
*/
6276
private $id;
6377

@@ -69,7 +83,6 @@ class GH4252City
6983

7084
/**
7185
* @var GH4252Resident[]|Collection
72-
*
7386
* @OneToMany(targetEntity="Doctrine\Tests\ORM\Functional\Ticket\GH4252Resident", mappedBy="city", cascade={"persist","refresh"})
7487
*/
7588
private $residents;
@@ -81,25 +94,26 @@ public function __construct(array $residents)
8194
$this->residents->add($resident);
8295
$resident->setCity($this);
8396
}
97+
8498
$this->flag = true;
8599
}
86100

87-
public function getId() : int
101+
public function getId(): int
88102
{
89103
return $this->id;
90104
}
91105

92-
public function getFlag() : bool
106+
public function getFlag(): bool
93107
{
94108
return $this->flag;
95109
}
96110

97-
public function setFlag(bool $flag) : void
111+
public function setFlag(bool $flag): void
98112
{
99113
$this->flag = $flag;
100114
}
101115

102-
public function getResidents() : Collection
116+
public function getResidents(): Collection
103117
{
104118
return $this->residents;
105119
}
@@ -130,7 +144,6 @@ class GH4252Resident
130144

131145
/**
132146
* @var GH4252Address[]|Collection
133-
*
134147
* @ManyToMany(targetEntity="Doctrine\Tests\ORM\Functional\Ticket\GH4252Address", fetch="EXTRA_LAZY", cascade={"persist","refresh"})
135148
*/
136149
private $addresses;
@@ -141,35 +154,36 @@ public function __construct(array $addresses)
141154
foreach ($addresses as $address) {
142155
$this->addresses->add($address);
143156
}
157+
144158
$this->flag = true;
145159
}
146160

147-
public function getId() : int
161+
public function getId(): int
148162
{
149163
return $this->id;
150164
}
151165

152-
public function getCity() : GH4252City
166+
public function getCity(): GH4252City
153167
{
154168
return $this->city;
155169
}
156170

157-
public function setCity(GH4252City $city) : void
171+
public function setCity(GH4252City $city): void
158172
{
159173
$this->city = $city;
160174
}
161175

162-
public function getFlag() : bool
176+
public function getFlag(): bool
163177
{
164178
return $this->flag;
165179
}
166180

167-
public function setFlag(bool $flag) : void
181+
public function setFlag(bool $flag): void
168182
{
169183
$this->flag = $flag;
170184
}
171185

172-
public function getAddresses() : Collection
186+
public function getAddresses(): Collection
173187
{
174188
return $this->addresses;
175189
}
@@ -180,7 +194,9 @@ class GH4252Address
180194
{
181195
/**
182196
* @var int
183-
* @Id @Column(type="integer") @GeneratedValue
197+
* @Id
198+
* @Column(type="integer")
199+
* @GeneratedValue
184200
*/
185201
private $id;
186202

@@ -195,17 +211,17 @@ public function __construct()
195211
$this->flag = true;
196212
}
197213

198-
public function getId() : int
214+
public function getId(): int
199215
{
200216
return $this->id;
201217
}
202218

203-
public function getFlag() : bool
219+
public function getFlag(): bool
204220
{
205221
return $this->flag;
206222
}
207223

208-
public function setFlag(bool $flag) : void
224+
public function setFlag(bool $flag): void
209225
{
210226
$this->flag = $flag;
211227
}

0 commit comments

Comments
 (0)