From 16a14f223839f173b5058d4ac35915e30c6c659d Mon Sep 17 00:00:00 2001 From: Gabe van der Weijde <gabe.vanderweijde@triasinformatica.nl> Date: Wed, 28 Jun 2017 20:40:25 +0200 Subject: [PATCH 1/8] -- Created test for validation issue #6499. --- .../ORM/Functional/Ticket/DDC6499Test.php | 172 ++++++++++++++++++ 1 file changed, 172 insertions(+) create mode 100644 tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php new file mode 100644 index 000000000..98b0e42a7 --- /dev/null +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -0,0 +1,172 @@ +<?php + +namespace Doctrine\Tests\Functional\Ticket; + +use Doctrine\Common\Collections\ArrayCollection; +use Doctrine\Tests\OrmFunctionalTestCase; + +/** + * @group DDC-6499 + */ +class DDC6499Test extends OrmFunctionalTestCase +{ + /** + * {@inheritDoc} + */ + protected function setUp() + { + parent::setUp(); + + $this->_schemaTool->createSchema( + [ + $this->_em->getClassMetadata(DDC6499A::class), + $this->_em->getClassMetadata(DDC6499B::class), + ] + ); + } + + /** + * Test for the bug described in issue #6499. + */ + public function testIssue() + { + $a = new DDC6499A(); + $this->_em->persist($a); + + $b = new DDC6499B(); + $a->setB($b); + $this->_em->persist($b); + + $this->_em->flush(); + + // Issue #6499 will result in a Integrity constraint violation before reaching this point + $this->assertEquals(true, true); + } +} + +/** @Entity */ +class DDC6499A +{ + /** + * @Id() + * @GeneratedValue(strategy="AUTO") + * @Column(name="id", type="integer") + */ + private $id; + + /** + * @OneToMany(targetEntity="DDC6499B", mappedBy="a", cascade={"persist", "remove"}, orphanRemoval=true) + */ + private $bs; + + /** + * @OneToOne(targetEntity="DDC6499B", cascade={"persist"}) + * @JoinColumn(nullable=false) + */ + private $b; + + /** + * DDC6499A constructor. + */ + public function __construct() + { + $this->bs = new ArrayCollection(); + } + + /** + * @return int + */ + public function getId() + { + return $this->id; + } + + /** + * @return DDC6499B[]|ArrayCollection + */ + public function getBs() + { + return $this->bs; + } + + /** + * @param DDC6499B $b + */ + public function addB(DDC6499B $b) + { + if ($this->bs->contains($b)) return; + + $this->bs->add($b); + + // Update owning side + $b->setA($this); + } + + /** + * @param DDC6499B $b + */ + public function removeB(DDC6499B $b) + { + if (!$this->bs->contains($b)) return; + + $this->bs->removeElement($b); + + // Not updating owning side due to orphan removal + } + + /** + * @return DDC6499B + */ + public function getB() + { + return $this->b; + } + + /** + * @param DDC6499B $b + */ + public function setB(DDC6499B $b) + { + $this->b = $b; + } +} + +/** @Entity */ +class DDC6499B +{ + /** + * @Id() + * @GeneratedValue(strategy="AUTO") + * @Column(name="id", type="integer") + */ + private $id; + + /** + * @ManyToOne(targetEntity="DDC6499A", inversedBy="bs", cascade={"persist"}) + */ + private $a; + + /** + * @return int + */ + public function getId() + { + return $this->id; + } + + /** + * @return DDC6499A + */ + public function getA() + { + return $this->a; + } + + /** + * @param DDC6499A $a + */ + public function setA(DDC6499A $a) + { + $this->a = $a; + } +} \ No newline at end of file From db2530d6fdbda608d2eaada74c16d253c89eb59c Mon Sep 17 00:00:00 2001 From: Gabe van der Weijde <gabe.vanderweijde@triasinformatica.nl> Date: Wed, 28 Jun 2017 20:55:58 +0200 Subject: [PATCH 2/8] -- Proposed fix due to logic error. --- lib/Doctrine/ORM/UnitOfWork.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/Doctrine/ORM/UnitOfWork.php b/lib/Doctrine/ORM/UnitOfWork.php index c0b0dd8c3..0afabf805 100644 --- a/lib/Doctrine/ORM/UnitOfWork.php +++ b/lib/Doctrine/ORM/UnitOfWork.php @@ -1154,7 +1154,7 @@ class UnitOfWork implements PropertyChangedListener $joinColumns = reset($assoc['joinColumns']); - $calc->addDependency($targetClass->name, $class->name, (int)empty($joinColumns['nullable'])); + $calc->addDependency($targetClass->name, $class->name, (int) ! ( ! isset($joinColumns['nullable']) || $joinColumns['nullable'] === true)); // If the target class has mapped subclasses, these share the same dependency. if ( ! $targetClass->subClasses) { From da3cd04993b107e6dff8e2e382d5769add8ed0b4 Mon Sep 17 00:00:00 2001 From: Gabe van der Weijde <gabe.vanderweijde@triasinformatica.nl> Date: Thu, 29 Jun 2017 17:16:35 +0200 Subject: [PATCH 3/8] -- Transformed into a minimal example. -- Processed Ocramius' feedback. --- .../ORM/Functional/Ticket/DDC6499Test.php | 158 ++++++------------ 1 file changed, 47 insertions(+), 111 deletions(-) diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php index 98b0e42a7..3cee68cf8 100644 --- a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -13,7 +13,7 @@ class DDC6499Test extends OrmFunctionalTestCase /** * {@inheritDoc} */ - protected function setUp() + protected function setUp() : void { parent::setUp(); @@ -25,22 +25,56 @@ class DDC6499Test extends OrmFunctionalTestCase ); } + /** + * {@inheritDoc} + */ + protected function tearDown() : void + { + parent::tearDown(); + + $this->_schemaTool->dropSchema( + [ + $this->_em->getClassMetadata(DDC6499A::class), + $this->_em->getClassMetadata(DDC6499B::class), + ] + ); + } + /** * Test for the bug described in issue #6499. */ - public function testIssue() + public function testIssue() : void { $a = new DDC6499A(); $this->_em->persist($a); $b = new DDC6499B(); - $a->setB($b); + $a->b = $b; $this->_em->persist($b); $this->_em->flush(); + $this->_em->clear(); - // Issue #6499 will result in a Integrity constraint violation before reaching this point - $this->assertEquals(true, true); + self::assertEquals($this->_em->find(DDC6499A::class, $a->id)->b->id, $b->id, "Issue #6499 will result in a Integrity constraint violation before reaching this point."); + } + + /** + * Test for the bug described in issue #6499 (reversed order). + */ + public function testIssueReversed() : void + { + $a = new DDC6499A(); + + $b = new DDC6499B(); + $a->b = $b; + + $this->_em->persist($b); + $this->_em->persist($a); + + $this->_em->flush(); + $this->_em->clear(); + + self::assertEquals($this->_em->find(DDC6499A::class, $a->id)->b->id, $b->id, "Issue #6499 will result in a Integrity constraint violation before reaching this point."); } } @@ -48,125 +82,27 @@ class DDC6499Test extends OrmFunctionalTestCase class DDC6499A { /** - * @Id() - * @GeneratedValue(strategy="AUTO") - * @Column(name="id", type="integer") + * @Id @Column(type="integer") @GeneratedValue */ - private $id; + public $id; /** - * @OneToMany(targetEntity="DDC6499B", mappedBy="a", cascade={"persist", "remove"}, orphanRemoval=true) - */ - private $bs; - - /** - * @OneToOne(targetEntity="DDC6499B", cascade={"persist"}) + * @OneToOne(targetEntity="DDC6499B") * @JoinColumn(nullable=false) */ - private $b; - - /** - * DDC6499A constructor. - */ - public function __construct() - { - $this->bs = new ArrayCollection(); - } - - /** - * @return int - */ - public function getId() - { - return $this->id; - } - - /** - * @return DDC6499B[]|ArrayCollection - */ - public function getBs() - { - return $this->bs; - } - - /** - * @param DDC6499B $b - */ - public function addB(DDC6499B $b) - { - if ($this->bs->contains($b)) return; - - $this->bs->add($b); - - // Update owning side - $b->setA($this); - } - - /** - * @param DDC6499B $b - */ - public function removeB(DDC6499B $b) - { - if (!$this->bs->contains($b)) return; - - $this->bs->removeElement($b); - - // Not updating owning side due to orphan removal - } - - /** - * @return DDC6499B - */ - public function getB() - { - return $this->b; - } - - /** - * @param DDC6499B $b - */ - public function setB(DDC6499B $b) - { - $this->b = $b; - } + public $b; } /** @Entity */ class DDC6499B { /** - * @Id() - * @GeneratedValue(strategy="AUTO") - * @Column(name="id", type="integer") + * @Id @Column(type="integer") @GeneratedValue */ - private $id; + public $id; /** - * @ManyToOne(targetEntity="DDC6499A", inversedBy="bs", cascade={"persist"}) + * @ManyToOne(targetEntity="DDC6499A", inversedBy="bs") */ - private $a; - - /** - * @return int - */ - public function getId() - { - return $this->id; - } - - /** - * @return DDC6499A - */ - public function getA() - { - return $this->a; - } - - /** - * @param DDC6499A $a - */ - public function setA(DDC6499A $a) - { - $this->a = $a; - } + public $a; } \ No newline at end of file From b352cd3e222db4680a365a1408fafea27daabb90 Mon Sep 17 00:00:00 2001 From: Marco Pivetta <ocramius@gmail.com> Date: Fri, 11 Aug 2017 21:54:30 +0200 Subject: [PATCH 4/8] #6499 #6533 minor CS fixes in the test --- tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php | 6 ++---- 1 file changed, 2 insertions(+), 4 deletions(-) diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php index 3cee68cf8..4b27478c6 100644 --- a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -17,12 +17,10 @@ class DDC6499Test extends OrmFunctionalTestCase { parent::setUp(); - $this->_schemaTool->createSchema( - [ + $this->_schemaTool->createSchema([ $this->_em->getClassMetadata(DDC6499A::class), $this->_em->getClassMetadata(DDC6499B::class), - ] - ); + ]); } /** From ebd521c56eef4a9ad9cec239997554182db1e9a4 Mon Sep 17 00:00:00 2001 From: Marco Pivetta <ocramius@gmail.com> Date: Fri, 11 Aug 2017 21:55:02 +0200 Subject: [PATCH 5/8] #6499 #6533 minor CS fixes in the test --- .../Tests/ORM/Functional/Ticket/DDC6499Test.php | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php index 4b27478c6..066c0b1b4 100644 --- a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -30,12 +30,10 @@ class DDC6499Test extends OrmFunctionalTestCase { parent::tearDown(); - $this->_schemaTool->dropSchema( - [ - $this->_em->getClassMetadata(DDC6499A::class), - $this->_em->getClassMetadata(DDC6499B::class), - ] - ); + $this->_schemaTool->dropSchema([ + $this->_em->getClassMetadata(DDC6499A::class), + $this->_em->getClassMetadata(DDC6499B::class), + ]); } /** From 25829ea450f60fb9f1507444f339edc8b0915669 Mon Sep 17 00:00:00 2001 From: Marco Pivetta <ocramius@gmail.com> Date: Fri, 11 Aug 2017 22:05:00 +0200 Subject: [PATCH 6/8] #6499 #6533 simplifying test scenario to the bone, adding description of what happened at persistence-level --- .../ORM/Functional/Ticket/DDC6499Test.php | 58 ++++++++----------- 1 file changed, 25 insertions(+), 33 deletions(-) diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php index 066c0b1b4..6e8b25336 100644 --- a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -6,7 +6,12 @@ use Doctrine\Common\Collections\ArrayCollection; use Doctrine\Tests\OrmFunctionalTestCase; /** - * @group DDC-6499 + * @group #6499 + * + * + * Specifically, DDC6499B has a dependency on DDC6499A, and DDC6499A + * has a dependency on DDC6499B. Since DDC6499A#b is not nullable, + * the DDC6499B should be inserted first. */ class DDC6499Test extends OrmFunctionalTestCase { @@ -36,69 +41,56 @@ class DDC6499Test extends OrmFunctionalTestCase ]); } - /** - * Test for the bug described in issue #6499. - */ public function testIssue() : void { - $a = new DDC6499A(); - $this->_em->persist($a); - $b = new DDC6499B(); - $a->b = $b; + $a = new DDC6499A($b); + + $this->_em->persist($a); $this->_em->persist($b); $this->_em->flush(); - $this->_em->clear(); - self::assertEquals($this->_em->find(DDC6499A::class, $a->id)->b->id, $b->id, "Issue #6499 will result in a Integrity constraint violation before reaching this point."); + self::assertInternalType('integer', $a->id); + self::assertInternalType('integer', $b->id); } - /** - * Test for the bug described in issue #6499 (reversed order). - */ public function testIssueReversed() : void { - $a = new DDC6499A(); - $b = new DDC6499B(); - $a->b = $b; + $a = new DDC6499A($b); $this->_em->persist($b); $this->_em->persist($a); $this->_em->flush(); - $this->_em->clear(); - self::assertEquals($this->_em->find(DDC6499A::class, $a->id)->b->id, $b->id, "Issue #6499 will result in a Integrity constraint violation before reaching this point."); + self::assertInternalType('integer', $a->id); + self::assertInternalType('integer', $b->id); } } /** @Entity */ class DDC6499A { - /** - * @Id @Column(type="integer") @GeneratedValue - */ + /** @Id @Column(type="integer") @GeneratedValue */ public $id; - /** - * @OneToOne(targetEntity="DDC6499B") - * @JoinColumn(nullable=false) - */ + /** @JoinColumn(nullable=false) @OneToOne(targetEntity=DDC6499B::class) */ public $b; + + public function __construct(DDC6499B $b) + { + $this->b = $b; + } } /** @Entity */ class DDC6499B { - /** - * @Id @Column(type="integer") @GeneratedValue - */ + /** @Id @Column(type="integer") @GeneratedValue */ public $id; - /** - * @ManyToOne(targetEntity="DDC6499A", inversedBy="bs") - */ - public $a; -} \ No newline at end of file + /** @ManyToOne(targetEntity="DDC6499A") */ + private $a; +} From 166c5816b6296fc14860149fe1a6d2cf3f40e092 Mon Sep 17 00:00:00 2001 From: Marco Pivetta <ocramius@gmail.com> Date: Fri, 11 Aug 2017 22:10:02 +0200 Subject: [PATCH 7/8] #6499 #6533 calibrating test so that the association is populated after persistence in some edge cases --- .../Tests/ORM/Functional/Ticket/DDC6499Test.php | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php index 6e8b25336..0d86fffb0 100644 --- a/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php +++ b/tests/Doctrine/Tests/ORM/Functional/Ticket/DDC6499Test.php @@ -44,9 +44,12 @@ class DDC6499Test extends OrmFunctionalTestCase public function testIssue() : void { $b = new DDC6499B(); - $a = new DDC6499A($b); + $a = new DDC6499A(); $this->_em->persist($a); + + $a->b = $b; + $this->_em->persist($b); $this->_em->flush(); @@ -58,7 +61,9 @@ class DDC6499Test extends OrmFunctionalTestCase public function testIssueReversed() : void { $b = new DDC6499B(); - $a = new DDC6499A($b); + $a = new DDC6499A(); + + $a->b = $b; $this->_em->persist($b); $this->_em->persist($a); @@ -78,11 +83,6 @@ class DDC6499A /** @JoinColumn(nullable=false) @OneToOne(targetEntity=DDC6499B::class) */ public $b; - - public function __construct(DDC6499B $b) - { - $this->b = $b; - } } /** @Entity */ From 1ede3c514f78837ffa5f4bb0621a188791d80cd0 Mon Sep 17 00:00:00 2001 From: Marco Pivetta <ocramius@gmail.com> Date: Fri, 11 Aug 2017 22:13:20 +0200 Subject: [PATCH 8/8] #6499 #6533 simplifying nullable column check - null coalesce operator makes it *SOMEWHAT* more readable (no miracles) --- lib/Doctrine/ORM/UnitOfWork.php | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/lib/Doctrine/ORM/UnitOfWork.php b/lib/Doctrine/ORM/UnitOfWork.php index 0afabf805..56615807f 100644 --- a/lib/Doctrine/ORM/UnitOfWork.php +++ b/lib/Doctrine/ORM/UnitOfWork.php @@ -1154,7 +1154,11 @@ class UnitOfWork implements PropertyChangedListener $joinColumns = reset($assoc['joinColumns']); - $calc->addDependency($targetClass->name, $class->name, (int) ! ( ! isset($joinColumns['nullable']) || $joinColumns['nullable'] === true)); + $calc->addDependency( + $targetClass->name, + $class->name, + (int) (($joinColumns['nullable'] ?? true) === false) + ); // If the target class has mapped subclasses, these share the same dependency. if ( ! $targetClass->subClasses) {