1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186
|
<?php
declare(strict_types=1);
namespace Doctrine\Tests\ORM\Functional\Ticket;
use Doctrine\ORM\Mapping\Column;
use Doctrine\ORM\Mapping\DiscriminatorColumn;
use Doctrine\ORM\Mapping\DiscriminatorMap;
use Doctrine\ORM\Mapping\Entity;
use Doctrine\ORM\Mapping\GeneratedValue;
use Doctrine\ORM\Mapping\Id;
use Doctrine\ORM\Mapping\InheritanceType;
use Doctrine\ORM\Mapping\JoinColumn;
use Doctrine\ORM\Mapping\ManyToOne;
use Doctrine\ORM\Mapping\OneToOne;
use Doctrine\Tests\OrmFunctionalTestCase;
use PHPUnit\Framework\Attributes\Group;
use function assert;
#[Group('DDC-1163')]
class DDC1163Test extends OrmFunctionalTestCase
{
private int|null $productId = null;
private int|null $proxyHolderId = null;
protected function setUp(): void
{
parent::setUp();
$this->createSchemaForModels(
DDC1163Product::class,
DDC1163SpecialProduct::class,
DDC1163ProxyHolder::class,
DDC1163Tag::class,
);
}
public function testIssue(): void
{
$this->createSpecialProductAndProxyHolderReferencingIt();
$this->_em->clear();
$this->createProxyForSpecialProduct();
$this->setPropertyAndAssignTagToSpecialProduct();
// fails
$this->_em->flush();
}
private function createSpecialProductAndProxyHolderReferencingIt(): void
{
$specialProduct = new DDC1163SpecialProduct();
$this->_em->persist($specialProduct);
$proxyHolder = new DDC1163ProxyHolder();
$this->_em->persist($proxyHolder);
$proxyHolder->setSpecialProduct($specialProduct);
$this->_em->flush();
$this->productId = $specialProduct->getId();
$this->proxyHolderId = $proxyHolder->getId();
}
/**
* We want Doctrine to instantiate a lazy-load proxy for the previously created
* 'SpecialProduct' and register it.
*
* When Doctrine loads the 'ProxyHolder', it will do just that because the 'ProxyHolder'
* references the 'SpecialProduct'.
*/
private function createProxyForSpecialProduct(): void
{
$proxyHolder = $this->_em->find(DDC1163ProxyHolder::class, $this->proxyHolderId);
assert($proxyHolder instanceof DDC1163ProxyHolder);
self::assertInstanceOf(DDC1163SpecialProduct::class, $proxyHolder->getSpecialProduct());
}
private function setPropertyAndAssignTagToSpecialProduct(): void
{
$specialProduct = $this->_em->find(DDC1163SpecialProduct::class, $this->productId);
assert($specialProduct instanceof DDC1163SpecialProduct);
self::assertInstanceOf(DDC1163SpecialProduct::class, $specialProduct);
self::assertTrue($this->isUninitializedObject($specialProduct));
$specialProduct->setSubclassProperty('foobar');
// this screams violation of law of demeter ;)
self::assertEquals(
DDC1163SpecialProduct::class,
$this->_em->getUnitOfWork()->getEntityPersister($specialProduct::class)->getClassMetadata()->name,
);
$tag = new DDC1163Tag('Foo');
$this->_em->persist($tag);
$tag->setProduct($specialProduct);
}
}
#[Entity]
class DDC1163ProxyHolder
{
#[Column(name: 'id', type: 'integer')]
#[Id]
#[GeneratedValue(strategy: 'AUTO')]
private int $id;
#[OneToOne(targetEntity: 'DDC1163SpecialProduct')]
private DDC1163SpecialProduct|null $specialProduct = null;
public function getId(): int
{
return $this->id;
}
public function setSpecialProduct(DDC1163SpecialProduct $specialProduct): void
{
$this->specialProduct = $specialProduct;
}
public function getSpecialProduct(): DDC1163SpecialProduct
{
return $this->specialProduct;
}
}
#[Entity]
#[InheritanceType('JOINED')]
#[DiscriminatorColumn(name: 'type', type: 'string')]
#[DiscriminatorMap(['special' => 'DDC1163SpecialProduct'])]
abstract class DDC1163Product
{
/** @var int */
#[Column(name: 'id', type: 'integer')]
#[Id]
#[GeneratedValue(strategy: 'AUTO')]
protected $id;
public function getId(): int
{
return $this->id;
}
}
#[Entity]
class DDC1163SpecialProduct extends DDC1163Product
{
#[Column(name: 'subclass_property', type: 'string', nullable: true)]
private string|null $subclassProperty = null;
public function setSubclassProperty(string $value): void
{
$this->subclassProperty = $value;
}
}
#[Entity]
class DDC1163Tag
{
#[Column(name: 'id', type: 'integer')]
#[Id]
#[GeneratedValue(strategy: 'AUTO')]
private int $id;
/** @var Product */
#[JoinColumn(name: 'product_id', referencedColumnName: 'id')]
#[ManyToOne(targetEntity: 'DDC1163Product', inversedBy: 'tags')]
private $product;
public function __construct(
#[Column(name: 'name', type: 'string', length: 255)]
private string $name,
) {
}
public function setProduct(DDC1163Product $product): void
{
$this->product = $product;
}
}
|