Skip to content

Commit eb3b984

Browse files
beberleinicolas-grekasgreg0ire
authored
Add support for PHP 8.4 Lazy Objects RFC with configuration flag (#11853)
* Introduce PHP 8.4 lazy proxy/ghost API. * Call setRawValueWithoutLazyInitialization for support with lazy proxy. * Refactorings * Revert test change partially and skip with lazy objects. * Houskeeping: phpcs * Run with ENABLE_LAZY_PROXY=1 in php 8.4 matrix. * Fix ci * Transient properties are not skipping lazy initialization anymore, to expensive and could lead to errors. Adjust lifecycle test that uses transient properittes for assertions. * Restore behavior preventing property hook use in 8.4 in unsupported coditions * Add \ReflectionClass::SKIP_INITIALIZATION_ON_SERIALIZE Co-authored-by: Nicolas Grekas <[email protected]> * Rename isNativeLazyObjectsEnabled/enableNativeLazyObjects. * Housekeeping: phpcs * Update advanced-configuration docs and make proxy config variables not required anymore with native lazy objects. * Move code around * Apply suggestions from code review Co-authored-by: Grégoire Paris <[email protected]> * Pick suggestions --------- Co-authored-by: Nicolas Grekas <[email protected]> Co-authored-by: Grégoire Paris <[email protected]>
1 parent 04395f9 commit eb3b984

17 files changed

+167
-59
lines changed

.github/workflows/continuous-integration.yml

+13-1
Original file line numberDiff line numberDiff line change
@@ -43,17 +43,27 @@ jobs:
4343
- "pdo_sqlite"
4444
deps:
4545
- "highest"
46+
lazy_proxy:
47+
- "0"
4648
include:
4749
- php-version: "8.2"
4850
dbal-version: "4@dev"
4951
extension: "pdo_sqlite"
52+
lazy_proxy: "0"
5053
- php-version: "8.2"
5154
dbal-version: "4@dev"
5255
extension: "sqlite3"
56+
lazy_proxy: "0"
5357
- php-version: "8.1"
5458
dbal-version: "default"
5559
deps: "lowest"
5660
extension: "pdo_sqlite"
61+
lazy_proxy: "0"
62+
- php-version: "8.4"
63+
dbal-version: "default"
64+
deps: "highest"
65+
extension: "pdo_sqlite"
66+
lazy_proxy: "1"
5767

5868
steps:
5969
- name: "Checkout"
@@ -83,16 +93,18 @@ jobs:
8393
run: "vendor/bin/phpunit -c ci/github/phpunit/${{ matrix.extension }}.xml --coverage-clover=coverage-no-cache.xml"
8494
env:
8595
ENABLE_SECOND_LEVEL_CACHE: 0
96+
ENABLE_LAZY_PROXY: ${{ matrix.lazy_proxy }}
8697

8798
- name: "Run PHPUnit with Second Level Cache"
8899
run: "vendor/bin/phpunit -c ci/github/phpunit/${{ matrix.extension }}.xml --exclude-group performance,non-cacheable,locking_functional --coverage-clover=coverage-cache.xml"
89100
env:
90101
ENABLE_SECOND_LEVEL_CACHE: 1
102+
ENABLE_LAZY_PROXY: ${{ matrix.lazy_proxy }}
91103

92104
- name: "Upload coverage file"
93105
uses: "actions/upload-artifact@v4"
94106
with:
95-
name: "phpunit-${{ matrix.extension }}-${{ matrix.php-version }}-${{ matrix.dbal-version }}-${{ matrix.deps }}-coverage"
107+
name: "phpunit-${{ matrix.extension }}-${{ matrix.php-version }}-${{ matrix.dbal-version }}-${{ matrix.deps }}-${{ matrix.lazy_proxy }}-coverage"
96108
path: "coverage*.xml"
97109

98110

docs/en/reference/advanced-configuration.rst

+38-10
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ steps of configuration.
1919
2020
// ...
2121
22-
if ($applicationMode == "development") {
22+
if ($applicationMode === "development") {
2323
$queryCache = new ArrayAdapter();
2424
$metadataCache = new ArrayAdapter();
2525
} else {
@@ -32,13 +32,18 @@ steps of configuration.
3232
$driverImpl = new AttributeDriver(['/path/to/lib/MyProject/Entities'], true);
3333
$config->setMetadataDriverImpl($driverImpl);
3434
$config->setQueryCache($queryCache);
35-
$config->setProxyDir('/path/to/myproject/lib/MyProject/Proxies');
36-
$config->setProxyNamespace('MyProject\Proxies');
3735
38-
if ($applicationMode == "development") {
39-
$config->setAutoGenerateProxyClasses(true);
36+
if (PHP_VERSION_ID > 80400) {
37+
$config->enableNativeLazyObjects(true);
4038
} else {
41-
$config->setAutoGenerateProxyClasses(false);
39+
$config->setProxyDir('/path/to/myproject/lib/MyProject/Proxies');
40+
$config->setProxyNamespace('MyProject\Proxies');
41+
42+
if ($applicationMode === "development") {
43+
$config->setAutoGenerateProxyClasses(true);
44+
} else {
45+
$config->setAutoGenerateProxyClasses(false);
46+
}
4247
}
4348
4449
$connection = DriverManager::getConnection([
@@ -71,8 +76,25 @@ Configuration Options
7176
The following sections describe all the configuration options
7277
available on a ``Doctrine\ORM\Configuration`` instance.
7378

74-
Proxy Directory (**REQUIRED**)
75-
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
79+
Native Lazy Objects (**OPTIONAL**)
80+
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
81+
82+
With PHP 8.4 we recommend that you use native lazy objects instead of
83+
the code generation approach using the ``symfony/var-exporter`` Ghost trait.
84+
85+
With Doctrine 4, the minimal requirement will become PHP 8.4 and native lazy objects
86+
will become the only approach to lazy loading.
87+
88+
.. code-block:: php
89+
90+
<?php
91+
$config->enableNativeLazyObjects(true);
92+
93+
Proxy Directory
94+
~~~~~~~~~~~~~~~
95+
96+
Required except if you use native lazy objects with PHP 8.4.
97+
This setting will be removed in the future.
7698

7799
.. code-block:: php
78100
@@ -85,8 +107,11 @@ classes. For a detailed explanation on proxy classes and how they
85107
are used in Doctrine, refer to the "Proxy Objects" section further
86108
down.
87109

88-
Proxy Namespace (**REQUIRED**)
89-
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
110+
Proxy Namespace
111+
~~~~~~~~~~~~~~~
112+
113+
Required except if you use native lazy objects with PHP 8.4.
114+
This setting will be removed in the future.
90115

91116
.. code-block:: php
92117
@@ -200,6 +225,9 @@ deprecated ``Doctrine\DBAL\Logging\SQLLogger`` interface.
200225
Auto-generating Proxy Classes (**OPTIONAL**)
201226
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
202227

228+
This setting is not required if you use native lazy objects with PHP 8.4
229+
and will be removed in the future.
230+
203231
Proxy classes can either be generated manually through the Doctrine
204232
Console or automatically at runtime by Doctrine. The configuration
205233
option that controls this behavior is:

phpstan-baseline.neon

+1-7
Original file line numberDiff line numberDiff line change
@@ -583,7 +583,7 @@ parameters:
583583
path: src/EntityManager.php
584584

585585
-
586-
message: '#^Method Doctrine\\ORM\\EntityManager\:\:getReference\(\) should return \(T of object\)\|null but returns Doctrine\\ORM\\Proxy\\InternalProxy\.$#'
586+
message: '#^Method Doctrine\\ORM\\EntityManager\:\:getReference\(\) should return \(T of object\)\|null but returns object\.$#'
587587
identifier: return.type
588588
count: 1
589589
path: src/EntityManager.php
@@ -2322,12 +2322,6 @@ parameters:
23222322
count: 1
23232323
path: src/Proxy/ProxyFactory.php
23242324

2325-
-
2326-
message: '#^Method Doctrine\\ORM\\Proxy\\ProxyFactory\:\:getProxy\(\) return type with generic interface Doctrine\\ORM\\Proxy\\InternalProxy does not specify its types\: T$#'
2327-
identifier: missingType.generics
2328-
count: 1
2329-
path: src/Proxy/ProxyFactory.php
2330-
23312325
-
23322326
message: '#^Method Doctrine\\ORM\\Proxy\\ProxyFactory\:\:loadProxyClass\(\) has parameter \$class with generic interface Doctrine\\Persistence\\Mapping\\ClassMetadata but does not specify its types\: T$#'
23332327
identifier: missingType.generics

src/Configuration.php

+16
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,8 @@
3030
use function is_a;
3131
use function strtolower;
3232

33+
use const PHP_VERSION_ID;
34+
3335
/**
3436
* Configuration container for all configuration options of Doctrine.
3537
* It combines all configuration options from DBAL & ORM.
@@ -593,6 +595,20 @@ public function setSchemaIgnoreClasses(array $schemaIgnoreClasses): void
593595
$this->attributes['schemaIgnoreClasses'] = $schemaIgnoreClasses;
594596
}
595597

598+
public function isNativeLazyObjectsEnabled(): bool
599+
{
600+
return $this->attributes['nativeLazyObjects'] ?? false;
601+
}
602+
603+
public function enableNativeLazyObjects(bool $nativeLazyObjects): void
604+
{
605+
if (PHP_VERSION_ID < 80400) {
606+
throw new LogicException('Lazy loading proxies require PHP 8.4 or higher.');
607+
}
608+
609+
$this->attributes['nativeLazyObjects'] = $nativeLazyObjects;
610+
}
611+
596612
/**
597613
* To be deprecated in 3.1.0
598614
*

src/Mapping/ClassMetadataFactory.php

+11
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424
use Doctrine\Persistence\Mapping\ClassMetadata as ClassMetadataInterface;
2525
use Doctrine\Persistence\Mapping\Driver\MappingDriver;
2626
use Doctrine\Persistence\Mapping\ReflectionService;
27+
use LogicException;
2728
use ReflectionClass;
2829
use ReflectionException;
2930

@@ -41,6 +42,8 @@
4142
use function strtolower;
4243
use function substr;
4344

45+
use const PHP_VERSION_ID;
46+
4447
/**
4548
* The ClassMetadataFactory is used to create ClassMetadata objects that contain all the
4649
* metadata mapping information of a class which describes how a class should be mapped
@@ -296,6 +299,14 @@ protected function validateRuntimeMetadata(ClassMetadata $class, ClassMetadataIn
296299
// second condition is necessary for mapped superclasses in the middle of an inheritance hierarchy
297300
throw MappingException::noInheritanceOnMappedSuperClass($class->name);
298301
}
302+
303+
foreach ($class->propertyAccessors as $propertyAccessor) {
304+
$property = $propertyAccessor->getUnderlyingReflector();
305+
306+
if (PHP_VERSION_ID >= 80400 && count($property->getHooks()) > 0) {
307+
throw new LogicException('Doctrine ORM does not support property hooks without also enabling Configuration::enableNativeLazyObjects(true). Check https://github.com/doctrine/orm/issues/11624 for details of versions that support property hooks.');
308+
}
309+
}
299310
}
300311

301312
protected function newClassMetadataInstance(string $className): ClassMetadata

src/Mapping/PropertyAccessors/RawValuePropertyAccessor.php

+7-1
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,13 @@
55
namespace Doctrine\ORM\Mapping\PropertyAccessors;
66

77
use Doctrine\ORM\Proxy\InternalProxy;
8+
use LogicException;
89
use ReflectionProperty;
910

1011
use function ltrim;
1112

13+
use const PHP_VERSION_ID;
14+
1215
/**
1316
* This is a PHP 8.4 and up only class and replaces ObjectCastPropertyAccessor.
1417
*
@@ -28,12 +31,15 @@ public static function fromReflectionProperty(ReflectionProperty $reflectionProp
2831

2932
private function __construct(private ReflectionProperty $reflectionProperty, private string $key)
3033
{
34+
if (PHP_VERSION_ID < 80400) {
35+
throw new LogicException('This class requires PHP 8.4 or higher.');
36+
}
3137
}
3238

3339
public function setValue(object $object, mixed $value): void
3440
{
3541
if (! ($object instanceof InternalProxy && ! $object->__isInitialized())) {
36-
$this->reflectionProperty->setRawValue($object, $value);
42+
$this->reflectionProperty->setRawValueWithoutLazyInitialization($object, $value);
3743

3844
return;
3945
}

src/Proxy/ProxyFactory.php

+23-3
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313
use Doctrine\ORM\Utility\IdentifierFlattener;
1414
use Doctrine\Persistence\Mapping\ClassMetadata;
1515
use Doctrine\Persistence\Proxy;
16+
use ReflectionClass;
1617
use ReflectionProperty;
1718
use Symfony\Component\VarExporter\ProxyHelper;
1819

@@ -142,11 +143,11 @@ public function __construct(
142143
private readonly string $proxyNs,
143144
bool|int $autoGenerate = self::AUTOGENERATE_NEVER,
144145
) {
145-
if (! $proxyDir) {
146+
if (! $proxyDir && ! $em->getConfiguration()->isNativeLazyObjectsEnabled()) {
146147
throw ORMInvalidArgumentException::proxyDirectoryRequired();
147148
}
148149

149-
if (! $proxyNs) {
150+
if (! $proxyNs && ! $em->getConfiguration()->isNativeLazyObjectsEnabled()) {
150151
throw ORMInvalidArgumentException::proxyNamespaceRequired();
151152
}
152153

@@ -163,8 +164,23 @@ public function __construct(
163164
* @param class-string $className
164165
* @param array<mixed> $identifier
165166
*/
166-
public function getProxy(string $className, array $identifier): InternalProxy
167+
public function getProxy(string $className, array $identifier): object
167168
{
169+
if ($this->em->getConfiguration()->isNativeLazyObjectsEnabled()) {
170+
$classMetadata = $this->em->getClassMetadata($className);
171+
$entityPersister = $this->uow->getEntityPersister($className);
172+
173+
$proxy = $classMetadata->reflClass->newLazyGhost(static function (object $object) use ($identifier, $entityPersister): void {
174+
$entityPersister->loadById($identifier, $object);
175+
}, ReflectionClass::SKIP_INITIALIZATION_ON_SERIALIZE);
176+
177+
foreach ($identifier as $idField => $value) {
178+
$classMetadata->propertyAccessors[$idField]->setValue($proxy, $value);
179+
}
180+
181+
return $proxy;
182+
}
183+
168184
$proxyFactory = $this->proxyFactories[$className] ?? $this->getProxyFactory($className);
169185

170186
return $proxyFactory($identifier);
@@ -182,6 +198,10 @@ public function getProxy(string $className, array $identifier): InternalProxy
182198
*/
183199
public function generateProxyClasses(array $classes, string|null $proxyDir = null): int
184200
{
201+
if ($this->em->getConfiguration()->isNativeLazyObjectsEnabled()) {
202+
return 0;
203+
}
204+
185205
$generated = 0;
186206

187207
foreach ($classes as $class) {

src/UnitOfWork.php

+16-1
Original file line numberDiff line numberDiff line change
@@ -2378,7 +2378,11 @@ public function createEntity(string $className, array $data, array &$hints = [])
23782378
}
23792379

23802380
if ($this->isUninitializedObject($entity)) {
2381-
$entity->__setInitialized(true);
2381+
if ($this->em->getConfiguration()->isNativeLazyObjectsEnabled()) {
2382+
$class->reflClass->markLazyObjectAsInitialized($entity);
2383+
} else {
2384+
$entity->__setInitialized(true);
2385+
}
23822386
} else {
23832387
if (
23842388
! isset($hints[Query::HINT_REFRESH])
@@ -3033,6 +3037,13 @@ public function initializeObject(object $obj): void
30333037

30343038
if ($obj instanceof PersistentCollection) {
30353039
$obj->initialize();
3040+
3041+
return;
3042+
}
3043+
3044+
if ($this->em->getConfiguration()->isNativeLazyObjectsEnabled()) {
3045+
$reflection = $this->em->getClassMetadata($obj::class)->getReflectionClass();
3046+
$reflection->initializeLazyObject($obj);
30363047
}
30373048
}
30383049

@@ -3043,6 +3054,10 @@ public function initializeObject(object $obj): void
30433054
*/
30443055
public function isUninitializedObject(mixed $obj): bool
30453056
{
3057+
if ($this->em->getConfiguration()->isNativeLazyObjectsEnabled() && ! ($obj instanceof Collection)) {
3058+
return $this->em->getClassMetadata($obj::class)->reflClass->isUninitializedLazyObject($obj);
3059+
}
3060+
30463061
return $obj instanceof InternalProxy && ! $obj->__isInitialized();
30473062
}
30483063

tests/Tests/ORM/Functional/BasicFunctionalTest.php

+3-5
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@
99
use Doctrine\ORM\Mapping\ClassMetadata;
1010
use Doctrine\ORM\ORMInvalidArgumentException;
1111
use Doctrine\ORM\PersistentCollection;
12-
use Doctrine\ORM\Proxy\InternalProxy;
1312
use Doctrine\ORM\Query;
1413
use Doctrine\ORM\UnitOfWork;
1514
use Doctrine\Tests\IterableTester;
@@ -557,7 +556,7 @@ public function testSetToOneAssociationWithGetReference(): void
557556
$this->_em->persist($article);
558557
$this->_em->flush();
559558

560-
self::assertFalse($userRef->__isInitialized());
559+
self::assertTrue($this->isUninitializedObject($userRef));
561560

562561
$this->_em->clear();
563562

@@ -592,7 +591,7 @@ public function testAddToToManyAssociationWithGetReference(): void
592591
$this->_em->persist($user);
593592
$this->_em->flush();
594593

595-
self::assertFalse($groupRef->__isInitialized());
594+
self::assertTrue($this->isUninitializedObject($groupRef));
596595

597596
$this->_em->clear();
598597

@@ -940,8 +939,7 @@ public function testManyToOneFetchModeQuery(): void
940939
->setParameter(1, $article->id)
941940
->setFetchMode(CmsArticle::class, 'user', ClassMetadata::FETCH_EAGER)
942941
->getSingleResult();
943-
self::assertInstanceOf(InternalProxy::class, $article->user, 'It IS a proxy, ...');
944-
self::assertFalse($this->isUninitializedObject($article->user), '...but its initialized!');
942+
self::assertFalse($this->isUninitializedObject($article->user));
945943
$this->assertQueryCount(2);
946944
}
947945

0 commit comments

Comments
 (0)