mirror of
https://github.com/doctrine/orm.git
synced 2026-03-24 15:02:22 +01:00
Compare commits
104 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
17500f56ea | ||
|
|
fc2f724e2d | ||
|
|
2f9e98754b | ||
|
|
bb5524099c | ||
|
|
3a8cafe228 | ||
|
|
8259a16681 | ||
|
|
5577d51c44 | ||
|
|
d1922a3065 | ||
|
|
597a63a86c | ||
|
|
6b220e3c90 | ||
|
|
6de4b68705 | ||
|
|
16c0151831 | ||
|
|
440b244ebc | ||
|
|
a616914887 | ||
|
|
fd0bdc69b0 | ||
|
|
f50803ccb9 | ||
|
|
eeefc6bc0f | ||
|
|
710dde83aa | ||
|
|
495cd06b9a | ||
|
|
fd7a14ad22 | ||
|
|
0d3ce5d4f8 | ||
|
|
f01d107edc | ||
|
|
3cc30c4024 | ||
|
|
d2de4ec03c | ||
|
|
64ee76e94e | ||
|
|
8e20e1598e | ||
|
|
24df74d61d | ||
|
|
442f073d25 | ||
|
|
a5161e9485 | ||
|
|
ddc7d953b9 | ||
|
|
db51ed4f4c | ||
|
|
6d27797b2e | ||
|
|
89250b8ca2 | ||
|
|
bc61d7d21e | ||
|
|
dca7ddf969 | ||
|
|
e781639812 | ||
|
|
b5987ad29a | ||
|
|
8c513a6523 | ||
|
|
3b3056f910 | ||
|
|
fa5c37e972 | ||
|
|
f3e36debfe | ||
|
|
ca7abd04a2 | ||
|
|
a555626150 | ||
|
|
f26946b477 | ||
|
|
efc83bce8e | ||
|
|
81ddeb426c | ||
|
|
42e63bf358 | ||
|
|
bb21865cba | ||
|
|
606da9280d | ||
|
|
21708e43c4 | ||
|
|
8eb69922e6 | ||
|
|
e9b6fd89a4 | ||
|
|
01a14327d2 | ||
|
|
2df1071e7a | ||
|
|
7fc359c2bb | ||
|
|
853b9f75ae | ||
|
|
44d2a83e09 | ||
|
|
5f079c2061 | ||
|
|
7ef4afc688 | ||
|
|
70bcff7410 | ||
|
|
f778d8cf98 | ||
|
|
18b32ab9db | ||
|
|
aa3ff458c7 | ||
|
|
8bc74c624a | ||
|
|
6c0a5ecbf9 | ||
|
|
5f6501f842 | ||
|
|
41f704cd96 | ||
|
|
5c74795893 | ||
|
|
8961bfe90c | ||
|
|
15e3a7e861 | ||
|
|
1280e005b6 | ||
|
|
79f53d5dae | ||
|
|
bf2937e63a | ||
|
|
ed212ab924 | ||
|
|
dd0e02e912 | ||
|
|
aad875eea1 | ||
|
|
84ab535e56 | ||
|
|
a72a0c3597 | ||
|
|
6217285544 | ||
|
|
330c0bc67e | ||
|
|
b5595ca041 | ||
|
|
b779b112f3 | ||
|
|
04e08640fb | ||
|
|
aa27b3a35f | ||
|
|
d76fc4ebf6 | ||
|
|
ae60cf005f | ||
|
|
ddd3066bc4 | ||
|
|
6662195936 | ||
|
|
d951aa05b9 | ||
|
|
be297b9fd3 | ||
|
|
4e137f77a5 | ||
|
|
a33aa15c79 | ||
|
|
255ce51526 | ||
|
|
38e47fdeab | ||
|
|
9ac063d879 | ||
|
|
b52a8f8b9e | ||
|
|
1b2771f964 | ||
|
|
9766b6b03e | ||
|
|
37c8953015 | ||
|
|
a199ca3002 | ||
|
|
ed34327941 | ||
|
|
b42cf99402 | ||
|
|
33a19f1bf3 | ||
|
|
b6669746b7 |
@@ -12,21 +12,27 @@
|
||||
"upcoming": true
|
||||
},
|
||||
{
|
||||
"name": "2.16",
|
||||
"branchName": "2.16.x",
|
||||
"slug": "2.16",
|
||||
"name": "2.17",
|
||||
"branchName": "2.17.x",
|
||||
"slug": "2.17",
|
||||
"upcoming": true
|
||||
},
|
||||
{
|
||||
"name": "2.15",
|
||||
"branchName": "2.15.x",
|
||||
"slug": "2.15",
|
||||
"name": "2.16",
|
||||
"branchName": "2.16.x",
|
||||
"slug": "2.16",
|
||||
"current": true,
|
||||
"aliases": [
|
||||
"current",
|
||||
"stable"
|
||||
]
|
||||
},
|
||||
{
|
||||
"name": "2.15",
|
||||
"branchName": "2.15.x",
|
||||
"slug": "2.15",
|
||||
"maintained": false
|
||||
},
|
||||
{
|
||||
"name": "2.14",
|
||||
"branchName": "2.14.x",
|
||||
|
||||
54
UPGRADE.md
54
UPGRADE.md
@@ -1,3 +1,57 @@
|
||||
# Upgrade to 2.16
|
||||
|
||||
## Deprecated accepting duplicate IDs in the identity map
|
||||
|
||||
For any given entity class and ID value, there should be only one object instance
|
||||
representing the entity.
|
||||
|
||||
In https://github.com/doctrine/orm/pull/10785, a check was added that will guard this
|
||||
in the identity map. The most probable cause for violations of this rule are collisions
|
||||
of application-provided IDs.
|
||||
|
||||
In ORM 2.16.0, the check was added by throwing an exception. In ORM 2.16.1, this will be
|
||||
changed to a deprecation notice. ORM 3.0 will make it an exception again. Use
|
||||
`\Doctrine\ORM\Configuration::setRejectIdCollisionInIdentityMap()` if you want to opt-in
|
||||
to the new mode.
|
||||
|
||||
## Potential changes to the order in which `INSERT`s are executed
|
||||
|
||||
In https://github.com/doctrine/orm/pull/10547, the commit order computation was improved
|
||||
to fix a series of bugs where a correct (working) commit order was previously not found.
|
||||
Also, the new computation may get away with fewer queries being executed: By inserting
|
||||
referred-to entities first and using their ID values for foreign key fields in subsequent
|
||||
`INSERT` statements, additional `UPDATE` statements that were previously necessary can be
|
||||
avoided.
|
||||
|
||||
When using database-provided, auto-incrementing IDs, this may lead to IDs being assigned
|
||||
to entities in a different order than it was previously the case.
|
||||
|
||||
## Deprecated `\Doctrine\ORM\Internal\CommitOrderCalculator` and related classes
|
||||
|
||||
With changes made to the commit order computation, the internal classes
|
||||
`\Doctrine\ORM\Internal\CommitOrderCalculator`, `\Doctrine\ORM\Internal\CommitOrder\Edge`,
|
||||
`\Doctrine\ORM\Internal\CommitOrder\Vertex` and `\Doctrine\ORM\Internal\CommitOrder\VertexState`
|
||||
have been deprecated and will be removed in ORM 3.0.
|
||||
|
||||
## Deprecated returning post insert IDs from `EntityPersister::executeInserts()`
|
||||
|
||||
Persisters implementing `\Doctrine\ORM\Persisters\Entity\EntityPersister` should no longer
|
||||
return an array of post insert IDs from their `::executeInserts()` method. Make the
|
||||
persister call `Doctrine\ORM\UnitOfWork::assignPostInsertId()` instead.
|
||||
|
||||
## Changing the way how reflection-based mapping drivers report fields, deprecated the "old" mode
|
||||
|
||||
In ORM 3.0, a change will be made regarding how the `AttributeDriver` reports field mappings.
|
||||
This change is necessary to be able to detect (and reject) some invalid mapping configurations.
|
||||
|
||||
To avoid surprises during 2.x upgrades, the new mode is opt-in. It can be activated on the
|
||||
`AttributeDriver` and `AnnotationDriver` by setting the `$reportFieldsWhereDeclared`
|
||||
constructor parameter to `true`. It will cause `MappingException`s to be thrown when invalid
|
||||
configurations are detected.
|
||||
|
||||
Not enabling the new mode will cause a deprecation notice to be raised. In ORM 3.0,
|
||||
only the new mode will be available.
|
||||
|
||||
# Upgrade to 2.15
|
||||
|
||||
## Deprecated configuring `JoinColumn` on the inverse side of one-to-one associations
|
||||
|
||||
@@ -42,14 +42,14 @@
|
||||
"doctrine/annotations": "^1.13 || ^2",
|
||||
"doctrine/coding-standard": "^9.0.2 || ^12.0",
|
||||
"phpbench/phpbench": "^0.16.10 || ^1.0",
|
||||
"phpstan/phpstan": "~1.4.10 || 1.10.25",
|
||||
"phpstan/phpstan": "~1.4.10 || 1.10.28",
|
||||
"phpunit/phpunit": "^7.5 || ^8.5 || ^9.6",
|
||||
"psr/log": "^1 || ^2 || ^3",
|
||||
"squizlabs/php_codesniffer": "3.7.2",
|
||||
"symfony/cache": "^4.4 || ^5.4 || ^6.0",
|
||||
"symfony/var-exporter": "^4.4 || ^5.4 || ^6.2",
|
||||
"symfony/yaml": "^3.4 || ^4.0 || ^5.0 || ^6.0",
|
||||
"vimeo/psalm": "4.30.0 || 5.13.1"
|
||||
"vimeo/psalm": "4.30.0 || 5.14.1"
|
||||
},
|
||||
"conflict": {
|
||||
"doctrine/annotations": "<1.13 || >= 3.0"
|
||||
|
||||
@@ -13,7 +13,7 @@ for all our domain objects.
|
||||
.. note::
|
||||
|
||||
The notify change tracking policy is deprecated and will be removed in ORM 3.0.
|
||||
(`Details <https://github.com/doctrine/orm/issues/8383>`_)
|
||||
(\ `Details <https://github.com/doctrine/orm/issues/8383>`_)
|
||||
|
||||
Implementing NotifyPropertyChanged
|
||||
----------------------------------
|
||||
|
||||
@@ -58,6 +58,10 @@ First Attributes:
|
||||
.. code-block:: php
|
||||
|
||||
<?php
|
||||
use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\HasLifecycleCallbacks;
|
||||
use Doctrine\ORM\Mapping\PrePersist;
|
||||
use Doctrine\ORM\Mapping\PreUpdate;
|
||||
|
||||
#[Entity]
|
||||
#[HasLifecycleCallbacks]
|
||||
|
||||
@@ -29,7 +29,7 @@ steps of configuration.
|
||||
|
||||
$config = new Configuration;
|
||||
$config->setMetadataCache($metadataCache);
|
||||
$driverImpl = new AttributeDriver(['/path/to/lib/MyProject/Entities']);
|
||||
$driverImpl = new AttributeDriver(['/path/to/lib/MyProject/Entities'], true);
|
||||
$config->setMetadataDriverImpl($driverImpl);
|
||||
$config->setQueryCache($queryCache);
|
||||
$config->setProxyDir('/path/to/myproject/lib/MyProject/Proxies');
|
||||
@@ -134,7 +134,7 @@ The attribute driver can be injected in the ``Doctrine\ORM\Configuration``:
|
||||
<?php
|
||||
use Doctrine\ORM\Mapping\Driver\AttributeDriver;
|
||||
|
||||
$driverImpl = new AttributeDriver(['/path/to/lib/MyProject/Entities']);
|
||||
$driverImpl = new AttributeDriver(['/path/to/lib/MyProject/Entities'], true);
|
||||
$config->setMetadataDriverImpl($driverImpl);
|
||||
|
||||
The path information to the entities is required for the attribute
|
||||
|
||||
@@ -47,8 +47,7 @@ mapping metadata:
|
||||
- :doc:`Attributes <attributes-reference>`
|
||||
- :doc:`XML <xml-mapping>`
|
||||
- :doc:`PHP code <php-mapping>`
|
||||
- :doc:`Docblock Annotations <annotations-reference>` (deprecated and
|
||||
will be removed in ``doctrine/orm`` 3.0)
|
||||
- :doc:`Docblock Annotations <annotations-reference>` (deprecated and will be removed in ``doctrine/orm`` 3.0)
|
||||
- :doc:`YAML <yaml-mapping>` (deprecated and will be removed in ``doctrine/orm`` 3.0.)
|
||||
|
||||
This manual will usually show mapping metadata via attributes, though
|
||||
|
||||
@@ -63,7 +63,7 @@ Notify
|
||||
.. note::
|
||||
|
||||
The notify change tracking policy is deprecated and will be removed in ORM 3.0.
|
||||
(`Details <https://github.com/doctrine/orm/issues/8383>`_)
|
||||
(\ `Details <https://github.com/doctrine/orm/issues/8383>`_)
|
||||
|
||||
This policy is based on the assumption that the entities notify
|
||||
interested listeners of changes to their properties. For that
|
||||
|
||||
@@ -1425,7 +1425,7 @@ userland:
|
||||
reloading this data. Partially loaded objects have to be passed to
|
||||
``EntityManager::refresh()`` if they are to be reloaded fully from
|
||||
the database. This query hint is deprecated and will be removed
|
||||
in the future (`Details <https://github.com/doctrine/orm/issues/8471>`_)
|
||||
in the future (\ `Details <https://github.com/doctrine/orm/issues/8471>`_)
|
||||
- ``Query::HINT_REFRESH`` - This query is used internally by
|
||||
``EntityManager::refresh()`` and can be used in userland as well.
|
||||
If you specify this hint and a query returns the data for an entity
|
||||
|
||||
@@ -215,6 +215,10 @@ specific to a particular entity class's lifecycle.
|
||||
<?php
|
||||
use Doctrine\DBAL\Types\Types;
|
||||
use Doctrine\ORM\Event\PrePersistEventArgs;
|
||||
use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\HasLifecycleCallbacks;
|
||||
use Doctrine\ORM\Mapping\PrePersist;
|
||||
use Doctrine\ORM\Mapping\PreUpdate;
|
||||
|
||||
#[Entity]
|
||||
#[HasLifecycleCallbacks]
|
||||
@@ -705,13 +709,21 @@ not directly mapped by Doctrine.
|
||||
- The ``postUpdate`` event occurs after the database
|
||||
update operations to entity data. It is not called for a DQL
|
||||
``UPDATE`` statement.
|
||||
- The ``postPersist`` event occurs for an entity after
|
||||
the entity has been made persistent. It will be invoked after the
|
||||
database insert operations. Generated primary key values are
|
||||
available in the postPersist event.
|
||||
- The ``postPersist`` event occurs for an entity after the entity has
|
||||
been made persistent. It will be invoked after all database insert
|
||||
operations for new entities have been performed. Generated primary
|
||||
key values will be available for all entities at the time this
|
||||
event is triggered.
|
||||
- The ``postRemove`` event occurs for an entity after the
|
||||
entity has been deleted. It will be invoked after the database
|
||||
delete operations. It is not called for a DQL ``DELETE`` statement.
|
||||
entity has been deleted. It will be invoked after all database
|
||||
delete operations for entity rows have been executed. This event is
|
||||
not called for a DQL ``DELETE`` statement.
|
||||
|
||||
.. note::
|
||||
|
||||
At the time ``postPersist`` is called, there may still be collection and/or
|
||||
"extra" updates pending. The database may not yet be completely in
|
||||
sync with the entity states in memory, not even for the new entities.
|
||||
|
||||
.. warning::
|
||||
|
||||
@@ -720,6 +732,19 @@ not directly mapped by Doctrine.
|
||||
cascade remove relations. In this case, you should load yourself the proxy in
|
||||
the associated ``pre*`` event.
|
||||
|
||||
.. warning::
|
||||
|
||||
Making changes to entities and calling ``EntityManager::flush()`` from within
|
||||
``post*`` event handlers is strongly discouraged, and might be deprecated and
|
||||
eventually prevented in the future.
|
||||
|
||||
The reason is that it causes re-entrance into ``UnitOfWork::commit()`` while a commit
|
||||
is currently being processed. The ``UnitOfWork`` was never designed to support this,
|
||||
and its behavior in this situation is not covered by any tests.
|
||||
|
||||
This may lead to entity or collection updates being missed, applied only in parts and
|
||||
changes being lost at the end of the commit phase.
|
||||
|
||||
.. _reference-events-post-load:
|
||||
|
||||
postLoad
|
||||
|
||||
@@ -93,3 +93,34 @@ object.
|
||||
want to refresh or reload an object after having modified a filter or the
|
||||
FilterCollection, then you should clear the EntityManager and re-fetch your
|
||||
entities, having the new rules for filtering applied.
|
||||
|
||||
|
||||
Suspending/Restoring Filters
|
||||
----------------------------
|
||||
When a filter is disabled, the instance is fully deleted and all the filter
|
||||
parameters previously set are lost. Then, if you enable it again, a new filter
|
||||
is created without the previous filter parameters. If you want to keep a filter
|
||||
(in order to use it later) but temporary disable it, you'll need to use the
|
||||
``FilterCollection#suspend($name)`` and ``FilterCollection#restore($name)``
|
||||
methods instead.
|
||||
|
||||
.. code-block:: php
|
||||
|
||||
<?php
|
||||
$filter = $em->getFilters()->enable("locale");
|
||||
$filter->setParameter('locale', 'en');
|
||||
|
||||
// Temporary suspend the filter
|
||||
$filter = $em->getFilters()->suspend("locale");
|
||||
|
||||
// Do things
|
||||
|
||||
// Then restore it, the locale parameter will still be set
|
||||
$filter = $em->getFilters()->restore("locale");
|
||||
|
||||
.. warning::
|
||||
If you enable a previously disabled filter, doctrine will create a new
|
||||
one without keeping any of the previously parameter set with
|
||||
``SQLFilter#setParameter()`` or ``SQLFilter#getParameterList()``. If you
|
||||
want to restore the previously disabled filter instead, you must use the
|
||||
``FilterCollection#restore($name)`` method.
|
||||
|
||||
@@ -167,7 +167,7 @@ have produced, this is probably fine.
|
||||
|
||||
However, to mention known limitations, it is currently not possible to use "class"
|
||||
level `annotations <https://github.com/doctrine/orm/pull/1517>`_ or
|
||||
`attributes <https://github.com/doctrine/orm/issues/8868>` on traits, and attempts to
|
||||
`attributes <https://github.com/doctrine/orm/issues/8868>`_ on traits, and attempts to
|
||||
improve parser support for traits as `here <https://github.com/doctrine/annotations/pull/102>`_
|
||||
or `there <https://github.com/doctrine/annotations/pull/63>`_ have been abandoned
|
||||
due to complexity.
|
||||
|
||||
@@ -6,7 +6,7 @@ Partial Objects
|
||||
|
||||
Creating Partial Objects through DQL is deprecated and
|
||||
will be removed in the future, use data transfer object
|
||||
support in DQL instead. (`Details
|
||||
support in DQL instead. (\ `Details
|
||||
<https://github.com/doctrine/orm/issues/8471>`_)
|
||||
|
||||
A partial object is an object whose state is not fully initialized
|
||||
|
||||
@@ -37,8 +37,8 @@ will still end up with the same reference:
|
||||
public function testIdentityMapReference(): void
|
||||
{
|
||||
$objectA = $this->entityManager->getReference('EntityName', 1);
|
||||
// check for proxyinterface
|
||||
$this->assertInstanceOf('Doctrine\Persistence\Proxy', $objectA);
|
||||
// check entity is not initialized
|
||||
$this->assertTrue($this->entityManager->isUninitializedObject($objectA));
|
||||
|
||||
$objectB = $this->entityManager->find('EntityName', 1);
|
||||
|
||||
@@ -137,7 +137,7 @@ optimize the performance of the Flush Operation:
|
||||
.. note::
|
||||
|
||||
Flush only a single entity with ``$entityManager->flush($entity)`` is deprecated and will be removed in ORM 3.0.
|
||||
(`Details <https://github.com/doctrine/orm/issues/8459>`_)
|
||||
(\ `Details <https://github.com/doctrine/orm/issues/8459>`_)
|
||||
|
||||
Query Internals
|
||||
---------------
|
||||
|
||||
@@ -192,6 +192,11 @@ be properly synchronized with the database when
|
||||
database in the most efficient way and a single, short transaction,
|
||||
taking care of maintaining referential integrity.
|
||||
|
||||
.. note::
|
||||
|
||||
Do not make any assumptions in your code about the number of queries
|
||||
it takes to flush changes, about the ordering of ``INSERT``, ``UPDATE``
|
||||
and ``DELETE`` queries or the order in which entities will be processed.
|
||||
|
||||
Example:
|
||||
|
||||
|
||||
@@ -18,7 +18,7 @@ before. There are some prerequisites for the tutorial that have to be
|
||||
installed:
|
||||
|
||||
- PHP (latest stable version)
|
||||
- Composer Package Manager (`Install Composer
|
||||
- Composer Package Manager (\ `Install Composer
|
||||
<https://getcomposer.org/doc/00-intro.md>`_)
|
||||
|
||||
The code of this tutorial is `available on Github <https://github.com/doctrine/doctrine2-orm-tutorial>`_.
|
||||
@@ -321,7 +321,7 @@ data in your storage, and later in your application when the data is loaded agai
|
||||
.. note::
|
||||
|
||||
This method, although very common, is inappropriate for Domain Driven
|
||||
Design (`DDD <https://en.wikipedia.org/wiki/Domain-driven_design>`_)
|
||||
Design (\ `DDD <https://en.wikipedia.org/wiki/Domain-driven_design>`_)
|
||||
where methods should represent real business operations and not simple
|
||||
property change, And business invariants should be maintained both in the
|
||||
application state (entities in this case) and in the database, with no
|
||||
|
||||
@@ -1010,7 +1010,7 @@ abstract class AbstractQuery
|
||||
*
|
||||
* Alias for getSingleResult(HYDRATE_SINGLE_SCALAR).
|
||||
*
|
||||
* @return mixed The scalar result.
|
||||
* @return bool|float|int|string|null The scalar result.
|
||||
*
|
||||
* @throws NoResultException If the query returned no result.
|
||||
* @throws NonUniqueResultException If the query result is not unique.
|
||||
|
||||
@@ -16,7 +16,6 @@ use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\ORM\Query;
|
||||
use Doctrine\ORM\Query\ResultSetMapping;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
|
||||
use function array_map;
|
||||
use function array_shift;
|
||||
@@ -345,7 +344,7 @@ class DefaultQueryCache implements QueryCache
|
||||
$assocIdentifier = $this->uow->getEntityIdentifier($assocValue);
|
||||
$entityKey = new EntityCacheKey($assocMetadata->rootEntityName, $assocIdentifier);
|
||||
|
||||
if (! $assocValue instanceof Proxy && ($key->cacheMode & Cache::MODE_REFRESH) || ! $assocRegion->contains($entityKey)) {
|
||||
if (! $this->uow->isUninitializedObject($assocValue) && ($key->cacheMode & Cache::MODE_REFRESH) || ! $assocRegion->contains($entityKey)) {
|
||||
// Entity put fail
|
||||
if (! $assocPersister->storeEntityCache($assocValue, $entityKey)) {
|
||||
return null;
|
||||
|
||||
@@ -23,6 +23,7 @@ use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\ORM\Persisters\Entity\EntityPersister;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
|
||||
use function array_merge;
|
||||
use function assert;
|
||||
use function serialize;
|
||||
use function sha1;
|
||||
@@ -314,7 +315,13 @@ abstract class AbstractEntityPersister implements CachedEntityPersister
|
||||
*/
|
||||
public function executeInserts()
|
||||
{
|
||||
$this->queuedCache['insert'] = $this->persister->getInserts();
|
||||
// The commit order/foreign key relationships may make it necessary that multiple calls to executeInsert()
|
||||
// are performed, so collect all the new entities.
|
||||
$newInserts = $this->persister->getInserts();
|
||||
|
||||
if ($newInserts) {
|
||||
$this->queuedCache['insert'] = array_merge($this->queuedCache['insert'] ?? [], $newInserts);
|
||||
}
|
||||
|
||||
return $this->persister->executeInserts();
|
||||
}
|
||||
|
||||
@@ -62,8 +62,6 @@ use function trim;
|
||||
* It combines all configuration options from DBAL & ORM.
|
||||
*
|
||||
* Internal note: When adding a new configuration option just write a getter/setter pair.
|
||||
*
|
||||
* @psalm-import-type AutogenerateMode from ProxyFactory
|
||||
*/
|
||||
class Configuration extends \Doctrine\DBAL\Configuration
|
||||
{
|
||||
@@ -95,8 +93,7 @@ class Configuration extends \Doctrine\DBAL\Configuration
|
||||
/**
|
||||
* Gets the strategy for automatically generating proxy classes.
|
||||
*
|
||||
* @return int Possible values are constants of Doctrine\ORM\Proxy\ProxyFactory.
|
||||
* @psalm-return AutogenerateMode
|
||||
* @return ProxyFactory::AUTOGENERATE_*
|
||||
*/
|
||||
public function getAutoGenerateProxyClasses()
|
||||
{
|
||||
@@ -106,9 +103,7 @@ class Configuration extends \Doctrine\DBAL\Configuration
|
||||
/**
|
||||
* Sets the strategy for automatically generating proxy classes.
|
||||
*
|
||||
* @param bool|int $autoGenerate Possible values are constants of Doctrine\ORM\Proxy\ProxyFactory.
|
||||
* @psalm-param bool|AutogenerateMode $autoGenerate
|
||||
* True is converted to AUTOGENERATE_ALWAYS, false to AUTOGENERATE_NEVER.
|
||||
* @param bool|ProxyFactory::AUTOGENERATE_* $autoGenerate True is converted to AUTOGENERATE_ALWAYS, false to AUTOGENERATE_NEVER.
|
||||
*
|
||||
* @return void
|
||||
*/
|
||||
@@ -164,7 +159,7 @@ class Configuration extends \Doctrine\DBAL\Configuration
|
||||
*
|
||||
* @return AnnotationDriver
|
||||
*/
|
||||
public function newDefaultAnnotationDriver($paths = [], $useSimpleAnnotationReader = true)
|
||||
public function newDefaultAnnotationDriver($paths = [], $useSimpleAnnotationReader = true, bool $reportFieldsWhereDeclared = false)
|
||||
{
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
@@ -203,7 +198,8 @@ class Configuration extends \Doctrine\DBAL\Configuration
|
||||
|
||||
return new AnnotationDriver(
|
||||
$reader,
|
||||
(array) $paths
|
||||
(array) $paths,
|
||||
$reportFieldsWhereDeclared
|
||||
);
|
||||
}
|
||||
|
||||
@@ -1121,4 +1117,14 @@ class Configuration extends \Doctrine\DBAL\Configuration
|
||||
|
||||
$this->_attributes['isLazyGhostObjectEnabled'] = $flag;
|
||||
}
|
||||
|
||||
public function setRejectIdCollisionInIdentityMap(bool $flag): void
|
||||
{
|
||||
$this->_attributes['rejectIdCollisionInIdentityMap'] = $flag;
|
||||
}
|
||||
|
||||
public function isRejectIdCollisionInIdentityMapEnabled(): bool
|
||||
{
|
||||
return $this->_attributes['rejectIdCollisionInIdentityMap'] ?? false;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -953,6 +953,14 @@ class EntityManager implements EntityManagerInterface
|
||||
$this->unitOfWork->initializeObject($obj);
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
public function isUninitializedObject($obj): bool
|
||||
{
|
||||
return $this->unitOfWork->isUninitializedObject($obj);
|
||||
}
|
||||
|
||||
/**
|
||||
* Factory method to create EntityManager instances.
|
||||
*
|
||||
|
||||
@@ -0,0 +1,42 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Exception;
|
||||
|
||||
use function get_class;
|
||||
use function sprintf;
|
||||
|
||||
final class EntityIdentityCollisionException extends ORMException
|
||||
{
|
||||
/**
|
||||
* @param object $existingEntity
|
||||
* @param object $newEntity
|
||||
*/
|
||||
public static function create($existingEntity, $newEntity, string $idHash): self
|
||||
{
|
||||
return new self(
|
||||
sprintf(
|
||||
<<<'EXCEPTION'
|
||||
While adding an entity of class %s with an ID hash of "%s" to the identity map,
|
||||
another object of class %s was already present for the same ID. This exception
|
||||
is a safeguard against an internal inconsistency - IDs should uniquely map to
|
||||
entity object instances. This problem may occur if:
|
||||
|
||||
- you use application-provided IDs and reuse ID values;
|
||||
- database-provided IDs are reassigned after truncating the database without
|
||||
clearing the EntityManager;
|
||||
- you might have been using EntityManager#getReference() to create a reference
|
||||
for a nonexistent ID that was subsequently (by the RDBMS) assigned to another
|
||||
entity.
|
||||
|
||||
Otherwise, it might be an ORM-internal inconsistency, please report it.
|
||||
EXCEPTION
|
||||
,
|
||||
get_class($newEntity),
|
||||
$idHash,
|
||||
get_class($existingEntity)
|
||||
)
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -4,7 +4,12 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal\CommitOrder;
|
||||
|
||||
/** @internal */
|
||||
use Doctrine\Deprecations\Deprecation;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
* @deprecated
|
||||
*/
|
||||
final class Edge
|
||||
{
|
||||
/**
|
||||
@@ -27,6 +32,13 @@ final class Edge
|
||||
|
||||
public function __construct(string $from, string $to, int $weight)
|
||||
{
|
||||
Deprecation::triggerIfCalledFromOutside(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10547',
|
||||
'The %s class is deprecated and will be removed in ORM 3.0',
|
||||
self::class
|
||||
);
|
||||
|
||||
$this->from = $from;
|
||||
$this->to = $to;
|
||||
$this->weight = $weight;
|
||||
|
||||
@@ -4,9 +4,13 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal\CommitOrder;
|
||||
|
||||
use Doctrine\Deprecations\Deprecation;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
|
||||
/** @internal */
|
||||
/**
|
||||
* @internal
|
||||
* @deprecated
|
||||
*/
|
||||
final class Vertex
|
||||
{
|
||||
/**
|
||||
@@ -32,6 +36,13 @@ final class Vertex
|
||||
|
||||
public function __construct(string $hash, ClassMetadata $value)
|
||||
{
|
||||
Deprecation::triggerIfCalledFromOutside(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10547',
|
||||
'The %s class is deprecated and will be removed in ORM 3.0',
|
||||
self::class
|
||||
);
|
||||
|
||||
$this->hash = $hash;
|
||||
$this->value = $value;
|
||||
}
|
||||
|
||||
@@ -4,7 +4,12 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal\CommitOrder;
|
||||
|
||||
/** @internal */
|
||||
use Doctrine\Deprecations\Deprecation;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
* @deprecated
|
||||
*/
|
||||
final class VertexState
|
||||
{
|
||||
public const NOT_VISITED = 0;
|
||||
@@ -13,5 +18,11 @@ final class VertexState
|
||||
|
||||
private function __construct()
|
||||
{
|
||||
Deprecation::triggerIfCalledFromOutside(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10547',
|
||||
'The %s class is deprecated and will be removed in ORM 3.0',
|
||||
self::class
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -4,6 +4,7 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal;
|
||||
|
||||
use Doctrine\Deprecations\Deprecation;
|
||||
use Doctrine\ORM\Internal\CommitOrder\Edge;
|
||||
use Doctrine\ORM\Internal\CommitOrder\Vertex;
|
||||
use Doctrine\ORM\Internal\CommitOrder\VertexState;
|
||||
@@ -17,6 +18,8 @@ use function array_reverse;
|
||||
* using a depth-first searching (DFS) to traverse the graph built in memory.
|
||||
* This algorithm have a linear running time based on nodes (V) and dependency
|
||||
* between the nodes (E), resulting in a computational complexity of O(V + E).
|
||||
*
|
||||
* @deprecated
|
||||
*/
|
||||
class CommitOrderCalculator
|
||||
{
|
||||
@@ -45,6 +48,16 @@ class CommitOrderCalculator
|
||||
*/
|
||||
private $sortedNodeList = [];
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
Deprecation::triggerIfCalledFromOutside(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10547',
|
||||
'The %s class is deprecated and will be removed in ORM 3.0',
|
||||
self::class
|
||||
);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks for node (vertex) existence in graph.
|
||||
*
|
||||
|
||||
@@ -10,7 +10,6 @@ use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\ORM\Query;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
|
||||
use function array_fill_keys;
|
||||
use function array_keys;
|
||||
@@ -439,7 +438,7 @@ class ObjectHydrator extends AbstractHydrator
|
||||
// PATH B: Single-valued association
|
||||
$reflFieldValue = $reflField->getValue($parentObject);
|
||||
|
||||
if (! $reflFieldValue || isset($this->_hints[Query::HINT_REFRESH]) || ($reflFieldValue instanceof Proxy && ! $reflFieldValue->__isInitialized())) {
|
||||
if (! $reflFieldValue || isset($this->_hints[Query::HINT_REFRESH]) || $this->_uow->isUninitializedObject($reflFieldValue)) {
|
||||
// we only need to take action if this value is null,
|
||||
// we refresh the entity or its an uninitialized proxy.
|
||||
if (isset($nonemptyComponents[$dqlAlias])) {
|
||||
@@ -457,9 +456,6 @@ class ObjectHydrator extends AbstractHydrator
|
||||
$targetClass->reflFields[$inverseAssoc['fieldName']]->setValue($element, $parentObject);
|
||||
$this->_uow->setOriginalEntityProperty(spl_object_id($element), $inverseAssoc['fieldName'], $parentObject);
|
||||
}
|
||||
} elseif ($parentClass === $targetClass && $relation['mappedBy']) {
|
||||
// Special case: bi-directional self-referencing one-one on the same class
|
||||
$targetClass->reflFields[$relationField]->setValue($element, $parentObject);
|
||||
}
|
||||
} else {
|
||||
// For sure bidirectional, as there is no inverse side in unidirectional mappings
|
||||
|
||||
165
lib/Doctrine/ORM/Internal/TopologicalSort.php
Normal file
165
lib/Doctrine/ORM/Internal/TopologicalSort.php
Normal file
@@ -0,0 +1,165 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal;
|
||||
|
||||
use Doctrine\ORM\Internal\TopologicalSort\CycleDetectedException;
|
||||
|
||||
use function array_keys;
|
||||
use function array_reverse;
|
||||
use function array_unshift;
|
||||
use function spl_object_id;
|
||||
|
||||
/**
|
||||
* TopologicalSort implements topological sorting, which is an ordering
|
||||
* algorithm for directed graphs (DG) using a depth-first searching (DFS)
|
||||
* to traverse the graph built in memory.
|
||||
* This algorithm has a linear running time based on nodes (V) and edges
|
||||
* between the nodes (E), resulting in a computational complexity of O(V + E).
|
||||
*
|
||||
* @internal
|
||||
*/
|
||||
final class TopologicalSort
|
||||
{
|
||||
private const NOT_VISITED = 1;
|
||||
private const IN_PROGRESS = 2;
|
||||
private const VISITED = 3;
|
||||
|
||||
/**
|
||||
* Array of all nodes, indexed by object ids.
|
||||
*
|
||||
* @var array<int, object>
|
||||
*/
|
||||
private $nodes = [];
|
||||
|
||||
/**
|
||||
* DFS state for the different nodes, indexed by node object id and using one of
|
||||
* this class' constants as value.
|
||||
*
|
||||
* @var array<int, self::*>
|
||||
*/
|
||||
private $states = [];
|
||||
|
||||
/**
|
||||
* Edges between the nodes. The first-level key is the object id of the outgoing
|
||||
* node; the second array maps the destination node by object id as key. The final
|
||||
* boolean value indicates whether the edge is optional or not.
|
||||
*
|
||||
* @var array<int, array<int, bool>>
|
||||
*/
|
||||
private $edges = [];
|
||||
|
||||
/**
|
||||
* Builds up the result during the DFS.
|
||||
*
|
||||
* @var list<object>
|
||||
*/
|
||||
private $sortResult = [];
|
||||
|
||||
/** @param object $node */
|
||||
public function addNode($node): void
|
||||
{
|
||||
$id = spl_object_id($node);
|
||||
$this->nodes[$id] = $node;
|
||||
$this->states[$id] = self::NOT_VISITED;
|
||||
$this->edges[$id] = [];
|
||||
}
|
||||
|
||||
/** @param object $node */
|
||||
public function hasNode($node): bool
|
||||
{
|
||||
return isset($this->nodes[spl_object_id($node)]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds a new edge between two nodes to the graph
|
||||
*
|
||||
* @param object $from
|
||||
* @param object $to
|
||||
* @param bool $optional This indicates whether the edge may be ignored during the topological sort if it is necessary to break cycles.
|
||||
*/
|
||||
public function addEdge($from, $to, bool $optional): void
|
||||
{
|
||||
$fromId = spl_object_id($from);
|
||||
$toId = spl_object_id($to);
|
||||
|
||||
if (isset($this->edges[$fromId][$toId]) && $this->edges[$fromId][$toId] === false) {
|
||||
return; // we already know about this dependency, and it is not optional
|
||||
}
|
||||
|
||||
$this->edges[$fromId][$toId] = $optional;
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a topological sort of all nodes. When we have an edge A->B between two nodes
|
||||
* A and B, then A will be listed before B in the result.
|
||||
*
|
||||
* @return list<object>
|
||||
*/
|
||||
public function sort(): array
|
||||
{
|
||||
/*
|
||||
* When possible, keep objects in the result in the same order in which they were added as nodes.
|
||||
* Since nodes are unshifted into $this->>sortResult (see the visit() method), that means we
|
||||
* need to work them in array_reverse order here.
|
||||
*/
|
||||
foreach (array_reverse(array_keys($this->nodes)) as $oid) {
|
||||
if ($this->states[$oid] === self::NOT_VISITED) {
|
||||
$this->visit($oid);
|
||||
}
|
||||
}
|
||||
|
||||
return $this->sortResult;
|
||||
}
|
||||
|
||||
private function visit(int $oid): void
|
||||
{
|
||||
if ($this->states[$oid] === self::IN_PROGRESS) {
|
||||
// This node is already on the current DFS stack. We've found a cycle!
|
||||
throw new CycleDetectedException($this->nodes[$oid]);
|
||||
}
|
||||
|
||||
if ($this->states[$oid] === self::VISITED) {
|
||||
// We've reached a node that we've already seen, including all
|
||||
// other nodes that are reachable from here. We're done here, return.
|
||||
return;
|
||||
}
|
||||
|
||||
$this->states[$oid] = self::IN_PROGRESS;
|
||||
|
||||
// Continue the DFS downwards the edge list
|
||||
foreach ($this->edges[$oid] as $adjacentId => $optional) {
|
||||
try {
|
||||
$this->visit($adjacentId);
|
||||
} catch (CycleDetectedException $exception) {
|
||||
if ($exception->isCycleCollected()) {
|
||||
// There is a complete cycle downstream of the current node. We cannot
|
||||
// do anything about that anymore.
|
||||
throw $exception;
|
||||
}
|
||||
|
||||
if ($optional) {
|
||||
// The current edge is part of a cycle, but it is optional and the closest
|
||||
// such edge while backtracking. Break the cycle here by skipping the edge
|
||||
// and continuing with the next one.
|
||||
continue;
|
||||
}
|
||||
|
||||
// We have found a cycle and cannot break it at $edge. Best we can do
|
||||
// is to retreat from the current vertex, hoping that somewhere up the
|
||||
// stack this can be salvaged.
|
||||
$this->states[$oid] = self::NOT_VISITED;
|
||||
$exception->addToCycle($this->nodes[$oid]);
|
||||
|
||||
throw $exception;
|
||||
}
|
||||
}
|
||||
|
||||
// We have traversed all edges and visited all other nodes reachable from here.
|
||||
// So we're done with this vertex as well.
|
||||
|
||||
$this->states[$oid] = self::VISITED;
|
||||
array_unshift($this->sortResult, $this->nodes[$oid]);
|
||||
}
|
||||
}
|
||||
@@ -0,0 +1,55 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Internal\TopologicalSort;
|
||||
|
||||
use RuntimeException;
|
||||
|
||||
use function array_unshift;
|
||||
|
||||
class CycleDetectedException extends RuntimeException
|
||||
{
|
||||
/** @var list<object> */
|
||||
private $cycle;
|
||||
|
||||
/** @var object */
|
||||
private $startNode;
|
||||
|
||||
/**
|
||||
* Do we have the complete cycle collected?
|
||||
*
|
||||
* @var bool
|
||||
*/
|
||||
private $cycleCollected = false;
|
||||
|
||||
/** @param object $startNode */
|
||||
public function __construct($startNode)
|
||||
{
|
||||
parent::__construct('A cycle has been detected, so a topological sort is not possible. The getCycle() method provides the list of nodes that form the cycle.');
|
||||
|
||||
$this->startNode = $startNode;
|
||||
$this->cycle = [$startNode];
|
||||
}
|
||||
|
||||
/** @return list<object> */
|
||||
public function getCycle(): array
|
||||
{
|
||||
return $this->cycle;
|
||||
}
|
||||
|
||||
/** @param object $node */
|
||||
public function addToCycle($node): void
|
||||
{
|
||||
array_unshift($this->cycle, $node);
|
||||
|
||||
if ($node === $this->startNode) {
|
||||
$this->cycleCollected = true;
|
||||
}
|
||||
}
|
||||
|
||||
public function isCycleCollected(): bool
|
||||
{
|
||||
return $this->cycleCollected;
|
||||
}
|
||||
}
|
||||
@@ -113,7 +113,7 @@ class ClassMetadataFactory extends AbstractClassMetadataFactory
|
||||
if ($parent) {
|
||||
$class->setInheritanceType($parent->inheritanceType);
|
||||
$class->setDiscriminatorColumn($parent->discriminatorColumn);
|
||||
$class->setIdGeneratorType($parent->generatorType);
|
||||
$this->inheritIdGeneratorMapping($class, $parent);
|
||||
$this->addInheritedFields($class, $parent);
|
||||
$this->addInheritedRelations($class, $parent);
|
||||
$this->addInheritedEmbeddedClasses($class, $parent);
|
||||
@@ -141,12 +141,8 @@ class ClassMetadataFactory extends AbstractClassMetadataFactory
|
||||
throw MappingException::reflectionFailure($class->getName(), $e);
|
||||
}
|
||||
|
||||
// If this class has a parent the id generator strategy is inherited.
|
||||
// However this is only true if the hierarchy of parents contains the root entity,
|
||||
// if it consists of mapped superclasses these don't necessarily include the id field.
|
||||
if ($parent && $rootEntityFound) {
|
||||
$this->inheritIdGeneratorMapping($class, $parent);
|
||||
} else {
|
||||
// Complete id generator mapping when the generator was declared/added in this class
|
||||
if ($class->identifier && (! $parent || ! $parent->identifier)) {
|
||||
$this->completeIdGeneratorMapping($class);
|
||||
}
|
||||
|
||||
|
||||
@@ -1176,7 +1176,7 @@ class ClassMetadataInfo implements ClassMetadata
|
||||
$this->namespace = $reflService->getClassNamespace($this->name);
|
||||
|
||||
if ($this->reflClass) {
|
||||
$this->name = $this->rootEntityName = $this->reflClass->getName();
|
||||
$this->name = $this->rootEntityName = $this->reflClass->name;
|
||||
}
|
||||
|
||||
$this->table['name'] = $this->namingStrategy->classToTableName($this->name);
|
||||
@@ -2764,6 +2764,10 @@ class ClassMetadataInfo implements ClassMetadata
|
||||
$this->fieldMappings[$fieldMapping['fieldName']] = $fieldMapping;
|
||||
$this->columnNames[$fieldMapping['fieldName']] = $fieldMapping['columnName'];
|
||||
$this->fieldNames[$fieldMapping['columnName']] = $fieldMapping['fieldName'];
|
||||
|
||||
if (isset($fieldMapping['generated'])) {
|
||||
$this->requiresFetchAfterChange = true;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -3862,7 +3866,7 @@ class ClassMetadataInfo implements ClassMetadata
|
||||
{
|
||||
$reflectionProperty = $reflService->getAccessibleProperty($class, $field);
|
||||
if ($reflectionProperty !== null && PHP_VERSION_ID >= 80100 && $reflectionProperty->isReadOnly()) {
|
||||
$declaringClass = $reflectionProperty->getDeclaringClass()->name;
|
||||
$declaringClass = $reflectionProperty->class;
|
||||
if ($declaringClass !== $class) {
|
||||
$reflectionProperty = $reflService->getAccessibleProperty($declaringClass, $field);
|
||||
}
|
||||
|
||||
@@ -34,6 +34,7 @@ use function is_numeric;
|
||||
class AnnotationDriver extends CompatibilityAnnotationDriver
|
||||
{
|
||||
use ColocatedMappingDriver;
|
||||
use ReflectionBasedDriver;
|
||||
|
||||
/**
|
||||
* The annotation reader.
|
||||
@@ -60,7 +61,7 @@ class AnnotationDriver extends CompatibilityAnnotationDriver
|
||||
* @param Reader $reader The AnnotationReader to use
|
||||
* @param string|string[]|null $paths One or multiple paths where mapping classes can be found.
|
||||
*/
|
||||
public function __construct($reader, $paths = null)
|
||||
public function __construct($reader, $paths = null, bool $reportFieldsWhereDeclared = false)
|
||||
{
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
@@ -70,6 +71,17 @@ class AnnotationDriver extends CompatibilityAnnotationDriver
|
||||
$this->reader = $reader;
|
||||
|
||||
$this->addPaths((array) $paths);
|
||||
|
||||
if (! $reportFieldsWhereDeclared) {
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10455',
|
||||
'In ORM 3.0, the AttributeDriver will report fields for the classes where they are declared. This may uncover invalid mapping configurations. To opt into the new mode also with the AnnotationDriver today, set the "reportFieldsWhereDeclared" constructor parameter to true.',
|
||||
self::class
|
||||
);
|
||||
}
|
||||
|
||||
$this->reportFieldsWhereDeclared = $reportFieldsWhereDeclared;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -348,20 +360,12 @@ class AnnotationDriver extends CompatibilityAnnotationDriver
|
||||
|
||||
// Evaluate annotations on properties/fields
|
||||
foreach ($class->getProperties() as $property) {
|
||||
if (
|
||||
$metadata->isMappedSuperclass && ! $property->isPrivate()
|
||||
||
|
||||
$metadata->isInheritedField($property->name)
|
||||
||
|
||||
$metadata->isInheritedAssociation($property->name)
|
||||
||
|
||||
$metadata->isInheritedEmbeddedClass($property->name)
|
||||
) {
|
||||
if ($this->isRepeatedPropertyDeclaration($property, $metadata)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$mapping = [];
|
||||
$mapping['fieldName'] = $property->getName();
|
||||
$mapping['fieldName'] = $property->name;
|
||||
|
||||
// Evaluate @Cache annotation
|
||||
$cacheAnnot = $this->reader->getPropertyAnnotation($property, Mapping\Cache::class);
|
||||
@@ -394,7 +398,7 @@ class AnnotationDriver extends CompatibilityAnnotationDriver
|
||||
// @Column, @OneToOne, @OneToMany, @ManyToOne, @ManyToMany
|
||||
$columnAnnot = $this->reader->getPropertyAnnotation($property, Mapping\Column::class);
|
||||
if ($columnAnnot) {
|
||||
$mapping = $this->columnToArray($property->getName(), $columnAnnot);
|
||||
$mapping = $this->columnToArray($property->name, $columnAnnot);
|
||||
|
||||
$idAnnot = $this->reader->getPropertyAnnotation($property, Mapping\Id::class);
|
||||
if ($idAnnot) {
|
||||
|
||||
@@ -29,6 +29,7 @@ use const PHP_VERSION_ID;
|
||||
class AttributeDriver extends CompatibilityAnnotationDriver
|
||||
{
|
||||
use ColocatedMappingDriver;
|
||||
use ReflectionBasedDriver;
|
||||
|
||||
private const ENTITY_ATTRIBUTE_CLASSES = [
|
||||
Mapping\Entity::class => 1,
|
||||
@@ -52,7 +53,7 @@ class AttributeDriver extends CompatibilityAnnotationDriver
|
||||
protected $reader;
|
||||
|
||||
/** @param array<string> $paths */
|
||||
public function __construct(array $paths)
|
||||
public function __construct(array $paths, bool $reportFieldsWhereDeclared = false)
|
||||
{
|
||||
if (PHP_VERSION_ID < 80000) {
|
||||
throw new LogicException(
|
||||
@@ -72,6 +73,17 @@ class AttributeDriver extends CompatibilityAnnotationDriver
|
||||
self::class
|
||||
);
|
||||
}
|
||||
|
||||
if (! $reportFieldsWhereDeclared) {
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10455',
|
||||
'In ORM 3.0, the AttributeDriver will report fields for the classes where they are declared. This may uncover invalid mapping configurations. To opt into the new mode today, set the "reportFieldsWhereDeclared" constructor parameter to true.',
|
||||
self::class
|
||||
);
|
||||
}
|
||||
|
||||
$this->reportFieldsWhereDeclared = $reportFieldsWhereDeclared;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -297,20 +309,13 @@ class AttributeDriver extends CompatibilityAnnotationDriver
|
||||
|
||||
foreach ($reflectionClass->getProperties() as $property) {
|
||||
assert($property instanceof ReflectionProperty);
|
||||
if (
|
||||
$metadata->isMappedSuperclass && ! $property->isPrivate()
|
||||
||
|
||||
$metadata->isInheritedField($property->name)
|
||||
||
|
||||
$metadata->isInheritedAssociation($property->name)
|
||||
||
|
||||
$metadata->isInheritedEmbeddedClass($property->name)
|
||||
) {
|
||||
|
||||
if ($this->isRepeatedPropertyDeclaration($property, $metadata)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$mapping = [];
|
||||
$mapping['fieldName'] = $property->getName();
|
||||
$mapping['fieldName'] = $property->name;
|
||||
|
||||
// Evaluate #[Cache] attribute
|
||||
$cacheAttribute = $this->reader->getPropertyAttribute($property, Mapping\Cache::class);
|
||||
@@ -345,7 +350,7 @@ class AttributeDriver extends CompatibilityAnnotationDriver
|
||||
$embeddedAttribute = $this->reader->getPropertyAttribute($property, Mapping\Embedded::class);
|
||||
|
||||
if ($columnAttribute !== null) {
|
||||
$mapping = $this->columnToArray($property->getName(), $columnAttribute);
|
||||
$mapping = $this->columnToArray($property->name, $columnAttribute);
|
||||
|
||||
if ($this->reader->getPropertyAttribute($property, Mapping\Id::class)) {
|
||||
$mapping['id'] = true;
|
||||
|
||||
54
lib/Doctrine/ORM/Mapping/Driver/ReflectionBasedDriver.php
Normal file
54
lib/Doctrine/ORM/Mapping/Driver/ReflectionBasedDriver.php
Normal file
@@ -0,0 +1,54 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Mapping\Driver;
|
||||
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use ReflectionProperty;
|
||||
|
||||
/** @internal */
|
||||
trait ReflectionBasedDriver
|
||||
{
|
||||
/** @var bool */
|
||||
private $reportFieldsWhereDeclared = false;
|
||||
|
||||
/**
|
||||
* Helps to deal with the case that reflection may report properties inherited from parent classes.
|
||||
* When we know about the fields already (inheritance has been anticipated in ClassMetadataFactory),
|
||||
* the driver must skip them.
|
||||
*
|
||||
* The declaring classes may mismatch when there are private properties: The same property name may be
|
||||
* reported multiple times, but since it is private, it is in fact multiple (different) properties in
|
||||
* different classes. In that case, report the property as an individual field. (ClassMetadataFactory will
|
||||
* probably fail in that case, though.)
|
||||
*/
|
||||
private function isRepeatedPropertyDeclaration(ReflectionProperty $property, ClassMetadata $metadata): bool
|
||||
{
|
||||
if (! $this->reportFieldsWhereDeclared) {
|
||||
return $metadata->isMappedSuperclass && ! $property->isPrivate()
|
||||
|| $metadata->isInheritedField($property->name)
|
||||
|| $metadata->isInheritedAssociation($property->name)
|
||||
|| $metadata->isInheritedEmbeddedClass($property->name);
|
||||
}
|
||||
|
||||
$declaringClass = $property->class;
|
||||
|
||||
if (
|
||||
isset($metadata->fieldMappings[$property->name]['declared'])
|
||||
&& $metadata->fieldMappings[$property->name]['declared'] === $declaringClass
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if (
|
||||
isset($metadata->associationMappings[$property->name]['declared'])
|
||||
&& $metadata->associationMappings[$property->name]['declared'] === $declaringClass
|
||||
) {
|
||||
return true;
|
||||
}
|
||||
|
||||
return isset($metadata->embeddedClasses[$property->name]['declared'])
|
||||
&& $metadata->embeddedClasses[$property->name]['declared'] === $declaringClass;
|
||||
}
|
||||
}
|
||||
@@ -944,7 +944,10 @@ class XmlDriver extends FileDriver
|
||||
private function getCascadeMappings(SimpleXMLElement $cascadeElement): array
|
||||
{
|
||||
$cascades = [];
|
||||
foreach ($cascadeElement->children() as $action) {
|
||||
$children = $cascadeElement->children();
|
||||
assert($children !== null);
|
||||
|
||||
foreach ($children as $action) {
|
||||
// According to the JPA specifications, XML uses "cascade-persist"
|
||||
// instead of "persist". Here, both variations
|
||||
// are supported because YAML, Annotation and Attribute use "persist"
|
||||
|
||||
@@ -73,7 +73,7 @@ final class ReflectionPropertiesGetter
|
||||
|
||||
$parentClass = $currentClass->getParentClass();
|
||||
if ($parentClass) {
|
||||
$parentClassName = $parentClass->getName();
|
||||
$parentClassName = $parentClass->name;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -111,14 +111,14 @@ final class ReflectionPropertiesGetter
|
||||
private function getAccessibleProperty(ReflectionProperty $property): ?ReflectionProperty
|
||||
{
|
||||
return $this->reflectionService->getAccessibleProperty(
|
||||
$property->getDeclaringClass()->getName(),
|
||||
$property->getName()
|
||||
$property->class,
|
||||
$property->name
|
||||
);
|
||||
}
|
||||
|
||||
private function getLogicalName(ReflectionProperty $property): string
|
||||
{
|
||||
$propertyName = $property->getName();
|
||||
$propertyName = $property->name;
|
||||
|
||||
if ($property->isPublic()) {
|
||||
return $propertyName;
|
||||
@@ -128,6 +128,6 @@ final class ReflectionPropertiesGetter
|
||||
return "\0*\0" . $propertyName;
|
||||
}
|
||||
|
||||
return "\0" . $property->getDeclaringClass()->getName() . "\0" . $propertyName;
|
||||
return "\0" . $property->class . "\0" . $propertyName;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -38,7 +38,7 @@ class ReflectionEmbeddedProperty extends ReflectionProperty
|
||||
$this->childProperty = $childProperty;
|
||||
$this->embeddedClass = (string) $embeddedClass;
|
||||
|
||||
parent::__construct($childProperty->getDeclaringClass()->getName(), $childProperty->getName());
|
||||
parent::__construct($childProperty->class, $childProperty->name);
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -28,8 +28,8 @@ class ReflectionEnumProperty extends ReflectionProperty
|
||||
$this->enumType = $enumType;
|
||||
|
||||
parent::__construct(
|
||||
$originalReflectionProperty->getDeclaringClass()->getName(),
|
||||
$originalReflectionProperty->getName()
|
||||
$originalReflectionProperty->class,
|
||||
$originalReflectionProperty->name
|
||||
);
|
||||
}
|
||||
|
||||
@@ -98,7 +98,7 @@ class ReflectionEnumProperty extends ReflectionProperty
|
||||
} catch (ValueError $e) {
|
||||
throw MappingException::invalidEnumValue(
|
||||
get_class($object),
|
||||
$this->originalReflectionProperty->getName(),
|
||||
$this->originalReflectionProperty->name,
|
||||
(string) $value,
|
||||
$enumType,
|
||||
$e
|
||||
|
||||
@@ -30,7 +30,10 @@ class UnderscoreNamingStrategy implements NamingStrategy
|
||||
/** @var int */
|
||||
private $case;
|
||||
|
||||
/** @var string */
|
||||
/**
|
||||
* @var string
|
||||
* @psalm-var non-empty-string
|
||||
*/
|
||||
private $pattern;
|
||||
|
||||
/**
|
||||
|
||||
@@ -62,7 +62,8 @@ final class ORMSetup
|
||||
*/
|
||||
public static function createDefaultAnnotationDriver(
|
||||
array $paths = [],
|
||||
?CacheItemPoolInterface $cache = null
|
||||
?CacheItemPoolInterface $cache = null,
|
||||
bool $reportFieldsWhereDeclared = false
|
||||
): AnnotationDriver {
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
@@ -88,7 +89,7 @@ final class ORMSetup
|
||||
$reader = new PsrCachedReader($reader, $cache);
|
||||
}
|
||||
|
||||
return new AnnotationDriver($reader, $paths);
|
||||
return new AnnotationDriver($reader, $paths, $reportFieldsWhereDeclared);
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -100,10 +101,11 @@ final class ORMSetup
|
||||
array $paths,
|
||||
bool $isDevMode = false,
|
||||
?string $proxyDir = null,
|
||||
?CacheItemPoolInterface $cache = null
|
||||
?CacheItemPoolInterface $cache = null,
|
||||
bool $reportFieldsWhereDeclared = false
|
||||
): Configuration {
|
||||
$config = self::createConfiguration($isDevMode, $proxyDir, $cache);
|
||||
$config->setMetadataDriverImpl(new AttributeDriver($paths));
|
||||
$config->setMetadataDriverImpl(new AttributeDriver($paths, $reportFieldsWhereDeclared));
|
||||
|
||||
return $config;
|
||||
}
|
||||
|
||||
@@ -183,7 +183,7 @@ class BasicEntityPersister implements EntityPersister
|
||||
*
|
||||
* @var IdentifierFlattener
|
||||
*/
|
||||
private $identifierFlattener;
|
||||
protected $identifierFlattener;
|
||||
|
||||
/** @var CachedPersisterContext */
|
||||
protected $currentPersisterContext;
|
||||
@@ -256,17 +256,17 @@ class BasicEntityPersister implements EntityPersister
|
||||
public function executeInserts()
|
||||
{
|
||||
if (! $this->queuedInserts) {
|
||||
return [];
|
||||
return;
|
||||
}
|
||||
|
||||
$postInsertIds = [];
|
||||
$uow = $this->em->getUnitOfWork();
|
||||
$idGenerator = $this->class->idGenerator;
|
||||
$isPostInsertId = $idGenerator->isPostInsertGenerator();
|
||||
|
||||
$stmt = $this->conn->prepare($this->getInsertSQL());
|
||||
$tableName = $this->class->getTableName();
|
||||
|
||||
foreach ($this->queuedInserts as $entity) {
|
||||
foreach ($this->queuedInserts as $key => $entity) {
|
||||
$insertData = $this->prepareInsertData($entity);
|
||||
|
||||
if (isset($insertData[$tableName])) {
|
||||
@@ -280,12 +280,10 @@ class BasicEntityPersister implements EntityPersister
|
||||
$stmt->executeStatement();
|
||||
|
||||
if ($isPostInsertId) {
|
||||
$generatedId = $idGenerator->generateId($this->em, $entity);
|
||||
$id = [$this->class->identifier[0] => $generatedId];
|
||||
$postInsertIds[] = [
|
||||
'generatedId' => $generatedId,
|
||||
'entity' => $entity,
|
||||
];
|
||||
$generatedId = $idGenerator->generateId($this->em, $entity);
|
||||
$id = [$this->class->identifier[0] => $generatedId];
|
||||
|
||||
$uow->assignPostInsertId($entity, $generatedId);
|
||||
} else {
|
||||
$id = $this->class->getIdentifierValues($entity);
|
||||
}
|
||||
@@ -293,11 +291,16 @@ class BasicEntityPersister implements EntityPersister
|
||||
if ($this->class->requiresFetchAfterChange) {
|
||||
$this->assignDefaultVersionAndUpsertableValues($entity, $id);
|
||||
}
|
||||
|
||||
// Unset this queued insert, so that the prepareUpdateData() method knows right away
|
||||
// (for the next entity already) that the current entity has been written to the database
|
||||
// and no extra updates need to be scheduled to refer to it.
|
||||
//
|
||||
// In \Doctrine\ORM\UnitOfWork::executeInserts(), the UoW already removed entities
|
||||
// from its own list (\Doctrine\ORM\UnitOfWork::$entityInsertions) right after they
|
||||
// were given to our addInsert() method.
|
||||
unset($this->queuedInserts[$key]);
|
||||
}
|
||||
|
||||
$this->queuedInserts = [];
|
||||
|
||||
return $postInsertIds;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -376,7 +379,7 @@ class BasicEntityPersister implements EntityPersister
|
||||
* @return int[]|null[]|string[]
|
||||
* @psalm-return list<int|string|null>
|
||||
*/
|
||||
private function extractIdentifierTypes(array $id, ClassMetadata $versionedClass): array
|
||||
final protected function extractIdentifierTypes(array $id, ClassMetadata $versionedClass): array
|
||||
{
|
||||
$types = [];
|
||||
|
||||
@@ -675,10 +678,30 @@ class BasicEntityPersister implements EntityPersister
|
||||
if ($newVal !== null) {
|
||||
$oid = spl_object_id($newVal);
|
||||
|
||||
if (isset($this->queuedInserts[$oid]) || $uow->isScheduledForInsert($newVal)) {
|
||||
// The associated entity $newVal is not yet persisted, so we must
|
||||
// set $newVal = null, in order to insert a null value and schedule an
|
||||
// extra update on the UnitOfWork.
|
||||
// If the associated entity $newVal is not yet persisted and/or does not yet have
|
||||
// an ID assigned, we must set $newVal = null. This will insert a null value and
|
||||
// schedule an extra update on the UnitOfWork.
|
||||
//
|
||||
// This gives us extra time to a) possibly obtain a database-generated identifier
|
||||
// value for $newVal, and b) insert $newVal into the database before the foreign
|
||||
// key reference is being made.
|
||||
//
|
||||
// When looking at $this->queuedInserts and $uow->isScheduledForInsert, be aware
|
||||
// of the implementation details that our own executeInserts() method will remove
|
||||
// entities from the former as soon as the insert statement has been executed and
|
||||
// a post-insert ID has been assigned (if necessary), and that the UnitOfWork has
|
||||
// already removed entities from its own list at the time they were passed to our
|
||||
// addInsert() method.
|
||||
//
|
||||
// Then, there is one extra exception we can make: An entity that references back to itself
|
||||
// _and_ uses an application-provided ID (the "NONE" generator strategy) also does not
|
||||
// need the extra update, although it is still in the list of insertions itself.
|
||||
// This looks like a minor optimization at first, but is the capstone for being able to
|
||||
// use non-NULLable, self-referencing associations in applications that provide IDs (like UUIDs).
|
||||
if (
|
||||
(isset($this->queuedInserts[$oid]) || $uow->isScheduledForInsert($newVal))
|
||||
&& ! ($newVal === $entity && $this->class->isIdentifierNatural())
|
||||
) {
|
||||
$uow->scheduleExtraUpdate($entity, [$field => [null, $newVal]]);
|
||||
|
||||
$newVal = null;
|
||||
|
||||
@@ -109,17 +109,15 @@ interface EntityPersister
|
||||
public function addInsert($entity);
|
||||
|
||||
/**
|
||||
* Executes all queued entity insertions and returns any generated post-insert
|
||||
* identifiers that were created as a result of the insertions.
|
||||
* Executes all queued entity insertions.
|
||||
*
|
||||
* If no inserts are queued, invoking this method is a NOOP.
|
||||
*
|
||||
* @psalm-return list<array{
|
||||
* @psalm-return void|list<array{
|
||||
* generatedId: int,
|
||||
* entity: object
|
||||
* }> An array of any generated post-insert IDs. This will be
|
||||
* an empty array if the entity class does not use the
|
||||
* IDENTITY generation strategy.
|
||||
* }> Returning an array of generated post-insert IDs is deprecated, implementations
|
||||
* should call UnitOfWork::assignPostInsertId() and return void.
|
||||
*/
|
||||
public function executeInserts();
|
||||
|
||||
|
||||
@@ -11,8 +11,11 @@ use Doctrine\DBAL\Types\Types;
|
||||
use Doctrine\ORM\Internal\SQLResultCasing;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\Utility\PersisterHelper;
|
||||
use LengthException;
|
||||
|
||||
use function array_combine;
|
||||
use function array_keys;
|
||||
use function array_values;
|
||||
use function implode;
|
||||
|
||||
/**
|
||||
@@ -109,10 +112,10 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||
public function executeInserts()
|
||||
{
|
||||
if (! $this->queuedInserts) {
|
||||
return [];
|
||||
return;
|
||||
}
|
||||
|
||||
$postInsertIds = [];
|
||||
$uow = $this->em->getUnitOfWork();
|
||||
$idGenerator = $this->class->idGenerator;
|
||||
$isPostInsertId = $idGenerator->isPostInsertGenerator();
|
||||
$rootClass = $this->class->name !== $this->class->rootEntityName
|
||||
@@ -157,20 +160,14 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||
$rootTableStmt->executeStatement();
|
||||
|
||||
if ($isPostInsertId) {
|
||||
$generatedId = $idGenerator->generateId($this->em, $entity);
|
||||
$id = [$this->class->identifier[0] => $generatedId];
|
||||
$postInsertIds[] = [
|
||||
'generatedId' => $generatedId,
|
||||
'entity' => $entity,
|
||||
];
|
||||
$generatedId = $idGenerator->generateId($this->em, $entity);
|
||||
$id = [$this->class->identifier[0] => $generatedId];
|
||||
|
||||
$uow->assignPostInsertId($entity, $generatedId);
|
||||
} else {
|
||||
$id = $this->em->getUnitOfWork()->getEntityIdentifier($entity);
|
||||
}
|
||||
|
||||
if ($this->class->requiresFetchAfterChange) {
|
||||
$this->assignDefaultVersionAndUpsertableValues($entity, $id);
|
||||
}
|
||||
|
||||
// Execute inserts on subtables.
|
||||
// The order doesn't matter because all child tables link to the root table via FK.
|
||||
foreach ($subTableStmts as $tableName => $stmt) {
|
||||
@@ -191,11 +188,13 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||
|
||||
$stmt->executeStatement();
|
||||
}
|
||||
|
||||
if ($this->class->requiresFetchAfterChange) {
|
||||
$this->assignDefaultVersionAndUpsertableValues($entity, $id);
|
||||
}
|
||||
}
|
||||
|
||||
$this->queuedInserts = [];
|
||||
|
||||
return $postInsertIds;
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -514,6 +513,7 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||
|| isset($this->class->associationMappings[$name]['inherited'])
|
||||
|| ($this->class->isVersioned && $this->class->versionField === $name)
|
||||
|| isset($this->class->embeddedClasses[$name])
|
||||
|| isset($this->class->fieldMappings[$name]['notInsertable'])
|
||||
) {
|
||||
continue;
|
||||
}
|
||||
@@ -556,6 +556,60 @@ class JoinedSubclassPersister extends AbstractEntityInheritancePersister
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* {@inheritDoc}
|
||||
*/
|
||||
protected function fetchVersionAndNotUpsertableValues($versionedClass, array $id)
|
||||
{
|
||||
$columnNames = [];
|
||||
foreach ($this->class->fieldMappings as $key => $column) {
|
||||
$class = null;
|
||||
if ($this->class->isVersioned && $key === $versionedClass->versionField) {
|
||||
$class = $versionedClass;
|
||||
} elseif (isset($column['generated'])) {
|
||||
$class = isset($column['inherited'])
|
||||
? $this->em->getClassMetadata($column['inherited'])
|
||||
: $this->class;
|
||||
} else {
|
||||
continue;
|
||||
}
|
||||
|
||||
$columnNames[$key] = $this->getSelectColumnSQL($key, $class);
|
||||
}
|
||||
|
||||
$tableName = $this->quoteStrategy->getTableName($versionedClass, $this->platform);
|
||||
$baseTableAlias = $this->getSQLTableAlias($this->class->name);
|
||||
$joinSql = $this->getJoinSql($baseTableAlias);
|
||||
$identifier = $this->quoteStrategy->getIdentifierColumnNames($versionedClass, $this->platform);
|
||||
foreach ($identifier as $i => $idValue) {
|
||||
$identifier[$i] = $baseTableAlias . '.' . $idValue;
|
||||
}
|
||||
|
||||
$sql = 'SELECT ' . implode(', ', $columnNames)
|
||||
. ' FROM ' . $tableName . ' ' . $baseTableAlias
|
||||
. $joinSql
|
||||
. ' WHERE ' . implode(' = ? AND ', $identifier) . ' = ?';
|
||||
|
||||
$flatId = $this->identifierFlattener->flattenIdentifier($versionedClass, $id);
|
||||
$values = $this->conn->fetchNumeric(
|
||||
$sql,
|
||||
array_values($flatId),
|
||||
$this->extractIdentifierTypes($id, $versionedClass)
|
||||
);
|
||||
|
||||
if ($values === false) {
|
||||
throw new LengthException('Unexpected empty result for database query.');
|
||||
}
|
||||
|
||||
$values = array_combine(array_keys($columnNames), $values);
|
||||
|
||||
if (! $values) {
|
||||
throw new LengthException('Unexpected number of database columns.');
|
||||
}
|
||||
|
||||
return $values;
|
||||
}
|
||||
|
||||
private function getJoinSql(string $baseTableAlias): string
|
||||
{
|
||||
$joinSql = '';
|
||||
|
||||
19
lib/Doctrine/ORM/Proxy/InternalProxy.php
Normal file
19
lib/Doctrine/ORM/Proxy/InternalProxy.php
Normal file
@@ -0,0 +1,19 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\ORM\Proxy;
|
||||
|
||||
use Doctrine\Persistence\Proxy;
|
||||
|
||||
/**
|
||||
* @internal
|
||||
*
|
||||
* @template T of object
|
||||
* @template-extends Proxy<T>
|
||||
*
|
||||
* @method void __setInitialized(bool $initialized)
|
||||
*/
|
||||
interface InternalProxy extends Proxy
|
||||
{
|
||||
}
|
||||
@@ -10,7 +10,11 @@ use Doctrine\Common\Proxy\Proxy as BaseProxy;
|
||||
* Interface for proxy classes.
|
||||
*
|
||||
* @deprecated 2.14. Use \Doctrine\Persistence\Proxy instead
|
||||
*
|
||||
* @template T of object
|
||||
* @template-extends BaseProxy<T>
|
||||
* @template-extends InternalProxy<T>
|
||||
*/
|
||||
interface Proxy extends BaseProxy
|
||||
interface Proxy extends BaseProxy, InternalProxy
|
||||
{
|
||||
}
|
||||
|
||||
@@ -17,7 +17,6 @@ use Doctrine\ORM\Proxy\Proxy as LegacyProxy;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\ORM\Utility\IdentifierFlattener;
|
||||
use Doctrine\Persistence\Mapping\ClassMetadata;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use ReflectionProperty;
|
||||
use Symfony\Component\VarExporter\ProxyHelper;
|
||||
use Symfony\Component\VarExporter\VarExporter;
|
||||
@@ -31,8 +30,6 @@ use function uksort;
|
||||
|
||||
/**
|
||||
* This factory is used to create proxy objects for entities at runtime.
|
||||
*
|
||||
* @psalm-type AutogenerateMode = ProxyFactory::AUTOGENERATE_NEVER|ProxyFactory::AUTOGENERATE_ALWAYS|ProxyFactory::AUTOGENERATE_FILE_NOT_EXISTS|ProxyFactory::AUTOGENERATE_EVAL|ProxyFactory::AUTOGENERATE_FILE_NOT_EXISTS_OR_CHANGED
|
||||
*/
|
||||
class ProxyFactory extends AbstractProxyFactory
|
||||
{
|
||||
@@ -93,19 +90,17 @@ EOPHP;
|
||||
* Initializes a new instance of the <tt>ProxyFactory</tt> class that is
|
||||
* connected to the given <tt>EntityManager</tt>.
|
||||
*
|
||||
* @param EntityManagerInterface $em The EntityManager the new factory works for.
|
||||
* @param string $proxyDir The directory to use for the proxy classes. It must exist.
|
||||
* @param string $proxyNs The namespace to use for the proxy classes.
|
||||
* @param bool|int $autoGenerate The strategy for automatically generating proxy classes. Possible
|
||||
* values are constants of {@see ProxyFactory::AUTOGENERATE_*}.
|
||||
* @psalm-param bool|AutogenerateMode $autoGenerate
|
||||
* @param EntityManagerInterface $em The EntityManager the new factory works for.
|
||||
* @param string $proxyDir The directory to use for the proxy classes. It must exist.
|
||||
* @param string $proxyNs The namespace to use for the proxy classes.
|
||||
* @param bool|self::AUTOGENERATE_* $autoGenerate The strategy for automatically generating proxy classes.
|
||||
*/
|
||||
public function __construct(EntityManagerInterface $em, $proxyDir, $proxyNs, $autoGenerate = self::AUTOGENERATE_NEVER)
|
||||
{
|
||||
$proxyGenerator = new ProxyGenerator($proxyDir, $proxyNs);
|
||||
|
||||
if ($em->getConfiguration()->isLazyGhostObjectEnabled()) {
|
||||
$proxyGenerator->setPlaceholder('baseProxyInterface', Proxy::class);
|
||||
$proxyGenerator->setPlaceholder('baseProxyInterface', InternalProxy::class);
|
||||
$proxyGenerator->setPlaceholder('useLazyGhostTrait', Closure::fromCallable([$this, 'generateUseLazyGhostTrait']));
|
||||
$proxyGenerator->setPlaceholder('skippedProperties', Closure::fromCallable([$this, 'generateSkippedProperties']));
|
||||
$proxyGenerator->setPlaceholder('serializeImpl', Closure::fromCallable([$this, 'generateSerializeImpl']));
|
||||
@@ -135,7 +130,7 @@ EOPHP;
|
||||
|
||||
$initializer = $this->definitions[$className]->initializer;
|
||||
|
||||
$proxy->__construct(static function (Proxy $object) use ($initializer, $proxy): void {
|
||||
$proxy->__construct(static function (InternalProxy $object) use ($initializer, $proxy): void {
|
||||
$initializer($object, $proxy);
|
||||
});
|
||||
|
||||
@@ -242,13 +237,13 @@ EOPHP;
|
||||
/**
|
||||
* Creates a closure capable of initializing a proxy
|
||||
*
|
||||
* @return Closure(Proxy, Proxy):void
|
||||
* @return Closure(InternalProxy, InternalProxy):void
|
||||
*
|
||||
* @throws EntityNotFoundException
|
||||
*/
|
||||
private function createLazyInitializer(ClassMetadata $classMetadata, EntityPersister $entityPersister): Closure
|
||||
{
|
||||
return function (Proxy $proxy, Proxy $original) use ($entityPersister, $classMetadata): void {
|
||||
return function (InternalProxy $proxy, InternalProxy $original) use ($entityPersister, $classMetadata): void {
|
||||
$identifier = $classMetadata->getIdentifierValues($original);
|
||||
$entity = $entityPersister->loadById($identifier, $original);
|
||||
|
||||
@@ -340,13 +335,13 @@ EOPHP;
|
||||
|
||||
while ($reflector) {
|
||||
foreach ($reflector->getProperties($filter) as $property) {
|
||||
$name = $property->getName();
|
||||
$name = $property->name;
|
||||
|
||||
if ($property->isStatic() || (($class->hasField($name) || $class->hasAssociation($name)) && ! isset($identifiers[$name]))) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$prefix = $property->isPrivate() ? "\0" . $property->getDeclaringClass()->getName() . "\0" : ($property->isProtected() ? "\0*\0" : '');
|
||||
$prefix = $property->isPrivate() ? "\0" . $property->class . "\0" : ($property->isProtected() ? "\0*\0" : '');
|
||||
|
||||
$skippedProperties[$prefix . $name] = true;
|
||||
}
|
||||
@@ -381,7 +376,7 @@ EOPHP;
|
||||
return $code . '$data = [];
|
||||
|
||||
foreach (parent::__sleep() as $name) {
|
||||
$value = $properties[$k = $name] ?? $properties[$k = "\0*\0$name"] ?? $properties[$k = "\0' . $reflector->getName() . '\0$name"] ?? $k = null;
|
||||
$value = $properties[$k = $name] ?? $properties[$k = "\0*\0$name"] ?? $properties[$k = "\0' . $reflector->name . '\0$name"] ?? $k = null;
|
||||
|
||||
if (null === $k) {
|
||||
trigger_error(sprintf(\'serialize(): "%s" returned as member variable from __sleep() but does not exist\', $name), \E_USER_NOTICE);
|
||||
|
||||
@@ -51,6 +51,14 @@ class FilterCollection
|
||||
*/
|
||||
private $enabledFilters = [];
|
||||
|
||||
/**
|
||||
* Instances of suspended filters.
|
||||
*
|
||||
* @var SQLFilter[]
|
||||
* @psalm-var array<string, SQLFilter>
|
||||
*/
|
||||
private $suspendedFilters = [];
|
||||
|
||||
/**
|
||||
* The filter hash from the last time the query was parsed.
|
||||
*
|
||||
@@ -83,6 +91,17 @@ class FilterCollection
|
||||
return $this->enabledFilters;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets all the suspended filters.
|
||||
*
|
||||
* @return SQLFilter[] The suspended filters.
|
||||
* @psalm-return array<string, SQLFilter>
|
||||
*/
|
||||
public function getSuspendedFilters(): array
|
||||
{
|
||||
return $this->suspendedFilters;
|
||||
}
|
||||
|
||||
/**
|
||||
* Enables a filter from the collection.
|
||||
*
|
||||
@@ -105,6 +124,9 @@ class FilterCollection
|
||||
|
||||
$this->enabledFilters[$name] = new $filterClass($this->em);
|
||||
|
||||
// In case a suspended filter with the same name was forgotten
|
||||
unset($this->suspendedFilters[$name]);
|
||||
|
||||
// Keep the enabled filters sorted for the hash
|
||||
ksort($this->enabledFilters);
|
||||
|
||||
@@ -135,6 +157,54 @@ class FilterCollection
|
||||
return $filter;
|
||||
}
|
||||
|
||||
/**
|
||||
* Suspend a filter.
|
||||
*
|
||||
* @param string $name Name of the filter.
|
||||
*
|
||||
* @return SQLFilter The suspended filter.
|
||||
*
|
||||
* @throws InvalidArgumentException If the filter does not exist.
|
||||
*/
|
||||
public function suspend(string $name): SQLFilter
|
||||
{
|
||||
// Get the filter to return it
|
||||
$filter = $this->getFilter($name);
|
||||
|
||||
$this->suspendedFilters[$name] = $filter;
|
||||
unset($this->enabledFilters[$name]);
|
||||
|
||||
$this->setFiltersStateDirty();
|
||||
|
||||
return $filter;
|
||||
}
|
||||
|
||||
/**
|
||||
* Restore a disabled filter from the collection.
|
||||
*
|
||||
* @param string $name Name of the filter.
|
||||
*
|
||||
* @return SQLFilter The restored filter.
|
||||
*
|
||||
* @throws InvalidArgumentException If the filter does not exist.
|
||||
*/
|
||||
public function restore(string $name): SQLFilter
|
||||
{
|
||||
if (! $this->isSuspended($name)) {
|
||||
throw new InvalidArgumentException("Filter '" . $name . "' is not suspended.");
|
||||
}
|
||||
|
||||
$this->enabledFilters[$name] = $this->suspendedFilters[$name];
|
||||
unset($this->suspendedFilters[$name]);
|
||||
|
||||
// Keep the enabled filters sorted for the hash
|
||||
ksort($this->enabledFilters);
|
||||
|
||||
$this->setFiltersStateDirty();
|
||||
|
||||
return $this->enabledFilters[$name];
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets an enabled filter from the collection.
|
||||
*
|
||||
@@ -177,6 +247,18 @@ class FilterCollection
|
||||
return isset($this->enabledFilters[$name]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if a filter is suspended.
|
||||
*
|
||||
* @param string $name Name of the filter.
|
||||
*
|
||||
* @return bool True if the filter is suspended, false otherwise.
|
||||
*/
|
||||
public function isSuspended(string $name): bool
|
||||
{
|
||||
return isset($this->suspendedFilters[$name]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Checks if the filter collection is clean.
|
||||
*
|
||||
|
||||
@@ -17,6 +17,7 @@ use Symfony\Component\Console\Input\InputOption;
|
||||
use Symfony\Component\Console\Output\OutputInterface;
|
||||
use Symfony\Component\Console\Style\SymfonyStyle;
|
||||
|
||||
use function assert;
|
||||
use function get_debug_type;
|
||||
use function sprintf;
|
||||
|
||||
@@ -63,32 +64,46 @@ EOT
|
||||
{
|
||||
$ui = (new SymfonyStyle($input, $output))->getErrorStyle();
|
||||
|
||||
$em = $this->getEntityManager($input);
|
||||
$cache = $em->getConfiguration()->getQueryCache();
|
||||
$cacheDriver = $em->getConfiguration()->getQueryCacheImpl();
|
||||
$em = $this->getEntityManager($input);
|
||||
$cache = $em->getConfiguration()->getQueryCache();
|
||||
|
||||
if (! $cacheDriver) {
|
||||
throw new InvalidArgumentException('No Query cache driver is configured on given EntityManager.');
|
||||
}
|
||||
|
||||
if ($cacheDriver instanceof ApcCache || $cache instanceof ApcuAdapter) {
|
||||
if ($cache instanceof ApcuAdapter) {
|
||||
throw new LogicException('Cannot clear APCu Cache from Console, it\'s shared in the Webserver memory and not accessible from the CLI.');
|
||||
}
|
||||
|
||||
if ($cacheDriver instanceof XcacheCache) {
|
||||
throw new LogicException('Cannot clear XCache Cache from Console, it\'s shared in the Webserver memory and not accessible from the CLI.');
|
||||
}
|
||||
$cacheDriver = null;
|
||||
if (! $cache) {
|
||||
$cacheDriver = $em->getConfiguration()->getQueryCacheImpl();
|
||||
|
||||
if (! ($cacheDriver instanceof ClearableCache)) {
|
||||
throw new LogicException(sprintf(
|
||||
'Can only clear cache when ClearableCache interface is implemented, %s does not implement.',
|
||||
get_debug_type($cacheDriver)
|
||||
));
|
||||
if (! $cacheDriver) {
|
||||
throw new InvalidArgumentException('No Query cache driver is configured on given EntityManager.');
|
||||
}
|
||||
|
||||
if ($cacheDriver instanceof ApcCache) {
|
||||
throw new LogicException('Cannot clear APCu Cache from Console, it\'s shared in the Webserver memory and not accessible from the CLI.');
|
||||
}
|
||||
|
||||
if ($cacheDriver instanceof XcacheCache) {
|
||||
throw new LogicException('Cannot clear XCache Cache from Console, it\'s shared in the Webserver memory and not accessible from the CLI.');
|
||||
}
|
||||
|
||||
if (! ($cacheDriver instanceof ClearableCache)) {
|
||||
throw new LogicException(sprintf(
|
||||
'Can only clear cache when ClearableCache interface is implemented, %s does not implement.',
|
||||
get_debug_type($cacheDriver)
|
||||
));
|
||||
}
|
||||
}
|
||||
|
||||
$ui->comment('Clearing <info>all</info> Query cache entries');
|
||||
|
||||
$result = $cache ? $cache->clear() : $cacheDriver->deleteAll();
|
||||
if ($cache) {
|
||||
$result = $cache->clear();
|
||||
} else {
|
||||
assert($cacheDriver !== null);
|
||||
$result = $cacheDriver->deleteAll();
|
||||
}
|
||||
|
||||
$message = $result ? 'Successfully deleted cache entries.' : 'No cache entries were deleted.';
|
||||
|
||||
if ($input->getOption('flush') === true && ! $cache) {
|
||||
|
||||
@@ -9,7 +9,6 @@ use Doctrine\ORM\Event\OnFlushEventArgs;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use ReflectionObject;
|
||||
|
||||
use function count;
|
||||
@@ -87,7 +86,7 @@ class DebugUnitOfWorkListener
|
||||
if ($value === null) {
|
||||
fwrite($fh, " NULL\n");
|
||||
} else {
|
||||
if ($value instanceof Proxy && ! $value->__isInitialized()) {
|
||||
if ($uow->isUninitializedObject($value)) {
|
||||
fwrite($fh, '[PROXY] ');
|
||||
}
|
||||
|
||||
|
||||
@@ -969,7 +969,7 @@ public function __construct(<params>)
|
||||
{
|
||||
$refl = new ReflectionClass($this->getClassToExtend());
|
||||
|
||||
return '\\' . $refl->getName();
|
||||
return '\\' . $refl->name;
|
||||
}
|
||||
|
||||
/** @return string */
|
||||
@@ -1376,7 +1376,7 @@ public function __construct(<params>)
|
||||
$this->staticReflection[$metadata->name]['methods'][] = strtolower($methodName);
|
||||
|
||||
$var = sprintf('%sMethodTemplate', $type);
|
||||
$template = static::$$var;
|
||||
$template = (string) static::$$var;
|
||||
|
||||
$methodTypeHint = '';
|
||||
$types = Type::getTypesMap();
|
||||
@@ -1695,7 +1695,7 @@ public function __construct(<params>)
|
||||
}
|
||||
|
||||
if (isset($fieldMapping['options']['comment']) && $fieldMapping['options']['comment']) {
|
||||
$options[] = '"comment"="' . str_replace('"', '""', $fieldMapping['options']['comment']) . '"';
|
||||
$options[] = '"comment"="' . str_replace('"', '""', (string) $fieldMapping['options']['comment']) . '"';
|
||||
}
|
||||
|
||||
if (isset($fieldMapping['options']['collation']) && $fieldMapping['options']['collation']) {
|
||||
|
||||
@@ -404,7 +404,7 @@ class LimitSubqueryOutputWalker extends SqlWalker
|
||||
|
||||
/**
|
||||
* @return string[][]
|
||||
* @psalm-return array{0: list<string>, 1: list<string>}
|
||||
* @psalm-return array{0: list<non-empty-string>, 1: list<string>}
|
||||
*/
|
||||
private function generateSqlAliasReplacements(): array
|
||||
{
|
||||
|
||||
@@ -821,8 +821,8 @@ class SchemaTool
|
||||
return [];
|
||||
}
|
||||
|
||||
$options = array_intersect_key($mappingOptions, array_flip(self::KNOWN_COLUMN_OPTIONS));
|
||||
$options['customSchemaOptions'] = array_diff_key($mappingOptions, $options);
|
||||
$options = array_intersect_key($mappingOptions, array_flip(self::KNOWN_COLUMN_OPTIONS));
|
||||
$options['platformOptions'] = array_diff_key($mappingOptions, $options);
|
||||
|
||||
return $options;
|
||||
}
|
||||
|
||||
@@ -23,11 +23,13 @@ use Doctrine\ORM\Event\PreFlushEventArgs;
|
||||
use Doctrine\ORM\Event\PrePersistEventArgs;
|
||||
use Doctrine\ORM\Event\PreRemoveEventArgs;
|
||||
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
||||
use Doctrine\ORM\Exception\EntityIdentityCollisionException;
|
||||
use Doctrine\ORM\Exception\ORMException;
|
||||
use Doctrine\ORM\Exception\UnexpectedAssociationValue;
|
||||
use Doctrine\ORM\Id\AssignedGenerator;
|
||||
use Doctrine\ORM\Internal\CommitOrderCalculator;
|
||||
use Doctrine\ORM\Internal\HydrationCompleteHandler;
|
||||
use Doctrine\ORM\Internal\TopologicalSort;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\Mapping\MappingException;
|
||||
use Doctrine\ORM\Mapping\Reflection\ReflectionPropertiesGetter;
|
||||
@@ -38,12 +40,12 @@ use Doctrine\ORM\Persisters\Entity\BasicEntityPersister;
|
||||
use Doctrine\ORM\Persisters\Entity\EntityPersister;
|
||||
use Doctrine\ORM\Persisters\Entity\JoinedSubclassPersister;
|
||||
use Doctrine\ORM\Persisters\Entity\SingleTablePersister;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\ORM\Utility\IdentifierFlattener;
|
||||
use Doctrine\Persistence\Mapping\RuntimeReflectionService;
|
||||
use Doctrine\Persistence\NotifyPropertyChanged;
|
||||
use Doctrine\Persistence\ObjectManagerAware;
|
||||
use Doctrine\Persistence\PropertyChangedListener;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Exception;
|
||||
use InvalidArgumentException;
|
||||
use RuntimeException;
|
||||
@@ -56,11 +58,9 @@ use function array_filter;
|
||||
use function array_key_exists;
|
||||
use function array_map;
|
||||
use function array_merge;
|
||||
use function array_pop;
|
||||
use function array_sum;
|
||||
use function array_values;
|
||||
use function assert;
|
||||
use function count;
|
||||
use function current;
|
||||
use function func_get_arg;
|
||||
use function func_num_args;
|
||||
@@ -74,6 +74,7 @@ use function method_exists;
|
||||
use function reset;
|
||||
use function spl_object_id;
|
||||
use function sprintf;
|
||||
use function strtolower;
|
||||
|
||||
/**
|
||||
* The UnitOfWork is responsible for tracking changes to objects during an
|
||||
@@ -419,9 +420,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||
|
||||
$this->dispatchOnFlushEvent();
|
||||
|
||||
// Now we need a commit order to maintain referential integrity
|
||||
$commitOrder = $this->getCommitOrder();
|
||||
|
||||
$conn = $this->em->getConnection();
|
||||
$conn->beginTransaction();
|
||||
|
||||
@@ -437,32 +435,37 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
if ($this->entityInsertions) {
|
||||
foreach ($commitOrder as $class) {
|
||||
$this->executeInserts($class);
|
||||
}
|
||||
// Perform entity insertions first, so that all new entities have their rows in the database
|
||||
// and can be referred to by foreign keys. The commit order only needs to take new entities
|
||||
// into account (new entities referring to other new entities), since all other types (entities
|
||||
// with updates or scheduled deletions) are currently not a problem, since they are already
|
||||
// in the database.
|
||||
$this->executeInserts();
|
||||
}
|
||||
|
||||
if ($this->entityUpdates) {
|
||||
foreach ($commitOrder as $class) {
|
||||
$this->executeUpdates($class);
|
||||
}
|
||||
// Updates do not need to follow a particular order
|
||||
$this->executeUpdates();
|
||||
}
|
||||
|
||||
// Extra updates that were requested by persisters.
|
||||
// This may include foreign keys that could not be set when an entity was inserted,
|
||||
// which may happen in the case of circular foreign key relationships.
|
||||
if ($this->extraUpdates) {
|
||||
$this->executeExtraUpdates();
|
||||
}
|
||||
|
||||
// Collection updates (deleteRows, updateRows, insertRows)
|
||||
// No particular order is necessary, since all entities themselves are already
|
||||
// in the database
|
||||
foreach ($this->collectionUpdates as $collectionToUpdate) {
|
||||
$this->getCollectionPersister($collectionToUpdate->getMapping())->update($collectionToUpdate);
|
||||
}
|
||||
|
||||
// Entity deletions come last and need to be in reverse commit order
|
||||
// Entity deletions come last. Their order only needs to take care of other deletions
|
||||
// (first delete entities depending upon others, before deleting depended-upon entities).
|
||||
if ($this->entityDeletions) {
|
||||
for ($count = count($commitOrder), $i = $count - 1; $i >= 0 && $this->entityDeletions; --$i) {
|
||||
$this->executeDeletions($commitOrder[$i]);
|
||||
}
|
||||
$this->executeDeletions();
|
||||
}
|
||||
|
||||
// Commit failed silently
|
||||
@@ -581,7 +584,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
// Ignore uninitialized proxy objects
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($entity)) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -690,6 +693,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
if ($class->isCollectionValuedAssociation($name) && $value !== null) {
|
||||
if ($value instanceof PersistentCollection) {
|
||||
if ($value->getOwner() === $entity) {
|
||||
$actualData[$name] = $value;
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -906,7 +910,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
|
||||
foreach ($entitiesToProcess as $entity) {
|
||||
// Ignore uninitialized proxy objects
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($entity)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
@@ -931,7 +935,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
*/
|
||||
private function computeAssociationChanges(array $assoc, $value): void
|
||||
{
|
||||
if ($value instanceof Proxy && ! $value->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($value)) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -1156,64 +1160,60 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes all entity insertions for entities of the specified type.
|
||||
* Executes entity insertions
|
||||
*/
|
||||
private function executeInserts(ClassMetadata $class): void
|
||||
private function executeInserts(): void
|
||||
{
|
||||
$entities = [];
|
||||
$className = $class->name;
|
||||
$persister = $this->getEntityPersister($className);
|
||||
$invoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postPersist);
|
||||
$entities = $this->computeInsertExecutionOrder();
|
||||
$eventsToDispatch = [];
|
||||
|
||||
$insertionsForClass = [];
|
||||
|
||||
foreach ($this->entityInsertions as $oid => $entity) {
|
||||
if ($this->em->getClassMetadata(get_class($entity))->name !== $className) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$insertionsForClass[$oid] = $entity;
|
||||
foreach ($entities as $entity) {
|
||||
$oid = spl_object_id($entity);
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
$persister = $this->getEntityPersister($class->name);
|
||||
|
||||
$persister->addInsert($entity);
|
||||
|
||||
unset($this->entityInsertions[$oid]);
|
||||
|
||||
if ($invoke !== ListenersInvoker::INVOKE_NONE) {
|
||||
$entities[] = $entity;
|
||||
}
|
||||
}
|
||||
$postInsertIds = $persister->executeInserts();
|
||||
|
||||
$postInsertIds = $persister->executeInserts();
|
||||
if (is_array($postInsertIds)) {
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10743/',
|
||||
'Returning post insert IDs from \Doctrine\ORM\Persisters\Entity\EntityPersister::executeInserts() is deprecated and will not be supported in Doctrine ORM 3.0. Make the persister call Doctrine\ORM\UnitOfWork::assignPostInsertId() instead.'
|
||||
);
|
||||
|
||||
if ($postInsertIds) {
|
||||
// Persister returned post-insert IDs
|
||||
foreach ($postInsertIds as $postInsertId) {
|
||||
$idField = $class->getSingleIdentifierFieldName();
|
||||
$idValue = $this->convertSingleFieldIdentifierToPHPValue($class, $postInsertId['generatedId']);
|
||||
|
||||
$entity = $postInsertId['entity'];
|
||||
$oid = spl_object_id($entity);
|
||||
|
||||
$class->reflFields[$idField]->setValue($entity, $idValue);
|
||||
|
||||
$this->entityIdentifiers[$oid] = [$idField => $idValue];
|
||||
$this->entityStates[$oid] = self::STATE_MANAGED;
|
||||
$this->originalEntityData[$oid][$idField] = $idValue;
|
||||
|
||||
$this->addToIdentityMap($entity);
|
||||
}
|
||||
} else {
|
||||
foreach ($insertionsForClass as $oid => $entity) {
|
||||
if (! isset($this->entityIdentifiers[$oid])) {
|
||||
//entity was not added to identity map because some identifiers are foreign keys to new entities.
|
||||
//add it now
|
||||
$this->addToEntityIdentifiersAndEntityMap($class, $oid, $entity);
|
||||
// Persister returned post-insert IDs
|
||||
foreach ($postInsertIds as $postInsertId) {
|
||||
$this->assignPostInsertId($postInsertId['entity'], $postInsertId['generatedId']);
|
||||
}
|
||||
}
|
||||
|
||||
if (! isset($this->entityIdentifiers[$oid])) {
|
||||
//entity was not added to identity map because some identifiers are foreign keys to new entities.
|
||||
//add it now
|
||||
$this->addToEntityIdentifiersAndEntityMap($class, $oid, $entity);
|
||||
}
|
||||
|
||||
$invoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postPersist);
|
||||
|
||||
if ($invoke !== ListenersInvoker::INVOKE_NONE) {
|
||||
$eventsToDispatch[] = ['class' => $class, 'entity' => $entity, 'invoke' => $invoke];
|
||||
}
|
||||
}
|
||||
|
||||
foreach ($entities as $entity) {
|
||||
$this->listenersInvoker->invoke($class, Events::postPersist, $entity, new PostPersistEventArgs($entity, $this->em), $invoke);
|
||||
// Defer dispatching `postPersist` events to until all entities have been inserted and post-insert
|
||||
// IDs have been assigned.
|
||||
foreach ($eventsToDispatch as $event) {
|
||||
$this->listenersInvoker->invoke(
|
||||
$event['class'],
|
||||
Events::postPersist,
|
||||
$event['entity'],
|
||||
new PostPersistEventArgs($event['entity'], $this->em),
|
||||
$event['invoke']
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1251,19 +1251,15 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes all entity updates for entities of the specified type.
|
||||
* Executes all entity updates
|
||||
*/
|
||||
private function executeUpdates(ClassMetadata $class): void
|
||||
private function executeUpdates(): void
|
||||
{
|
||||
$className = $class->name;
|
||||
$persister = $this->getEntityPersister($className);
|
||||
$preUpdateInvoke = $this->listenersInvoker->getSubscribedSystems($class, Events::preUpdate);
|
||||
$postUpdateInvoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postUpdate);
|
||||
|
||||
foreach ($this->entityUpdates as $oid => $entity) {
|
||||
if ($this->em->getClassMetadata(get_class($entity))->name !== $className) {
|
||||
continue;
|
||||
}
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
$persister = $this->getEntityPersister($class->name);
|
||||
$preUpdateInvoke = $this->listenersInvoker->getSubscribedSystems($class, Events::preUpdate);
|
||||
$postUpdateInvoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postUpdate);
|
||||
|
||||
if ($preUpdateInvoke !== ListenersInvoker::INVOKE_NONE) {
|
||||
$this->listenersInvoker->invoke($class, Events::preUpdate, $entity, new PreUpdateEventArgs($entity, $this->em, $this->getEntityChangeSet($entity)), $preUpdateInvoke);
|
||||
@@ -1284,18 +1280,18 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes all entity deletions for entities of the specified type.
|
||||
* Executes all entity deletions
|
||||
*/
|
||||
private function executeDeletions(ClassMetadata $class): void
|
||||
private function executeDeletions(): void
|
||||
{
|
||||
$className = $class->name;
|
||||
$persister = $this->getEntityPersister($className);
|
||||
$invoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postRemove);
|
||||
$entities = $this->computeDeleteExecutionOrder();
|
||||
$eventsToDispatch = [];
|
||||
|
||||
foreach ($this->entityDeletions as $oid => $entity) {
|
||||
if ($this->em->getClassMetadata(get_class($entity))->name !== $className) {
|
||||
continue;
|
||||
}
|
||||
foreach ($entities as $entity) {
|
||||
$oid = spl_object_id($entity);
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
$persister = $this->getEntityPersister($class->name);
|
||||
$invoke = $this->listenersInvoker->getSubscribedSystems($class, Events::postRemove);
|
||||
|
||||
$persister->delete($entity);
|
||||
|
||||
@@ -1314,78 +1310,132 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
if ($invoke !== ListenersInvoker::INVOKE_NONE) {
|
||||
$this->listenersInvoker->invoke($class, Events::postRemove, $entity, new PostRemoveEventArgs($entity, $this->em), $invoke);
|
||||
$eventsToDispatch[] = ['class' => $class, 'entity' => $entity, 'invoke' => $invoke];
|
||||
}
|
||||
}
|
||||
|
||||
// Defer dispatching `postRemove` events to until all entities have been removed.
|
||||
foreach ($eventsToDispatch as $event) {
|
||||
$this->listenersInvoker->invoke(
|
||||
$event['class'],
|
||||
Events::postRemove,
|
||||
$event['entity'],
|
||||
new PostRemoveEventArgs($event['entity'], $this->em),
|
||||
$event['invoke']
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the commit order.
|
||||
*
|
||||
* @return list<ClassMetadata>
|
||||
*/
|
||||
private function getCommitOrder(): array
|
||||
/** @return list<object> */
|
||||
private function computeInsertExecutionOrder(): array
|
||||
{
|
||||
$calc = $this->getCommitOrderCalculator();
|
||||
$sort = new TopologicalSort();
|
||||
|
||||
// See if there are any new classes in the changeset, that are not in the
|
||||
// commit order graph yet (don't have a node).
|
||||
// We have to inspect changeSet to be able to correctly build dependencies.
|
||||
// It is not possible to use IdentityMap here because post inserted ids
|
||||
// are not yet available.
|
||||
$newNodes = [];
|
||||
|
||||
foreach (array_merge($this->entityInsertions, $this->entityUpdates, $this->entityDeletions) as $entity) {
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
|
||||
if ($calc->hasNode($class->name)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$calc->addNode($class->name, $class);
|
||||
|
||||
$newNodes[] = $class;
|
||||
// First make sure we have all the nodes
|
||||
foreach ($this->entityInsertions as $entity) {
|
||||
$sort->addNode($entity);
|
||||
}
|
||||
|
||||
// Calculate dependencies for new nodes
|
||||
while ($class = array_pop($newNodes)) {
|
||||
// Now add edges
|
||||
foreach ($this->entityInsertions as $entity) {
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
|
||||
foreach ($class->associationMappings as $assoc) {
|
||||
// We only need to consider the owning sides of to-one associations,
|
||||
// since many-to-many associations are persisted at a later step and
|
||||
// have no insertion order problems (all entities already in the database
|
||||
// at that time).
|
||||
if (! ($assoc['isOwningSide'] && $assoc['type'] & ClassMetadata::TO_ONE)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$targetClass = $this->em->getClassMetadata($assoc['targetEntity']);
|
||||
$targetEntity = $class->getFieldValue($entity, $assoc['fieldName']);
|
||||
|
||||
if (! $calc->hasNode($targetClass->name)) {
|
||||
$calc->addNode($targetClass->name, $targetClass);
|
||||
|
||||
$newNodes[] = $targetClass;
|
||||
}
|
||||
|
||||
$joinColumns = reset($assoc['joinColumns']);
|
||||
|
||||
$calc->addDependency($targetClass->name, $class->name, (int) empty($joinColumns['nullable']));
|
||||
|
||||
// If the target class has mapped subclasses, these share the same dependency.
|
||||
if (! $targetClass->subClasses) {
|
||||
// If there is no entity that we need to refer to, or it is already in the
|
||||
// database (i. e. does not have to be inserted), no need to consider it.
|
||||
if ($targetEntity === null || ! $sort->hasNode($targetEntity)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
foreach ($targetClass->subClasses as $subClassName) {
|
||||
$targetSubClass = $this->em->getClassMetadata($subClassName);
|
||||
|
||||
if (! $calc->hasNode($subClassName)) {
|
||||
$calc->addNode($targetSubClass->name, $targetSubClass);
|
||||
|
||||
$newNodes[] = $targetSubClass;
|
||||
}
|
||||
|
||||
$calc->addDependency($targetSubClass->name, $class->name, 1);
|
||||
// An entity that references back to itself _and_ uses an application-provided ID
|
||||
// (the "NONE" generator strategy) can be exempted from commit order computation.
|
||||
// See https://github.com/doctrine/orm/pull/10735/ for more details on this edge case.
|
||||
// A non-NULLable self-reference would be a cycle in the graph.
|
||||
if ($targetEntity === $entity && $class->isIdentifierNatural()) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// According to https://www.doctrine-project.org/projects/doctrine-orm/en/2.14/reference/annotations-reference.html#annref_joincolumn,
|
||||
// the default for "nullable" is true. Unfortunately, it seems this default is not applied at the metadata driver, factory or other
|
||||
// level, but in fact we may have an undefined 'nullable' key here, so we must assume that default here as well.
|
||||
//
|
||||
// Same in \Doctrine\ORM\Tools\EntityGenerator::isAssociationIsNullable or \Doctrine\ORM\Persisters\Entity\BasicEntityPersister::getJoinSQLForJoinColumns,
|
||||
// to give two examples.
|
||||
assert(isset($assoc['joinColumns']));
|
||||
$joinColumns = reset($assoc['joinColumns']);
|
||||
$isNullable = ! isset($joinColumns['nullable']) || $joinColumns['nullable'];
|
||||
|
||||
// Add dependency. The dependency direction implies that "$targetEntity has to go before $entity",
|
||||
// so we can work through the topo sort result from left to right (with all edges pointing right).
|
||||
$sort->addEdge($targetEntity, $entity, $isNullable);
|
||||
}
|
||||
}
|
||||
|
||||
return $calc->sort();
|
||||
return $sort->sort();
|
||||
}
|
||||
|
||||
/** @return list<object> */
|
||||
private function computeDeleteExecutionOrder(): array
|
||||
{
|
||||
$sort = new TopologicalSort();
|
||||
|
||||
// First make sure we have all the nodes
|
||||
foreach ($this->entityDeletions as $entity) {
|
||||
$sort->addNode($entity);
|
||||
}
|
||||
|
||||
// Now add edges
|
||||
foreach ($this->entityDeletions as $entity) {
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
|
||||
foreach ($class->associationMappings as $assoc) {
|
||||
// We only need to consider the owning sides of to-one associations,
|
||||
// since many-to-many associations can always be (and have already been)
|
||||
// deleted in a preceding step.
|
||||
if (! ($assoc['isOwningSide'] && $assoc['type'] & ClassMetadata::TO_ONE)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// For associations that implement a database-level cascade/set null operation,
|
||||
// we do not have to follow a particular order: If the referred-to entity is
|
||||
// deleted first, the DBMS will either delete the current $entity right away
|
||||
// (CASCADE) or temporarily set the foreign key to NULL (SET NULL).
|
||||
// Either way, we can skip it in the computation.
|
||||
assert(isset($assoc['joinColumns']));
|
||||
$joinColumns = reset($assoc['joinColumns']);
|
||||
if (isset($joinColumns['onDelete'])) {
|
||||
$onDeleteOption = strtolower($joinColumns['onDelete']);
|
||||
if ($onDeleteOption === 'cascade' || $onDeleteOption === 'set null') {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
$targetEntity = $class->getFieldValue($entity, $assoc['fieldName']);
|
||||
|
||||
// If the association does not refer to another entity or that entity
|
||||
// is not to be deleted, there is no ordering problem and we can
|
||||
// skip this particular association.
|
||||
if ($targetEntity === null || ! $sort->hasNode($targetEntity)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Add dependency. The dependency direction implies that "$entity has to be removed before $targetEntity",
|
||||
// so we can work through the topo sort result from left to right (with all edges pointing right).
|
||||
$sort->addEdge($entity, $targetEntity, false);
|
||||
}
|
||||
}
|
||||
|
||||
return $sort->sort();
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -1601,6 +1651,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
* the entity in question is already managed.
|
||||
*
|
||||
* @throws ORMInvalidArgumentException
|
||||
* @throws EntityIdentityCollisionException
|
||||
*
|
||||
* @ignore
|
||||
*/
|
||||
@@ -1611,6 +1662,41 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$className = $classMetadata->rootEntityName;
|
||||
|
||||
if (isset($this->identityMap[$className][$idHash])) {
|
||||
if ($this->identityMap[$className][$idHash] !== $entity) {
|
||||
if ($this->em->getConfiguration()->isRejectIdCollisionInIdentityMapEnabled()) {
|
||||
throw EntityIdentityCollisionException::create($this->identityMap[$className][$idHash], $entity, $idHash);
|
||||
}
|
||||
|
||||
Deprecation::trigger(
|
||||
'doctrine/orm',
|
||||
'https://github.com/doctrine/orm/pull/10785',
|
||||
<<<'EXCEPTION'
|
||||
While adding an entity of class %s with an ID hash of "%s" to the identity map,
|
||||
another object of class %s was already present for the same ID. This will trigger
|
||||
an exception in ORM 3.0.
|
||||
|
||||
IDs should uniquely map to entity object instances. This problem may occur if:
|
||||
|
||||
- you use application-provided IDs and reuse ID values;
|
||||
- database-provided IDs are reassigned after truncating the database without
|
||||
clearing the EntityManager;
|
||||
- you might have been using EntityManager#getReference() to create a reference
|
||||
for a nonexistent ID that was subsequently (by the RDBMS) assigned to another
|
||||
entity.
|
||||
|
||||
Otherwise, it might be an ORM-internal inconsistency, please report it.
|
||||
|
||||
To opt-in to the new exception, call
|
||||
\Doctrine\ORM\Configuration::setRejectIdCollisionInIdentityMap on the entity
|
||||
manager's configuration.
|
||||
EXCEPTION
|
||||
,
|
||||
get_class($entity),
|
||||
$idHash,
|
||||
get_class($this->identityMap[$className][$idHash])
|
||||
);
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
@@ -2154,7 +2240,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$entity,
|
||||
$managedCopy
|
||||
): void {
|
||||
if (! ($class->isVersioned && $this->isLoaded($managedCopy) && $this->isLoaded($entity))) {
|
||||
if (! ($class->isVersioned && ! $this->isUninitializedObject($managedCopy) && ! $this->isUninitializedObject($entity))) {
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -2172,16 +2258,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||
throw OptimisticLockException::lockFailedVersionMismatch($entity, $entityVersion, $managedCopyVersion);
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests if an entity is loaded - must either be a loaded proxy or not a proxy
|
||||
*
|
||||
* @param object $entity
|
||||
*/
|
||||
private function isLoaded($entity): bool
|
||||
{
|
||||
return ! ($entity instanceof Proxy) || $entity->__isInitialized();
|
||||
}
|
||||
|
||||
/**
|
||||
* Sets/adds associated managed copies into the previous entity's association field
|
||||
*
|
||||
@@ -2477,7 +2553,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
*/
|
||||
private function cascadePersist($entity, array &$visited): void
|
||||
{
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($entity)) {
|
||||
// nothing to do - proxy is not initialized, therefore we don't do anything with it
|
||||
return;
|
||||
}
|
||||
@@ -2551,13 +2627,13 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
);
|
||||
|
||||
if ($associationMappings) {
|
||||
$this->initializeObject($entity);
|
||||
}
|
||||
|
||||
$entitiesToCascade = [];
|
||||
|
||||
foreach ($associationMappings as $assoc) {
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
$entity->__load();
|
||||
}
|
||||
|
||||
$relatedEntities = $class->reflFields[$assoc['fieldName']]->getValue($entity);
|
||||
|
||||
switch (true) {
|
||||
@@ -2613,9 +2689,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
return;
|
||||
}
|
||||
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
$entity->__load();
|
||||
}
|
||||
$this->initializeObject($entity);
|
||||
|
||||
assert($class->versionField !== null);
|
||||
$entityVersion = $class->reflFields[$class->versionField]->getValue($entity);
|
||||
@@ -2805,7 +2879,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$unmanagedProxy = $hints[Query::HINT_REFRESH_ENTITY];
|
||||
if (
|
||||
$unmanagedProxy !== $entity
|
||||
&& $unmanagedProxy instanceof Proxy
|
||||
&& $this->isIdentifierEquals($unmanagedProxy, $entity)
|
||||
) {
|
||||
// We will hydrate the given un-managed proxy anyway:
|
||||
@@ -2814,7 +2887,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
}
|
||||
|
||||
if ($entity instanceof Proxy && ! $entity->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($entity)) {
|
||||
$entity->__setInitialized(true);
|
||||
} else {
|
||||
if (
|
||||
@@ -2831,25 +2904,20 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
|
||||
$this->originalEntityData[$oid] = $data;
|
||||
|
||||
if ($entity instanceof NotifyPropertyChanged) {
|
||||
$entity->addPropertyChangedListener($this);
|
||||
}
|
||||
} else {
|
||||
$entity = $this->newInstance($class);
|
||||
$oid = spl_object_id($entity);
|
||||
|
||||
$this->entityIdentifiers[$oid] = $id;
|
||||
$this->entityStates[$oid] = self::STATE_MANAGED;
|
||||
$this->originalEntityData[$oid] = $data;
|
||||
|
||||
$this->identityMap[$class->rootEntityName][$idHash] = $entity;
|
||||
$this->registerManaged($entity, $id, $data);
|
||||
|
||||
if (isset($hints[Query::HINT_READ_ONLY])) {
|
||||
$this->readOnlyObjects[$oid] = true;
|
||||
}
|
||||
}
|
||||
|
||||
if ($entity instanceof NotifyPropertyChanged) {
|
||||
$entity->addPropertyChangedListener($this);
|
||||
}
|
||||
|
||||
foreach ($data as $field => $value) {
|
||||
if (isset($class->fieldMappings[$field])) {
|
||||
$class->reflFields[$field]->setValue($entity, $value);
|
||||
@@ -2965,8 +3033,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$hints['fetchMode'][$class->name][$field] === ClassMetadata::FETCH_EAGER &&
|
||||
isset($hints[self::HINT_DEFEREAGERLOAD]) &&
|
||||
! $targetClass->isIdentifierComposite &&
|
||||
$newValue instanceof Proxy &&
|
||||
$newValue->__isInitialized() === false
|
||||
$this->isUninitializedObject($newValue)
|
||||
) {
|
||||
$this->eagerLoadingEntities[$targetClass->rootEntityName][$relatedIdHash] = current($associatedId);
|
||||
}
|
||||
@@ -2987,6 +3054,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
// We are negating the condition here. Other cases will assume it is valid!
|
||||
case $hints['fetchMode'][$class->name][$field] !== ClassMetadata::FETCH_EAGER:
|
||||
$newValue = $this->em->getProxyFactory()->getProxy($assoc['targetEntity'], $normalizedAssociatedId);
|
||||
$this->registerManaged($newValue, $associatedId, []);
|
||||
break;
|
||||
|
||||
// Deferred eager load only works for single identifier classes
|
||||
@@ -2995,6 +3063,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$this->eagerLoadingEntities[$targetClass->rootEntityName][$relatedIdHash] = current($normalizedAssociatedId);
|
||||
|
||||
$newValue = $this->em->getProxyFactory()->getProxy($assoc['targetEntity'], $normalizedAssociatedId);
|
||||
$this->registerManaged($newValue, $associatedId, []);
|
||||
break;
|
||||
|
||||
default:
|
||||
@@ -3002,26 +3071,6 @@ class UnitOfWork implements PropertyChangedListener
|
||||
$newValue = $this->em->find($assoc['targetEntity'], $normalizedAssociatedId);
|
||||
break;
|
||||
}
|
||||
|
||||
if ($newValue === null) {
|
||||
break;
|
||||
}
|
||||
|
||||
// PERF: Inlined & optimized code from UnitOfWork#registerManaged()
|
||||
$newValueOid = spl_object_id($newValue);
|
||||
$this->entityIdentifiers[$newValueOid] = $associatedId;
|
||||
$this->identityMap[$targetClass->rootEntityName][$relatedIdHash] = $newValue;
|
||||
|
||||
if (
|
||||
$newValue instanceof NotifyPropertyChanged &&
|
||||
( ! $newValue instanceof Proxy || $newValue->__isInitialized())
|
||||
) {
|
||||
$newValue->addPropertyChangedListener($this);
|
||||
}
|
||||
|
||||
$this->entityStates[$newValueOid] = self::STATE_MANAGED;
|
||||
// make sure that when an proxy is then finally loaded, $this->originalEntityData is set also!
|
||||
break;
|
||||
}
|
||||
|
||||
$this->originalEntityData[$oid][$field] = $newValue;
|
||||
@@ -3377,7 +3426,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
|
||||
$this->addToIdentityMap($entity);
|
||||
|
||||
if ($entity instanceof NotifyPropertyChanged && ( ! $entity instanceof Proxy || $entity->__isInitialized())) {
|
||||
if ($entity instanceof NotifyPropertyChanged && ! $this->isUninitializedObject($entity)) {
|
||||
$entity->addPropertyChangedListener($this);
|
||||
}
|
||||
}
|
||||
@@ -3485,7 +3534,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
*/
|
||||
public function initializeObject($obj)
|
||||
{
|
||||
if ($obj instanceof Proxy) {
|
||||
if ($obj instanceof InternalProxy) {
|
||||
$obj->__load();
|
||||
|
||||
return;
|
||||
@@ -3496,6 +3545,18 @@ class UnitOfWork implements PropertyChangedListener
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Tests if a value is an uninitialized entity.
|
||||
*
|
||||
* @param mixed $obj
|
||||
*
|
||||
* @psalm-assert-if-true InternalProxy $obj
|
||||
*/
|
||||
public function isUninitializedObject($obj): bool
|
||||
{
|
||||
return $obj instanceof InternalProxy && ! $obj->__isInitialized();
|
||||
}
|
||||
|
||||
/**
|
||||
* Helper method to show an object as string.
|
||||
*
|
||||
@@ -3646,13 +3707,11 @@ class UnitOfWork implements PropertyChangedListener
|
||||
*/
|
||||
private function mergeEntityStateIntoManagedCopy($entity, $managedCopy): void
|
||||
{
|
||||
if (! $this->isLoaded($entity)) {
|
||||
if ($this->isUninitializedObject($entity)) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (! $this->isLoaded($managedCopy)) {
|
||||
$managedCopy->__load();
|
||||
}
|
||||
$this->initializeObject($managedCopy);
|
||||
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
|
||||
@@ -3673,7 +3732,7 @@ class UnitOfWork implements PropertyChangedListener
|
||||
if ($other === null) {
|
||||
$prop->setValue($managedCopy, null);
|
||||
} else {
|
||||
if ($other instanceof Proxy && ! $other->__isInitialized()) {
|
||||
if ($this->isUninitializedObject($other)) {
|
||||
// do not merge fields marked lazy that have not been fetched.
|
||||
continue;
|
||||
}
|
||||
@@ -3837,4 +3896,30 @@ class UnitOfWork implements PropertyChangedListener
|
||||
|
||||
return $normalizedAssociatedId;
|
||||
}
|
||||
|
||||
/**
|
||||
* Assign a post-insert generated ID to an entity
|
||||
*
|
||||
* This is used by EntityPersisters after they inserted entities into the database.
|
||||
* It will place the assigned ID values in the entity's fields and start tracking
|
||||
* the entity in the identity map.
|
||||
*
|
||||
* @param object $entity
|
||||
* @param mixed $generatedId
|
||||
*/
|
||||
final public function assignPostInsertId($entity, $generatedId): void
|
||||
{
|
||||
$class = $this->em->getClassMetadata(get_class($entity));
|
||||
$idField = $class->getSingleIdentifierFieldName();
|
||||
$idValue = $this->convertSingleFieldIdentifierToPHPValue($class, $generatedId);
|
||||
$oid = spl_object_id($entity);
|
||||
|
||||
$class->reflFields[$idField]->setValue($entity, $idValue);
|
||||
|
||||
$this->entityIdentifiers[$oid] = [$idField => $idValue];
|
||||
$this->entityStates[$oid] = self::STATE_MANAGED;
|
||||
$this->originalEntityData[$oid][$idField] = $idValue;
|
||||
|
||||
$this->addToIdentityMap($entity);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -34,10 +34,5 @@ parameters:
|
||||
count: 1
|
||||
path: lib/Doctrine/ORM/Tools/Console/Command/ClearCache/ResultCommand.php
|
||||
|
||||
-
|
||||
message: '/^Call to an undefined method Doctrine\\Persistence\\Proxy::__setInitialized\(\)\.$/'
|
||||
count: 1
|
||||
path: lib/Doctrine/ORM/UnitOfWork.php
|
||||
|
||||
# Symfony cache supports passing a key prefix to the clear method.
|
||||
- '/^Method Psr\\Cache\\CacheItemPoolInterface\:\:clear\(\) invoked with 1 parameter, 0 required\.$/'
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<files psalm-version="5.13.1@086b94371304750d1c673315321a55d15fc59015">
|
||||
<files psalm-version="5.14.1@b9d355e0829c397b9b3b47d0c0ed042a8a70284d">
|
||||
<file src="lib/Doctrine/ORM/AbstractQuery.php">
|
||||
<DeprecatedClass>
|
||||
<code>IterableResult</code>
|
||||
@@ -488,7 +488,6 @@
|
||||
<code>getValue</code>
|
||||
<code>setValue</code>
|
||||
<code>setValue</code>
|
||||
<code>setValue</code>
|
||||
</PossiblyNullReference>
|
||||
<PossiblyUndefinedArrayOffset>
|
||||
<code><![CDATA[$class->associationMappings[$class->identifier[0]]['joinColumns']]]></code>
|
||||
@@ -965,9 +964,6 @@
|
||||
<code><![CDATA[$joinColumnElement['options']->children()]]></code>
|
||||
<code><![CDATA[$option->children()]]></code>
|
||||
</PossiblyNullArgument>
|
||||
<PossiblyNullIterator>
|
||||
<code><![CDATA[$cascadeElement->children()]]></code>
|
||||
</PossiblyNullIterator>
|
||||
<TypeDoesNotContainType>
|
||||
<code><![CDATA[$xmlRoot->getName() === 'embeddable']]></code>
|
||||
<code><![CDATA[$xmlRoot->getName() === 'entity']]></code>
|
||||
@@ -1381,11 +1377,6 @@
|
||||
<code>$columnList</code>
|
||||
</PossiblyUndefinedVariable>
|
||||
</file>
|
||||
<file src="lib/Doctrine/ORM/Proxy/Proxy.php">
|
||||
<MissingTemplateParam>
|
||||
<code>BaseProxy</code>
|
||||
</MissingTemplateParam>
|
||||
</file>
|
||||
<file src="lib/Doctrine/ORM/Proxy/ProxyFactory.php">
|
||||
<ArgumentTypeCoercion>
|
||||
<code>$classMetadata</code>
|
||||
@@ -1393,7 +1384,7 @@
|
||||
<code>$classMetadata</code>
|
||||
</ArgumentTypeCoercion>
|
||||
<DirectConstructorCall>
|
||||
<code><![CDATA[$proxy->__construct(static function (Proxy $object) use ($initializer, $proxy): void {
|
||||
<code><![CDATA[$proxy->__construct(static function (InternalProxy $object) use ($initializer, $proxy): void {
|
||||
$initializer($object, $proxy);
|
||||
})]]></code>
|
||||
</DirectConstructorCall>
|
||||
@@ -2799,7 +2790,6 @@
|
||||
<code>setValue</code>
|
||||
</PossiblyNullReference>
|
||||
<PossiblyUndefinedArrayOffset>
|
||||
<code><![CDATA[$assoc['joinColumns']]]></code>
|
||||
<code><![CDATA[$assoc['orphanRemoval']]]></code>
|
||||
<code><![CDATA[$assoc['targetToSourceKeyColumns']]]></code>
|
||||
</PossiblyUndefinedArrayOffset>
|
||||
@@ -2808,10 +2798,6 @@
|
||||
<code>unwrap</code>
|
||||
<code>unwrap</code>
|
||||
</PossiblyUndefinedMethod>
|
||||
<RedundantCondition>
|
||||
<code><![CDATA[$i >= 0 && $this->entityDeletions]]></code>
|
||||
<code><![CDATA[$this->entityDeletions]]></code>
|
||||
</RedundantCondition>
|
||||
<RedundantConditionGivenDocblockType>
|
||||
<code>is_array($entity)</code>
|
||||
</RedundantConditionGivenDocblockType>
|
||||
|
||||
@@ -45,6 +45,10 @@
|
||||
<referencedClass name="Doctrine\ORM\Tools\Console\Command\GenerateRepositoriesCommand"/>
|
||||
<referencedClass name="Doctrine\ORM\Tools\Console\Helper\EntityManagerHelper"/>
|
||||
<referencedClass name="Doctrine\ORM\Tools\Console\EntityManagerProvider\HelperSetManagerProvider"/>
|
||||
<referencedClass name="Doctrine\ORM\Internal\CommitOrder\Edge"/>
|
||||
<referencedClass name="Doctrine\ORM\Internal\CommitOrder\Vertex"/>
|
||||
<referencedClass name="Doctrine\ORM\Internal\CommitOrder\VertexState"/>
|
||||
<referencedClass name="Doctrine\ORM\Internal\CommitOrderCalculator"/>
|
||||
</errorLevel>
|
||||
</DeprecatedClass>
|
||||
<DeprecatedConstant>
|
||||
|
||||
@@ -4,9 +4,9 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Performance\LazyLoading;
|
||||
|
||||
use Doctrine\ORM\Proxy\InternalProxy as Proxy;
|
||||
use Doctrine\Performance\EntityManagerFactory;
|
||||
use Doctrine\Performance\Mock\NonProxyLoadingEntityManager;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\CMS\CmsEmployee;
|
||||
use Doctrine\Tests\Models\CMS\CmsUser;
|
||||
|
||||
|
||||
@@ -21,6 +21,7 @@ abstract class DoctrineTestCase extends TestCase
|
||||
'assertMatchesRegularExpression' => 'assertRegExp', // can be removed when PHPUnit 9 is minimum
|
||||
'assertDoesNotMatchRegularExpression' => 'assertNotRegExp', // can be removed when PHPUnit 9 is minimum
|
||||
'assertFileDoesNotExist' => 'assertFileNotExists', // can be removed PHPUnit 9 is minimum
|
||||
'expectExceptionMessageMatches' => 'expectExceptionMessageRegExp', // can be removed when PHPUnit 8 is minimum
|
||||
];
|
||||
|
||||
/**
|
||||
|
||||
@@ -11,8 +11,6 @@ use Doctrine\ORM\Mapping\Column;
|
||||
use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\EntityResult;
|
||||
use Doctrine\ORM\Mapping\FieldResult;
|
||||
use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\JoinTable;
|
||||
use Doctrine\ORM\Mapping\ManyToMany;
|
||||
@@ -67,14 +65,6 @@ use Doctrine\ORM\Mapping\SqlResultSetMappings;
|
||||
#[ORM\Entity]
|
||||
class CompanyFlexContract extends CompanyContract
|
||||
{
|
||||
/**
|
||||
* @Id
|
||||
* @GeneratedValue
|
||||
* @Column(type="integer")
|
||||
* @var int
|
||||
*/
|
||||
public $id;
|
||||
|
||||
/**
|
||||
* @Column(type="integer")
|
||||
* @var int
|
||||
|
||||
44
tests/Doctrine/Tests/Models/Issue9300/Issue9300Child.php
Normal file
44
tests/Doctrine/Tests/Models/Issue9300/Issue9300Child.php
Normal file
@@ -0,0 +1,44 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\Models\Issue9300;
|
||||
|
||||
use Doctrine\Common\Collections\ArrayCollection;
|
||||
use Doctrine\Common\Collections\Collection;
|
||||
use Doctrine\ORM\Mapping\Column;
|
||||
use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\ManyToMany;
|
||||
|
||||
/**
|
||||
* @Entity
|
||||
*/
|
||||
class Issue9300Child
|
||||
{
|
||||
/**
|
||||
* @var int
|
||||
* @Id
|
||||
* @Column(type="integer")
|
||||
* @GeneratedValue
|
||||
*/
|
||||
public $id;
|
||||
|
||||
/**
|
||||
* @var Collection<int, Issue9300Parent>
|
||||
* @ManyToMany(targetEntity="Issue9300Parent")
|
||||
*/
|
||||
public $parents;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
* @Column(type="string")
|
||||
*/
|
||||
public $name;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->parents = new ArrayCollection();
|
||||
}
|
||||
}
|
||||
30
tests/Doctrine/Tests/Models/Issue9300/Issue9300Parent.php
Normal file
30
tests/Doctrine/Tests/Models/Issue9300/Issue9300Parent.php
Normal file
@@ -0,0 +1,30 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\Models\Issue9300;
|
||||
|
||||
use Doctrine\ORM\Mapping\Column;
|
||||
use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
|
||||
/**
|
||||
* @Entity
|
||||
*/
|
||||
class Issue9300Parent
|
||||
{
|
||||
/**
|
||||
* @var int
|
||||
* @Id
|
||||
* @Column(type="integer")
|
||||
* @GeneratedValue
|
||||
*/
|
||||
public $id;
|
||||
|
||||
/**
|
||||
* @var string
|
||||
* @Column(type="string")
|
||||
*/
|
||||
public $name;
|
||||
}
|
||||
@@ -101,7 +101,7 @@ class ConfigurationTest extends DoctrineTestCase
|
||||
$paths = [__DIR__];
|
||||
$reflectionClass = new ReflectionClass(ConfigurationTestAnnotationReaderChecker::class);
|
||||
|
||||
$annotationDriver = $this->configuration->newDefaultAnnotationDriver($paths, false);
|
||||
$annotationDriver = $this->configuration->newDefaultAnnotationDriver($paths, false, true);
|
||||
$reader = $annotationDriver->getReader();
|
||||
$annotation = $reader->getMethodAnnotation(
|
||||
$reflectionClass->getMethod('namespacedAnnotationMethod'),
|
||||
|
||||
@@ -5,16 +5,18 @@ declare(strict_types=1);
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\ORM\EntityNotFoundException;
|
||||
use Doctrine\ORM\Exception\EntityIdentityCollisionException;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\ORMInvalidArgumentException;
|
||||
use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\ORM\Query;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\IterableTester;
|
||||
use Doctrine\Tests\Models\CMS\CmsAddress;
|
||||
use Doctrine\Tests\Models\CMS\CmsArticle;
|
||||
use Doctrine\Tests\Models\CMS\CmsComment;
|
||||
use Doctrine\Tests\Models\CMS\CmsGroup;
|
||||
use Doctrine\Tests\Models\CMS\CmsPhonenumber;
|
||||
use Doctrine\Tests\Models\CMS\CmsUser;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -144,7 +146,7 @@ class BasicFunctionalTest extends OrmFunctionalTestCase
|
||||
|
||||
// Address has been eager-loaded because it cant be lazy
|
||||
self::assertInstanceOf(CmsAddress::class, $user2->address);
|
||||
self::assertNotInstanceOf(Proxy::class, $user2->address);
|
||||
self::assertFalse($this->isUninitializedObject($user2->address));
|
||||
}
|
||||
|
||||
/** @group DDC-1230 */
|
||||
@@ -515,42 +517,76 @@ class BasicFunctionalTest extends OrmFunctionalTestCase
|
||||
self::assertEquals(4, $gblanco2->getPhonenumbers()->count());
|
||||
}
|
||||
|
||||
public function testSetSetAssociationWithGetReference(): void
|
||||
public function testSetToOneAssociationWithGetReference(): void
|
||||
{
|
||||
$user = new CmsUser();
|
||||
$user->name = 'Guilherme';
|
||||
$user->username = 'gblanco';
|
||||
$user->status = 'developer';
|
||||
$this->_em->persist($user);
|
||||
|
||||
$address = new CmsAddress();
|
||||
$address->country = 'Germany';
|
||||
$address->city = 'Berlin';
|
||||
$address->zip = '12345';
|
||||
$this->_em->persist($address);
|
||||
|
||||
$this->_em->flush();
|
||||
$this->_em->clear(CmsAddress::class);
|
||||
|
||||
self::assertFalse($this->_em->contains($address));
|
||||
self::assertTrue($this->_em->contains($user));
|
||||
|
||||
// Assume we only got the identifier of the address and now want to attach
|
||||
// that address to the user without actually loading it, using getReference().
|
||||
$addressRef = $this->_em->getReference(CmsAddress::class, $address->getId());
|
||||
|
||||
$user->setAddress($addressRef); // Ugh! Initializes address 'cause of $address->setUser($user)!
|
||||
|
||||
$this->_em->flush();
|
||||
$this->_em->clear();
|
||||
|
||||
// Assume we only got the identifier of the user and now want to attach
|
||||
// the article to the user without actually loading it, using getReference().
|
||||
$userRef = $this->_em->getReference(CmsUser::class, $user->getId());
|
||||
self::assertTrue($this->isUninitializedObject($userRef));
|
||||
|
||||
$article = new CmsArticle();
|
||||
$article->topic = 'topic';
|
||||
$article->text = 'text';
|
||||
$article->setAuthor($userRef);
|
||||
|
||||
$this->_em->persist($article);
|
||||
$this->_em->flush();
|
||||
|
||||
self::assertFalse($userRef->__isInitialized());
|
||||
|
||||
$this->_em->clear();
|
||||
|
||||
// Check with a fresh load that the association is indeed there
|
||||
$query = $this->_em->createQuery("select u, a from Doctrine\Tests\Models\CMS\CmsUser u join u.address a where u.username='gblanco'");
|
||||
$query = $this->_em->createQuery("select u, a from Doctrine\Tests\Models\CMS\CmsUser u join u.articles a where u.username='gblanco'");
|
||||
$gblanco = $query->getSingleResult();
|
||||
|
||||
self::assertInstanceOf(CmsUser::class, $gblanco);
|
||||
self::assertInstanceOf(CmsAddress::class, $gblanco->getAddress());
|
||||
self::assertEquals('Berlin', $gblanco->getAddress()->getCity());
|
||||
self::assertInstanceOf(CmsArticle::class, $gblanco->articles[0]);
|
||||
self::assertSame($article->id, $gblanco->articles[0]->id);
|
||||
self::assertSame('text', $gblanco->articles[0]->text);
|
||||
}
|
||||
|
||||
public function testAddToToManyAssociationWithGetReference(): void
|
||||
{
|
||||
$group = new CmsGroup();
|
||||
$group->name = 'admins';
|
||||
$this->_em->persist($group);
|
||||
$this->_em->flush();
|
||||
$this->_em->clear();
|
||||
|
||||
// Assume we only got the identifier of the user and now want to attach
|
||||
// the article to the user without actually loading it, using getReference().
|
||||
$groupRef = $this->_em->getReference(CmsGroup::class, $group->id);
|
||||
self::assertTrue($this->isUninitializedObject($groupRef));
|
||||
|
||||
$user = new CmsUser();
|
||||
$user->name = 'Guilherme';
|
||||
$user->username = 'gblanco';
|
||||
$user->groups->add($groupRef);
|
||||
|
||||
$this->_em->persist($user);
|
||||
$this->_em->flush();
|
||||
|
||||
self::assertFalse($groupRef->__isInitialized());
|
||||
|
||||
$this->_em->clear();
|
||||
|
||||
// Check with a fresh load that the association is indeed there
|
||||
$query = $this->_em->createQuery("select u, a from Doctrine\Tests\Models\CMS\CmsUser u join u.groups a where u.username='gblanco'");
|
||||
$gblanco = $query->getSingleResult();
|
||||
|
||||
self::assertInstanceOf(CmsUser::class, $gblanco);
|
||||
self::assertInstanceOf(CmsGroup::class, $gblanco->groups[0]);
|
||||
self::assertSame($group->id, $gblanco->groups[0]->id);
|
||||
self::assertSame('admins', $gblanco->groups[0]->name);
|
||||
}
|
||||
|
||||
public function testOneToManyCascadeRemove(): void
|
||||
@@ -707,9 +743,8 @@ class BasicFunctionalTest extends OrmFunctionalTestCase
|
||||
->setParameter('user', $userRef)
|
||||
->getSingleResult();
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $address2->getUser());
|
||||
self::assertTrue($userRef === $address2->getUser());
|
||||
self::assertFalse($userRef->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($userRef));
|
||||
self::assertEquals('Germany', $address2->country);
|
||||
self::assertEquals('Berlin', $address2->city);
|
||||
self::assertEquals('12345', $address2->zip);
|
||||
@@ -1006,8 +1041,8 @@ class BasicFunctionalTest extends OrmFunctionalTestCase
|
||||
->setParameter(1, $article->id)
|
||||
->setFetchMode(CmsArticle::class, 'user', ClassMetadata::FETCH_EAGER)
|
||||
->getSingleResult();
|
||||
self::assertInstanceOf(Proxy::class, $article->user, 'It IS a proxy, ...');
|
||||
self::assertTrue($article->user->__isInitialized(), '...but its initialized!');
|
||||
self::assertInstanceOf(InternalProxy::class, $article->user, 'It IS a proxy, ...');
|
||||
self::assertFalse($this->isUninitializedObject($article->user), '...but its initialized!');
|
||||
$this->assertQueryCount(2);
|
||||
}
|
||||
|
||||
@@ -1291,4 +1326,35 @@ class BasicFunctionalTest extends OrmFunctionalTestCase
|
||||
|
||||
$this->_em->flush();
|
||||
}
|
||||
|
||||
public function testItThrowsWhenReferenceUsesIdAssignedByDatabase(): void
|
||||
{
|
||||
$this->_em->getConfiguration()->setRejectIdCollisionInIdentityMap(true);
|
||||
|
||||
$user = new CmsUser();
|
||||
$user->name = 'test';
|
||||
$user->username = 'test';
|
||||
$this->_em->persist($user);
|
||||
$this->_em->flush();
|
||||
|
||||
// Obtain a reference object for the next ID. This is a user error - references
|
||||
// should be fetched only for existing IDs
|
||||
$ref = $this->_em->getReference(CmsUser::class, $user->id + 1);
|
||||
|
||||
$user2 = new CmsUser();
|
||||
$user2->name = 'test2';
|
||||
$user2->username = 'test2';
|
||||
|
||||
// Now the database will assign an ID to the $user2 entity, but that place
|
||||
// in the identity map is already taken by user error.
|
||||
$this->expectException(EntityIdentityCollisionException::class);
|
||||
$this->expectExceptionMessageMatches('/another object .* was already present for the same ID/');
|
||||
|
||||
// depending on ID generation strategy, the ID may be asssigned already here
|
||||
// and the entity be put in the identity map
|
||||
$this->_em->persist($user2);
|
||||
|
||||
// post insert IDs will be assigned during flush
|
||||
$this->_em->flush();
|
||||
}
|
||||
}
|
||||
|
||||
@@ -7,7 +7,6 @@ namespace Doctrine\Tests\ORM\Functional;
|
||||
use DateTime;
|
||||
use Doctrine\Common\Collections\Criteria;
|
||||
use Doctrine\ORM\PersistentCollection;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\IterableTester;
|
||||
use Doctrine\Tests\Models\Company\CompanyAuction;
|
||||
use Doctrine\Tests\Models\Company\CompanyEmployee;
|
||||
@@ -300,7 +299,7 @@ class ClassTableInheritanceTest extends OrmFunctionalTestCase
|
||||
$mainEvent = $result[0]->getMainEvent();
|
||||
// mainEvent should have been loaded because it can't be lazy
|
||||
self::assertInstanceOf(CompanyAuction::class, $mainEvent);
|
||||
self::assertNotInstanceOf(Proxy::class, $mainEvent);
|
||||
self::assertFalse($this->isUninitializedObject($mainEvent));
|
||||
|
||||
$this->_em->clear();
|
||||
|
||||
@@ -432,13 +431,13 @@ class ClassTableInheritanceTest extends OrmFunctionalTestCase
|
||||
$this->_em->clear();
|
||||
|
||||
$ref = $this->_em->getReference(CompanyPerson::class, $manager->getId());
|
||||
self::assertNotInstanceOf(Proxy::class, $ref, 'Cannot Request a proxy from a class that has subclasses.');
|
||||
self::assertFalse($this->isUninitializedObject($ref), 'Cannot Request a proxy from a class that has subclasses.');
|
||||
self::assertInstanceOf(CompanyPerson::class, $ref);
|
||||
self::assertInstanceOf(CompanyEmployee::class, $ref, 'Direct fetch of the reference has to load the child class Employee directly.');
|
||||
$this->_em->clear();
|
||||
|
||||
$ref = $this->_em->getReference(CompanyManager::class, $manager->getId());
|
||||
self::assertInstanceOf(Proxy::class, $ref, 'A proxy can be generated only if no subclasses exists for the requested reference.');
|
||||
self::assertTrue($this->isUninitializedObject($ref), 'A proxy can be generated only if no subclasses exists for the requested reference.');
|
||||
}
|
||||
|
||||
/** @group DDC-992 */
|
||||
|
||||
@@ -43,7 +43,7 @@ class DefaultValuesTest extends OrmFunctionalTestCase
|
||||
$user2 = $this->_em->getReference(get_class($user), $userId);
|
||||
|
||||
$this->_em->flush();
|
||||
self::assertFalse($user2->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($user2));
|
||||
|
||||
$a = new DefaultValueAddress();
|
||||
$a->country = 'de';
|
||||
@@ -55,7 +55,7 @@ class DefaultValuesTest extends OrmFunctionalTestCase
|
||||
$this->_em->persist($a);
|
||||
$this->_em->flush();
|
||||
|
||||
self::assertFalse($user2->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($user2));
|
||||
$this->_em->clear();
|
||||
|
||||
$a2 = $this->_em->find(get_class($a), $a->id);
|
||||
|
||||
@@ -6,7 +6,6 @@ namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\DBAL\Exception\UniqueConstraintViolationException;
|
||||
use Doctrine\ORM\OptimisticLockException;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\CMS\CmsAddress;
|
||||
use Doctrine\Tests\Models\CMS\CmsArticle;
|
||||
use Doctrine\Tests\Models\CMS\CmsPhonenumber;
|
||||
@@ -150,16 +149,13 @@ class DetachedEntityTest extends OrmFunctionalTestCase
|
||||
$this->_em->clear();
|
||||
|
||||
$address2 = $this->_em->find(get_class($address), $address->id);
|
||||
self::assertInstanceOf(Proxy::class, $address2->user);
|
||||
self::assertFalse($address2->user->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($address2->user));
|
||||
$detachedAddress2 = unserialize(serialize($address2));
|
||||
self::assertInstanceOf(Proxy::class, $detachedAddress2->user);
|
||||
self::assertFalse($detachedAddress2->user->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($detachedAddress2->user));
|
||||
|
||||
$managedAddress2 = $this->_em->merge($detachedAddress2);
|
||||
self::assertInstanceOf(Proxy::class, $managedAddress2->user);
|
||||
self::assertFalse($managedAddress2->user === $detachedAddress2->user);
|
||||
self::assertFalse($managedAddress2->user->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($managedAddress2->user));
|
||||
}
|
||||
|
||||
/** @group DDC-822 */
|
||||
|
||||
@@ -4,12 +4,18 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\ORM\Event\PostPersistEventArgs;
|
||||
use Doctrine\ORM\Event\PostRemoveEventArgs;
|
||||
use Doctrine\ORM\Event\PreFlushEventArgs;
|
||||
use Doctrine\ORM\Event\PreUpdateEventArgs;
|
||||
use Doctrine\ORM\Events;
|
||||
use Doctrine\ORM\UnitOfWork;
|
||||
use Doctrine\Persistence\Event\LifecycleEventArgs;
|
||||
use Doctrine\Tests\Models\Company\CompanyContractListener;
|
||||
use Doctrine\Tests\Models\Company\CompanyFixContract;
|
||||
use Doctrine\Tests\Models\Company\CompanyPerson;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
use PHPUnit\Framework\Assert;
|
||||
|
||||
/** @group DDC-1955 */
|
||||
class EntityListenersTest extends OrmFunctionalTestCase
|
||||
@@ -96,6 +102,45 @@ class EntityListenersTest extends OrmFunctionalTestCase
|
||||
self::assertInstanceOf(LifecycleEventArgs::class, $this->listener->postPersistCalls[0][1]);
|
||||
}
|
||||
|
||||
public function testPostPersistCalledAfterAllInsertsHaveBeenPerformedAndIdsHaveBeenAssigned(): void
|
||||
{
|
||||
$object1 = new CompanyFixContract();
|
||||
$object1->setFixPrice(2000);
|
||||
|
||||
$object2 = new CompanyPerson();
|
||||
$object2->setName('J. Doe');
|
||||
|
||||
$this->_em->persist($object1);
|
||||
$this->_em->persist($object2);
|
||||
|
||||
$listener = new class ([$object1, $object2]) {
|
||||
/** @var array<object> */
|
||||
private $trackedObjects;
|
||||
|
||||
/** @var int */
|
||||
public $invocationCount = 0;
|
||||
|
||||
public function __construct(array $trackedObjects)
|
||||
{
|
||||
$this->trackedObjects = $trackedObjects;
|
||||
}
|
||||
|
||||
public function postPersist(PostPersistEventArgs $args): void
|
||||
{
|
||||
foreach ($this->trackedObjects as $object) {
|
||||
Assert::assertNotNull($object->getId());
|
||||
}
|
||||
|
||||
++$this->invocationCount;
|
||||
}
|
||||
};
|
||||
|
||||
$this->_em->getEventManager()->addEventListener(Events::postPersist, $listener);
|
||||
$this->_em->flush();
|
||||
|
||||
self::assertSame(2, $listener->invocationCount);
|
||||
}
|
||||
|
||||
public function testPreUpdateListeners(): void
|
||||
{
|
||||
$fix = new CompanyFixContract();
|
||||
@@ -175,4 +220,50 @@ class EntityListenersTest extends OrmFunctionalTestCase
|
||||
self::assertInstanceOf(CompanyFixContract::class, $this->listener->postRemoveCalls[0][0]);
|
||||
self::assertInstanceOf(LifecycleEventArgs::class, $this->listener->postRemoveCalls[0][1]);
|
||||
}
|
||||
|
||||
public function testPostRemoveCalledAfterAllRemovalsHaveBeenPerformed(): void
|
||||
{
|
||||
$object1 = new CompanyFixContract();
|
||||
$object1->setFixPrice(2000);
|
||||
|
||||
$object2 = new CompanyPerson();
|
||||
$object2->setName('J. Doe');
|
||||
|
||||
$this->_em->persist($object1);
|
||||
$this->_em->persist($object2);
|
||||
$this->_em->flush();
|
||||
|
||||
$listener = new class ($this->_em->getUnitOfWork(), [$object1, $object2]) {
|
||||
/** @var UnitOfWork */
|
||||
private $uow;
|
||||
|
||||
/** @var array<object> */
|
||||
private $trackedObjects;
|
||||
|
||||
/** @var int */
|
||||
public $invocationCount = 0;
|
||||
|
||||
public function __construct(UnitOfWork $uow, array $trackedObjects)
|
||||
{
|
||||
$this->uow = $uow;
|
||||
$this->trackedObjects = $trackedObjects;
|
||||
}
|
||||
|
||||
public function postRemove(PostRemoveEventArgs $args): void
|
||||
{
|
||||
foreach ($this->trackedObjects as $object) {
|
||||
Assert::assertFalse($this->uow->isInIdentityMap($object));
|
||||
}
|
||||
|
||||
++$this->invocationCount;
|
||||
}
|
||||
};
|
||||
|
||||
$this->_em->getEventManager()->addEventListener(Events::postRemove, $listener);
|
||||
$this->_em->remove($object1);
|
||||
$this->_em->remove($object2);
|
||||
$this->_em->flush();
|
||||
|
||||
self::assertSame(2, $listener->invocationCount);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -36,7 +36,7 @@ class EnumTest extends OrmFunctionalTestCase
|
||||
{
|
||||
parent::setUp();
|
||||
|
||||
$this->_em = $this->getEntityManager(null, new AttributeDriver([dirname(__DIR__, 2) . '/Models/Enums']));
|
||||
$this->_em = $this->getEntityManager(null, new AttributeDriver([dirname(__DIR__, 2) . '/Models/Enums'], true));
|
||||
$this->_schemaTool = new SchemaTool($this->_em);
|
||||
|
||||
if ($this->isSecondLevelCacheEnabled) {
|
||||
|
||||
135
tests/Doctrine/Tests/ORM/Functional/GH7877Test.php
Normal file
135
tests/Doctrine/Tests/ORM/Functional/GH7877Test.php
Normal file
@@ -0,0 +1,135 @@
|
||||
<?php
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\ORM\Mapping as ORM;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function uniqid;
|
||||
|
||||
/**
|
||||
* @group GH7877
|
||||
*/
|
||||
class GH7877Test extends OrmFunctionalTestCase
|
||||
{
|
||||
protected function setUp(): void
|
||||
{
|
||||
parent::setUp();
|
||||
|
||||
$this->createSchemaForModels(
|
||||
GH7877ApplicationGeneratedIdEntity::class,
|
||||
GH7877EntityWithNullableAssociation::class
|
||||
);
|
||||
}
|
||||
|
||||
public function testSelfReferenceWithApplicationGeneratedIdMayBeNotNullable(): void
|
||||
{
|
||||
$entity = new GH7877ApplicationGeneratedIdEntity();
|
||||
$entity->parent = $entity;
|
||||
|
||||
$this->expectNotToPerformAssertions();
|
||||
|
||||
$this->_em->persist($entity);
|
||||
$this->_em->flush();
|
||||
}
|
||||
|
||||
public function testCrossReferenceWithApplicationGeneratedIdMayBeNotNullable(): void
|
||||
{
|
||||
$entity1 = new GH7877ApplicationGeneratedIdEntity();
|
||||
$entity1->parent = $entity1;
|
||||
$entity2 = new GH7877ApplicationGeneratedIdEntity();
|
||||
$entity2->parent = $entity1;
|
||||
|
||||
$this->expectNotToPerformAssertions();
|
||||
|
||||
// As long as we do not have entity-level commit order computation
|
||||
// (see https://github.com/doctrine/orm/pull/10547),
|
||||
// this only works when the UoW processes $entity1 before $entity2,
|
||||
// so that the foreign key constraint E2 -> E1 can be satisfied.
|
||||
|
||||
$this->_em->persist($entity1);
|
||||
$this->_em->persist($entity2);
|
||||
$this->_em->flush();
|
||||
}
|
||||
|
||||
public function testNullableForeignKeysMakeInsertOrderLessRelevant(): void
|
||||
{
|
||||
$entity1 = new GH7877EntityWithNullableAssociation();
|
||||
$entity1->parent = $entity1;
|
||||
$entity2 = new GH7877EntityWithNullableAssociation();
|
||||
$entity2->parent = $entity1;
|
||||
|
||||
$this->expectNotToPerformAssertions();
|
||||
|
||||
// In contrast to the previous test, this case demonstrates that with NULLable
|
||||
// associations, even without entity-level commit order computation
|
||||
// (see https://github.com/doctrine/orm/pull/10547), we can get away with an
|
||||
// insertion order of E2 before E1. That is because the UoW will schedule an extra
|
||||
// update that saves the day - the foreign key reference will established only after
|
||||
// all insertions have been performed.
|
||||
|
||||
$this->_em->persist($entity2);
|
||||
$this->_em->persist($entity1);
|
||||
$this->_em->flush();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @ORM\Entity
|
||||
*/
|
||||
class GH7877ApplicationGeneratedIdEntity
|
||||
{
|
||||
/**
|
||||
* @ORM\Id
|
||||
* @ORM\Column(type="string")
|
||||
* @ORM\GeneratedValue(strategy="NONE")
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $id;
|
||||
|
||||
/**
|
||||
* (!) Note this uses "nullable=false"
|
||||
*
|
||||
* @ORM\ManyToOne(targetEntity="GH7877ApplicationGeneratedIdEntity")
|
||||
* @ORM\JoinColumn(name="parent_id", referencedColumnName="id", nullable=false)
|
||||
*
|
||||
* @var self
|
||||
*/
|
||||
public $parent;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->id = uniqid();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @ORM\Entity
|
||||
*/
|
||||
class GH7877EntityWithNullableAssociation
|
||||
{
|
||||
/**
|
||||
* @ORM\Id
|
||||
* @ORM\Column(type="string")
|
||||
* @ORM\GeneratedValue(strategy="NONE")
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
public $id;
|
||||
|
||||
/**
|
||||
* @ORM\ManyToOne(targetEntity="GH7877EntityWithNullableAssociation")
|
||||
* @ORM\JoinColumn(name="parent_id", referencedColumnName="id", nullable=true)
|
||||
*
|
||||
* @var self
|
||||
*/
|
||||
public $parent;
|
||||
|
||||
public function __construct()
|
||||
{
|
||||
$this->id = uniqid();
|
||||
}
|
||||
}
|
||||
@@ -4,7 +4,6 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\DirectoryTree\Directory;
|
||||
use Doctrine\Tests\Models\DirectoryTree\File;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -46,7 +45,7 @@ class MappedSuperclassTest extends OrmFunctionalTestCase
|
||||
$cleanFile = $this->_em->find(get_class($file), $file->getId());
|
||||
|
||||
self::assertInstanceOf(Directory::class, $cleanFile->getParent());
|
||||
self::assertInstanceOf(Proxy::class, $cleanFile->getParent());
|
||||
self::assertTrue($this->isUninitializedObject($cleanFile->getParent()));
|
||||
self::assertEquals($directory->getId(), $cleanFile->getParent()->getId());
|
||||
self::assertInstanceOf(Directory::class, $cleanFile->getParent()->getParent());
|
||||
self::assertEquals($root->getId(), $cleanFile->getParent()->getParent()->getId());
|
||||
|
||||
@@ -11,7 +11,6 @@ use Doctrine\ORM\EntityManager;
|
||||
use Doctrine\ORM\EntityManagerInterface;
|
||||
use Doctrine\ORM\ORMSetup;
|
||||
use Doctrine\ORM\Tools\SchemaTool;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\DbalExtensions\Connection;
|
||||
use Doctrine\Tests\DbalExtensions\QueryLog;
|
||||
use Doctrine\Tests\Models\Generic\DateTimeModel;
|
||||
@@ -48,8 +47,8 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
|
||||
self::assertSame($managed, $this->_em->merge($detachedUninitialized));
|
||||
|
||||
self::assertFalse($managed->__isInitialized());
|
||||
self::assertFalse($detachedUninitialized->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($managed));
|
||||
self::assertTrue($this->isUninitializedObject($detachedUninitialized));
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -71,8 +70,8 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
$this->_em->merge(unserialize(serialize($this->_em->merge($detachedUninitialized))))
|
||||
);
|
||||
|
||||
self::assertFalse($managed->__isInitialized());
|
||||
self::assertFalse($detachedUninitialized->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($managed));
|
||||
self::assertTrue($this->isUninitializedObject($detachedUninitialized));
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -87,7 +86,7 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
|
||||
self::assertSame($managed, $this->_em->merge($managed));
|
||||
|
||||
self::assertFalse($managed->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($managed));
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -109,13 +108,12 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
|
||||
$managed = $this->_em->getReference(DateTimeModel::class, $date->id);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $managed);
|
||||
self::assertFalse($managed->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($managed));
|
||||
|
||||
$date->date = $dateTime = new DateTime();
|
||||
|
||||
self::assertSame($managed, $this->_em->merge($date));
|
||||
self::assertTrue($managed->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($managed));
|
||||
self::assertSame($dateTime, $managed->date, 'Data was merged into the proxy after initialization');
|
||||
}
|
||||
|
||||
@@ -150,8 +148,8 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
self::assertNotSame($proxy1, $merged2);
|
||||
self::assertSame($proxy2, $merged2);
|
||||
|
||||
self::assertFalse($proxy1->__isInitialized());
|
||||
self::assertFalse($proxy2->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($proxy1));
|
||||
self::assertTrue($this->isUninitializedObject($proxy2));
|
||||
|
||||
$proxy1->__load();
|
||||
|
||||
@@ -207,9 +205,8 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
$unManagedProxy = $em1->getReference(DateTimeModel::class, $file1->id);
|
||||
$mergedInstance = $em2->merge($unManagedProxy);
|
||||
|
||||
self::assertNotInstanceOf(Proxy::class, $mergedInstance);
|
||||
self::assertNotSame($unManagedProxy, $mergedInstance);
|
||||
self::assertFalse($unManagedProxy->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($mergedInstance));
|
||||
self::assertTrue($this->isUninitializedObject($unManagedProxy));
|
||||
|
||||
self::assertCount(
|
||||
0,
|
||||
@@ -242,7 +239,9 @@ class MergeProxiesTest extends OrmFunctionalTestCase
|
||||
|
||||
TestUtil::configureProxies($config);
|
||||
$config->setMetadataDriverImpl(ORMSetup::createDefaultAnnotationDriver(
|
||||
[realpath(__DIR__ . '/../../Models/Cache')]
|
||||
[realpath(__DIR__ . '/../../Models/Cache')],
|
||||
null,
|
||||
true
|
||||
));
|
||||
|
||||
// always runs on sqlite to prevent multi-connection race-conditions with the test suite
|
||||
|
||||
@@ -6,7 +6,6 @@ namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\Common\Collections\Collection;
|
||||
use Doctrine\Common\Collections\Criteria;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceFeature;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceProduct;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -90,12 +89,12 @@ class OneToManyBidirectionalAssociationTest extends OrmFunctionalTestCase
|
||||
$features = $product->getFeatures();
|
||||
|
||||
self::assertInstanceOf(ECommerceFeature::class, $features[0]);
|
||||
self::assertNotInstanceOf(Proxy::class, $features[0]->getProduct());
|
||||
self::assertFalse($this->isUninitializedObject($features[0]->getProduct()));
|
||||
self::assertSame($product, $features[0]->getProduct());
|
||||
self::assertEquals('Model writing tutorial', $features[0]->getDescription());
|
||||
self::assertInstanceOf(ECommerceFeature::class, $features[1]);
|
||||
self::assertSame($product, $features[1]->getProduct());
|
||||
self::assertNotInstanceOf(Proxy::class, $features[1]->getProduct());
|
||||
self::assertFalse($this->isUninitializedObject($features[1]->getProduct()));
|
||||
self::assertEquals('Annotations examples', $features[1]->getDescription());
|
||||
}
|
||||
|
||||
@@ -126,11 +125,10 @@ class OneToManyBidirectionalAssociationTest extends OrmFunctionalTestCase
|
||||
$features = $query->getResult();
|
||||
|
||||
$product = $features[0]->getProduct();
|
||||
self::assertInstanceOf(Proxy::class, $product);
|
||||
self::assertInstanceOf(ECommerceProduct::class, $product);
|
||||
self::assertFalse($product->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($product));
|
||||
self::assertSame('Doctrine Cookbook', $product->getName());
|
||||
self::assertTrue($product->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($product));
|
||||
}
|
||||
|
||||
public function testLazyLoadsObjectsOnTheInverseSide2(): void
|
||||
@@ -141,7 +139,7 @@ class OneToManyBidirectionalAssociationTest extends OrmFunctionalTestCase
|
||||
$features = $query->getResult();
|
||||
|
||||
$product = $features[0]->getProduct();
|
||||
self::assertNotInstanceOf(Proxy::class, $product);
|
||||
self::assertFalse($this->isUninitializedObject($product));
|
||||
self::assertInstanceOf(ECommerceProduct::class, $product);
|
||||
self::assertSame('Doctrine Cookbook', $product->getName());
|
||||
|
||||
|
||||
@@ -5,7 +5,6 @@ declare(strict_types=1);
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceCart;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceCustomer;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -100,7 +99,7 @@ class OneToOneBidirectionalAssociationTest extends OrmFunctionalTestCase
|
||||
|
||||
self::assertNull($customer->getMentor());
|
||||
self::assertInstanceOf(ECommerceCart::class, $customer->getCart());
|
||||
self::assertNotInstanceOf(Proxy::class, $customer->getCart());
|
||||
self::assertFalse($this->isUninitializedObject($customer->getCart()));
|
||||
self::assertEquals('paypal', $customer->getCart()->getPayment());
|
||||
}
|
||||
|
||||
|
||||
@@ -14,7 +14,6 @@ use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function get_class;
|
||||
@@ -52,7 +51,7 @@ class OneToOneEagerLoadingTest extends OrmFunctionalTestCase
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
|
||||
$train = $this->_em->find(get_class($train), $train->id);
|
||||
self::assertNotInstanceOf(Proxy::class, $train->driver);
|
||||
self::assertFalse($this->isUninitializedObject($train->driver));
|
||||
self::assertEquals('Benjamin', $train->driver->name);
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
@@ -70,7 +69,6 @@ class OneToOneEagerLoadingTest extends OrmFunctionalTestCase
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
|
||||
$train = $this->_em->find(get_class($train), $train->id);
|
||||
self::assertNotInstanceOf(Proxy::class, $train->driver);
|
||||
self::assertNull($train->driver);
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
@@ -88,9 +86,9 @@ class OneToOneEagerLoadingTest extends OrmFunctionalTestCase
|
||||
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
|
||||
$driver = $this->_em->find(get_class($owner), $owner->id);
|
||||
self::assertNotInstanceOf(Proxy::class, $owner->train);
|
||||
self::assertNotNull($owner->train);
|
||||
$this->_em->find(get_class($owner), $owner->id);
|
||||
self::assertFalse($this->isUninitializedObject($owner->train));
|
||||
self::assertInstanceOf(Train::class, $owner->train);
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
}
|
||||
@@ -109,7 +107,6 @@ class OneToOneEagerLoadingTest extends OrmFunctionalTestCase
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
|
||||
$driver = $this->_em->find(get_class($driver), $driver->id);
|
||||
self::assertNotInstanceOf(Proxy::class, $driver->train);
|
||||
self::assertNull($driver->train);
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
@@ -126,8 +123,8 @@ class OneToOneEagerLoadingTest extends OrmFunctionalTestCase
|
||||
$this->_em->clear();
|
||||
|
||||
$waggon = $this->_em->find(get_class($waggon), $waggon->id);
|
||||
self::assertNotInstanceOf(Proxy::class, $waggon->train);
|
||||
self::assertNotNull($waggon->train);
|
||||
self::assertFalse($this->isUninitializedObject($waggon->train));
|
||||
self::assertInstanceOf(Train::class, $waggon->train);
|
||||
}
|
||||
|
||||
/** @group non-cacheable */
|
||||
|
||||
@@ -11,7 +11,6 @@ use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceCustomer;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
@@ -69,14 +68,16 @@ class OneToOneSelfReferentialAssociationTest extends OrmFunctionalTestCase
|
||||
$id = $this->createFixture();
|
||||
|
||||
$customer = $this->_em->find(ECommerceCustomer::class, $id);
|
||||
self::assertNotInstanceOf(Proxy::class, $customer->getMentor());
|
||||
self::assertFalse($this->isUninitializedObject($customer->getMentor()));
|
||||
}
|
||||
|
||||
public function testEagerLoadsAssociation(): void
|
||||
{
|
||||
$this->createFixture();
|
||||
$customerId = $this->createFixture();
|
||||
|
||||
$query = $this->_em->createQuery('select c, m from Doctrine\Tests\Models\ECommerce\ECommerceCustomer c left join c.mentor m where c.id = :id');
|
||||
$query->setParameter('id', $customerId);
|
||||
|
||||
$query = $this->_em->createQuery('select c, m from Doctrine\Tests\Models\ECommerce\ECommerceCustomer c left join c.mentor m order by c.id asc');
|
||||
$result = $query->getResult();
|
||||
$customer = $result[0];
|
||||
$this->assertLoadingOfAssociation($customer);
|
||||
|
||||
@@ -59,7 +59,7 @@ class ProxiesLikeEntitiesTest extends OrmFunctionalTestCase
|
||||
{
|
||||
// Considering case (a)
|
||||
$proxy = $this->_em->getProxyFactory()->getProxy(CmsUser::class, ['id' => 123]);
|
||||
$proxy->__setInitialized(true);
|
||||
|
||||
$proxy->id = null;
|
||||
$proxy->username = 'ocra';
|
||||
$proxy->name = 'Marco';
|
||||
@@ -84,7 +84,7 @@ class ProxiesLikeEntitiesTest extends OrmFunctionalTestCase
|
||||
|
||||
$this->_em->persist($uninitializedProxy);
|
||||
$this->_em->flush();
|
||||
self::assertFalse($uninitializedProxy->__isInitialized(), 'Proxy didn\'t get initialized during flush operations');
|
||||
self::assertTrue($this->isUninitializedObject($uninitializedProxy), 'Proxy didn\'t get initialized during flush operations');
|
||||
self::assertEquals($userId, $uninitializedProxy->getId());
|
||||
$this->_em->remove($uninitializedProxy);
|
||||
$this->_em->flush();
|
||||
@@ -95,7 +95,7 @@ class ProxiesLikeEntitiesTest extends OrmFunctionalTestCase
|
||||
*/
|
||||
public function testProxyAsDqlParameterPersist(): void
|
||||
{
|
||||
$proxy = $this->_em->getProxyFactory()->getProxy(CmsUser::class, ['id' => $this->user->getId()]);
|
||||
$proxy = $this->_em->getReference(CmsUser::class, ['id' => $this->user->getId()]);
|
||||
$proxy->id = $this->user->getId();
|
||||
$result = $this
|
||||
->_em
|
||||
|
||||
@@ -13,7 +13,6 @@ use Doctrine\ORM\Query;
|
||||
use Doctrine\ORM\Query\Parameter;
|
||||
use Doctrine\ORM\Query\QueryException;
|
||||
use Doctrine\ORM\UnexpectedResultException;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\IterableTester;
|
||||
use Doctrine\Tests\Models\CMS\CmsArticle;
|
||||
use Doctrine\Tests\Models\CMS\CmsPhonenumber;
|
||||
@@ -624,8 +623,7 @@ class QueryTest extends OrmFunctionalTestCase
|
||||
self::assertEquals(1, count($result));
|
||||
self::assertInstanceOf(CmsArticle::class, $result[0]);
|
||||
self::assertEquals('dr. dolittle', $result[0]->topic);
|
||||
self::assertInstanceOf(Proxy::class, $result[0]->user);
|
||||
self::assertFalse($result[0]->user->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($result[0]->user));
|
||||
}
|
||||
|
||||
/** @group DDC-952 */
|
||||
@@ -653,7 +651,7 @@ class QueryTest extends OrmFunctionalTestCase
|
||||
|
||||
self::assertCount(10, $articles);
|
||||
foreach ($articles as $article) {
|
||||
self::assertNotInstanceOf(Proxy::class, $article);
|
||||
self::assertFalse($this->isUninitializedObject($article));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -26,7 +26,8 @@ class ReadonlyPropertiesTest extends OrmFunctionalTestCase
|
||||
}
|
||||
|
||||
$this->_em = $this->getEntityManager(null, new AttributeDriver(
|
||||
[dirname(__DIR__, 2) . '/Models/ReadonlyProperties']
|
||||
[dirname(__DIR__, 2) . '/Models/ReadonlyProperties'],
|
||||
true
|
||||
));
|
||||
$this->_schemaTool = new SchemaTool($this->_em);
|
||||
|
||||
|
||||
@@ -6,7 +6,7 @@ namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\Common\Proxy\Proxy as CommonProxy;
|
||||
use Doctrine\Common\Util\ClassUtils;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\Tests\Models\Company\CompanyAuction;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceProduct;
|
||||
use Doctrine\Tests\Models\ECommerce\ECommerceShipping;
|
||||
@@ -120,9 +120,9 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
$entity = $this->_em->getReference(ECommerceProduct::class, $id);
|
||||
assert($entity instanceof ECommerceProduct);
|
||||
|
||||
self::assertFalse($entity->__isInitialized(), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertTrue($this->isUninitializedObject($entity), 'Pre-Condition: Object is unitialized proxy.');
|
||||
$this->_em->getUnitOfWork()->initializeObject($entity);
|
||||
self::assertTrue($entity->__isInitialized(), 'Should be initialized after called UnitOfWork::initializeObject()');
|
||||
self::assertFalse($this->isUninitializedObject($entity), 'Should be initialized after called UnitOfWork::initializeObject()');
|
||||
}
|
||||
|
||||
/** @group DDC-1163 */
|
||||
@@ -167,9 +167,9 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
$entity = $this->_em->getReference(ECommerceProduct::class, $id);
|
||||
assert($entity instanceof ECommerceProduct);
|
||||
|
||||
self::assertFalse($entity->__isInitialized(), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertTrue($this->isUninitializedObject($entity), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertEquals($id, $entity->getId());
|
||||
self::assertFalse($entity->__isInitialized(), "Getting the identifier doesn't initialize the proxy.");
|
||||
self::assertTrue($this->isUninitializedObject($entity), "Getting the identifier doesn't initialize the proxy.");
|
||||
}
|
||||
|
||||
/** @group DDC-1625 */
|
||||
@@ -180,9 +180,9 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
$entity = $this->_em->getReference(CompanyAuction::class, $id);
|
||||
assert($entity instanceof CompanyAuction);
|
||||
|
||||
self::assertFalse($entity->__isInitialized(), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertTrue($this->isUninitializedObject($entity), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertEquals($id, $entity->getId());
|
||||
self::assertFalse($entity->__isInitialized(), "Getting the identifier doesn't initialize the proxy when extending.");
|
||||
self::assertTrue($this->isUninitializedObject($entity), "Getting the identifier doesn't initialize the proxy when extending.");
|
||||
}
|
||||
|
||||
public function testDoNotInitializeProxyOnGettingTheIdentifierAndReturnTheRightType(): void
|
||||
@@ -202,10 +202,10 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
$product = $this->_em->getRepository(ECommerceProduct::class)->find($product->getId());
|
||||
|
||||
$entity = $product->getShipping();
|
||||
self::assertFalse($entity->__isInitialized(), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertTrue($this->isUninitializedObject($entity), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertEquals($id, $entity->getId());
|
||||
self::assertSame($id, $entity->getId(), "Check that the id's are the same value, and type.");
|
||||
self::assertFalse($entity->__isInitialized(), "Getting the identifier doesn't initialize the proxy.");
|
||||
self::assertTrue($this->isUninitializedObject($entity), "Getting the identifier doesn't initialize the proxy.");
|
||||
}
|
||||
|
||||
public function testInitializeProxyOnGettingSomethingOtherThanTheIdentifier(): void
|
||||
@@ -215,9 +215,9 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
$entity = $this->_em->getReference(ECommerceProduct::class, $id);
|
||||
assert($entity instanceof ECommerceProduct);
|
||||
|
||||
self::assertFalse($entity->__isInitialized(), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertTrue($this->isUninitializedObject($entity), 'Pre-Condition: Object is unitialized proxy.');
|
||||
self::assertEquals('Doctrine Cookbook', $entity->getName());
|
||||
self::assertTrue($entity->__isInitialized(), 'Getting something other than the identifier initializes the proxy.');
|
||||
self::assertFalse($this->isUninitializedObject($entity), 'Getting something other than the identifier initializes the proxy.');
|
||||
}
|
||||
|
||||
/** @group DDC-1604 */
|
||||
@@ -229,8 +229,8 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
assert($entity instanceof ECommerceProduct);
|
||||
$className = ClassUtils::getClass($entity);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $entity);
|
||||
self::assertFalse($entity->__isInitialized());
|
||||
self::assertInstanceOf(InternalProxy::class, $entity);
|
||||
self::assertTrue($this->isUninitializedObject($entity));
|
||||
self::assertEquals(ECommerceProduct::class, $className);
|
||||
|
||||
$restName = str_replace($this->_em->getConfiguration()->getProxyNamespace(), '', get_class($entity));
|
||||
@@ -239,6 +239,6 @@ class ReferenceProxyTest extends OrmFunctionalTestCase
|
||||
self::assertTrue(file_exists($proxyFileName), 'Proxy file name cannot be found generically.');
|
||||
|
||||
$entity->__load();
|
||||
self::assertTrue($entity->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($entity));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -10,9 +10,9 @@ use Doctrine\ORM\Cache\EntityCacheEntry;
|
||||
use Doctrine\ORM\Cache\EntityCacheKey;
|
||||
use Doctrine\ORM\Cache\Exception\CacheException;
|
||||
use Doctrine\ORM\Cache\QueryCacheKey;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\ORM\Query;
|
||||
use Doctrine\ORM\Query\ResultSetMapping;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\Cache\Attraction;
|
||||
use Doctrine\Tests\Models\Cache\City;
|
||||
use Doctrine\Tests\Models\Cache\Country;
|
||||
@@ -938,7 +938,7 @@ class SecondLevelCacheQueryCacheTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertNotNull($state1->getCountry());
|
||||
$this->assertQueryCount(1);
|
||||
self::assertInstanceOf(State::class, $state1);
|
||||
self::assertInstanceOf(Proxy::class, $state1->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $state1->getCountry());
|
||||
self::assertEquals($countryName, $state1->getCountry()->getName());
|
||||
self::assertEquals($stateId, $state1->getId());
|
||||
|
||||
@@ -956,7 +956,7 @@ class SecondLevelCacheQueryCacheTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertNotNull($state2->getCountry());
|
||||
$this->assertQueryCount(0);
|
||||
self::assertInstanceOf(State::class, $state2);
|
||||
self::assertInstanceOf(Proxy::class, $state2->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $state2->getCountry());
|
||||
self::assertEquals($countryName, $state2->getCountry()->getName());
|
||||
self::assertEquals($stateId, $state2->getId());
|
||||
}
|
||||
@@ -1030,7 +1030,7 @@ class SecondLevelCacheQueryCacheTest extends SecondLevelCacheFunctionalTestCase
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
self::assertInstanceOf(State::class, $state1);
|
||||
self::assertInstanceOf(Proxy::class, $state1->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $state1->getCountry());
|
||||
self::assertInstanceOf(City::class, $state1->getCities()->get(0));
|
||||
self::assertInstanceOf(State::class, $state1->getCities()->get(0)->getState());
|
||||
self::assertSame($state1, $state1->getCities()->get(0)->getState());
|
||||
@@ -1047,7 +1047,7 @@ class SecondLevelCacheQueryCacheTest extends SecondLevelCacheFunctionalTestCase
|
||||
|
||||
$this->assertQueryCount(0);
|
||||
self::assertInstanceOf(State::class, $state2);
|
||||
self::assertInstanceOf(Proxy::class, $state2->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $state2->getCountry());
|
||||
self::assertInstanceOf(City::class, $state2->getCities()->get(0));
|
||||
self::assertInstanceOf(State::class, $state2->getCities()->get(0)->getState());
|
||||
self::assertSame($state2, $state2->getCities()->get(0)->getState());
|
||||
|
||||
@@ -4,7 +4,7 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional;
|
||||
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\Tests\Models\Cache\Country;
|
||||
use Doctrine\Tests\Models\Cache\State;
|
||||
|
||||
@@ -197,8 +197,8 @@ class SecondLevelCacheRepositoryTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertInstanceOf(State::class, $entities[1]);
|
||||
self::assertInstanceOf(Country::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Country::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[1]->getCountry());
|
||||
|
||||
// load from cache
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
@@ -211,8 +211,8 @@ class SecondLevelCacheRepositoryTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertInstanceOf(State::class, $entities[1]);
|
||||
self::assertInstanceOf(Country::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Country::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[1]->getCountry());
|
||||
|
||||
// invalidate cache
|
||||
$this->_em->persist(new State('foo', $this->_em->find(Country::class, $this->countries[0]->getId())));
|
||||
@@ -230,8 +230,8 @@ class SecondLevelCacheRepositoryTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertInstanceOf(State::class, $entities[1]);
|
||||
self::assertInstanceOf(Country::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Country::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[1]->getCountry());
|
||||
|
||||
// load from cache
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
@@ -244,7 +244,7 @@ class SecondLevelCacheRepositoryTest extends SecondLevelCacheFunctionalTestCase
|
||||
self::assertInstanceOf(State::class, $entities[1]);
|
||||
self::assertInstanceOf(Country::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Country::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(Proxy::class, $entities[1]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[0]->getCountry());
|
||||
self::assertInstanceOf(InternalProxy::class, $entities[1]->getCountry());
|
||||
}
|
||||
}
|
||||
|
||||
@@ -7,7 +7,6 @@ namespace Doctrine\Tests\ORM\Functional;
|
||||
use Doctrine\Common\Collections\Criteria;
|
||||
use Doctrine\ORM\Mapping\ClassMetadata;
|
||||
use Doctrine\ORM\Persisters\MatchingAssociationFieldRequiresObject;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\Company\CompanyContract;
|
||||
use Doctrine\Tests\Models\Company\CompanyEmployee;
|
||||
use Doctrine\Tests\Models\Company\CompanyFixContract;
|
||||
@@ -397,13 +396,14 @@ class SingleTableInheritanceTest extends OrmFunctionalTestCase
|
||||
$this->loadFullFixture();
|
||||
|
||||
$ref = $this->_em->getReference(CompanyContract::class, $this->fix->getId());
|
||||
self::assertNotInstanceOf(Proxy::class, $ref, 'Cannot Request a proxy from a class that has subclasses.');
|
||||
self::assertFalse($this->isUninitializedObject($ref), 'Cannot Request a proxy from a class that has subclasses.');
|
||||
self::assertInstanceOf(CompanyContract::class, $ref);
|
||||
self::assertInstanceOf(CompanyFixContract::class, $ref, 'Direct fetch of the reference has to load the child class Employee directly.');
|
||||
$this->_em->clear();
|
||||
|
||||
$ref = $this->_em->getReference(CompanyFixContract::class, $this->fix->getId());
|
||||
self::assertInstanceOf(Proxy::class, $ref, 'A proxy can be generated only if no subclasses exists for the requested reference.');
|
||||
|
||||
self::assertTrue($this->isUninitializedObject($ref), 'A proxy can be generated only if no subclasses exists for the requested reference.');
|
||||
}
|
||||
|
||||
/** @group DDC-952 */
|
||||
@@ -417,6 +417,6 @@ class SingleTableInheritanceTest extends OrmFunctionalTestCase
|
||||
->setParameter(1, $this->fix->getId())
|
||||
->getSingleResult();
|
||||
|
||||
self::assertNotInstanceOf(Proxy::class, $contract->getSalesPerson());
|
||||
self::assertFalse($this->isUninitializedObject($contract->getSalesPerson()));
|
||||
}
|
||||
}
|
||||
|
||||
@@ -15,7 +15,6 @@ use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\JoinColumns;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function assert;
|
||||
@@ -92,7 +91,7 @@ class DDC1163Test extends OrmFunctionalTestCase
|
||||
assert($specialProduct instanceof DDC1163SpecialProduct);
|
||||
|
||||
self::assertInstanceOf(DDC1163SpecialProduct::class, $specialProduct);
|
||||
self::assertInstanceOf(Proxy::class, $specialProduct);
|
||||
self::assertTrue($this->isUninitializedObject($specialProduct));
|
||||
|
||||
$specialProduct->setSubclassProperty('foobar');
|
||||
|
||||
|
||||
@@ -45,7 +45,7 @@ class DDC1193Test extends OrmFunctionalTestCase
|
||||
$company = $this->_em->find(get_class($company), $companyId);
|
||||
|
||||
self::assertTrue($this->_em->getUnitOfWork()->isInIdentityMap($company), 'Company is in identity map.');
|
||||
self::assertFalse($company->member->__isInitialized(), 'Pre-Condition');
|
||||
self::assertTrue($this->isUninitializedObject($company->member), 'Pre-Condition');
|
||||
self::assertTrue($this->_em->getUnitOfWork()->isInIdentityMap($company->member), 'Member is in identity map.');
|
||||
|
||||
$this->_em->remove($company);
|
||||
|
||||
@@ -38,9 +38,9 @@ class DDC1228Test extends OrmFunctionalTestCase
|
||||
|
||||
$user = $this->_em->find(DDC1228User::class, $user->id);
|
||||
|
||||
self::assertFalse($user->getProfile()->__isInitialized(), 'Proxy is not initialized');
|
||||
self::assertTrue($this->isUninitializedObject($user->getProfile()), 'Proxy is not initialized');
|
||||
$user->getProfile()->setName('Bar');
|
||||
self::assertTrue($user->getProfile()->__isInitialized(), 'Proxy is not initialized');
|
||||
self::assertFalse($this->isUninitializedObject($user->getProfile()), 'Proxy is not initialized');
|
||||
|
||||
self::assertEquals('Bar', $user->getProfile()->getName());
|
||||
self::assertEquals(['id' => 1, 'name' => 'Foo'], $this->_em->getUnitOfWork()->getOriginalEntityData($user->getProfile()));
|
||||
|
||||
@@ -56,7 +56,7 @@ class DDC1238Test extends OrmFunctionalTestCase
|
||||
|
||||
$user2 = $this->_em->getReference(DDC1238User::class, $userId);
|
||||
|
||||
$user->__load();
|
||||
//$user->__load();
|
||||
|
||||
self::assertIsInt($user->getId(), 'Even if a proxy is detached, it should still have an identifier');
|
||||
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\CMS\CmsAddress;
|
||||
use Doctrine\Tests\Models\CMS\CmsUser;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -54,7 +53,7 @@ class DDC1452Test extends OrmFunctionalTestCase
|
||||
$results = $this->_em->createQuery($dql)->setMaxResults(1)->getResult();
|
||||
|
||||
self::assertSame($results[0], $results[0]->entitiesB[0]->entityAFrom);
|
||||
self::assertNotInstanceOf(Proxy::class, $results[0]->entitiesB[0]->entityATo);
|
||||
self::assertFalse($this->isUninitializedObject($results[0]->entitiesB[0]->entityATo));
|
||||
self::assertInstanceOf(Collection::class, $results[0]->entitiesB[0]->entityATo->getEntitiesB());
|
||||
}
|
||||
|
||||
@@ -82,12 +81,12 @@ class DDC1452Test extends OrmFunctionalTestCase
|
||||
$data = $this->_em->createQuery($dql)->getResult();
|
||||
$this->_em->clear();
|
||||
|
||||
self::assertNotInstanceOf(Proxy::class, $data[0]->user);
|
||||
self::assertFalse($this->isUninitializedObject($data[0]->user));
|
||||
|
||||
$dql = 'SELECT u, a FROM Doctrine\Tests\Models\CMS\CmsUser u INNER JOIN u.address a';
|
||||
$data = $this->_em->createQuery($dql)->getResult();
|
||||
|
||||
self::assertNotInstanceOf(Proxy::class, $data[0]->address);
|
||||
self::assertFalse($this->isUninitializedObject($data[0]->address));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\NotifyPropertyChanged;
|
||||
use Doctrine\Persistence\PropertyChangedListener;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function count;
|
||||
@@ -57,9 +56,10 @@ class DDC1690Test extends OrmFunctionalTestCase
|
||||
$child = $this->_em->find(DDC1690Child::class, $childId);
|
||||
|
||||
self::assertEquals(1, count($parent->listeners));
|
||||
self::assertInstanceOf(Proxy::class, $child, 'Verifying that $child is a proxy before using proxy API');
|
||||
self::assertCount(0, $child->listeners);
|
||||
$child->__load();
|
||||
|
||||
$this->_em->getUnitOfWork()->initializeObject($child);
|
||||
|
||||
self::assertCount(1, $child->listeners);
|
||||
unset($parent, $child);
|
||||
|
||||
|
||||
@@ -4,7 +4,7 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional\Ticket;
|
||||
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\ORM\Proxy\InternalProxy;
|
||||
use Doctrine\Tests\Models\CMS\CmsGroup;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
@@ -37,8 +37,7 @@ class DDC1734Test extends OrmFunctionalTestCase
|
||||
|
||||
$proxy = $this->getProxy($group);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $proxy);
|
||||
self::assertFalse($proxy->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($proxy));
|
||||
|
||||
$this->_em->detach($proxy);
|
||||
$this->_em->clear();
|
||||
@@ -67,8 +66,7 @@ class DDC1734Test extends OrmFunctionalTestCase
|
||||
|
||||
$proxy = $this->getProxy($group);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $proxy);
|
||||
self::assertFalse($proxy->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($proxy));
|
||||
|
||||
$this->_em->detach($proxy);
|
||||
$serializedProxy = serialize($proxy);
|
||||
@@ -79,7 +77,7 @@ class DDC1734Test extends OrmFunctionalTestCase
|
||||
}
|
||||
|
||||
/** @param object $object */
|
||||
private function getProxy($object): Proxy
|
||||
private function getProxy($object): InternalProxy
|
||||
{
|
||||
$metadataFactory = $this->_em->getMetadataFactory();
|
||||
$className = get_class($object);
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\NotifyPropertyChanged;
|
||||
use Doctrine\Persistence\PropertyChangedListener;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function assert;
|
||||
@@ -47,10 +46,8 @@ class DDC2230Test extends OrmFunctionalTestCase
|
||||
$mergedUser = $this->_em->merge($user);
|
||||
|
||||
$address = $mergedUser->address;
|
||||
assert($address instanceof Proxy);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $address);
|
||||
self::assertFalse($address->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($address));
|
||||
}
|
||||
|
||||
public function testNotifyTrackingCalledOnProxyInitialization(): void
|
||||
@@ -62,12 +59,12 @@ class DDC2230Test extends OrmFunctionalTestCase
|
||||
$this->_em->clear();
|
||||
|
||||
$addressProxy = $this->_em->getReference(DDC2230Address::class, $insertedAddress->id);
|
||||
assert($addressProxy instanceof Proxy || $addressProxy instanceof DDC2230Address);
|
||||
assert($addressProxy instanceof DDC2230Address);
|
||||
|
||||
self::assertFalse($addressProxy->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($addressProxy));
|
||||
self::assertNull($addressProxy->listener);
|
||||
|
||||
$addressProxy->__load();
|
||||
$this->_em->getUnitOfWork()->initializeObject($addressProxy);
|
||||
|
||||
self::assertSame($this->_em->getUnitOfWork(), $addressProxy->listener);
|
||||
}
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\Table;
|
||||
use Doctrine\Persistence\Mapping\ClassMetadata;
|
||||
use Doctrine\Persistence\ObjectManager;
|
||||
use Doctrine\Persistence\ObjectManagerAware;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function get_class;
|
||||
@@ -43,12 +42,11 @@ class DDC2231Test extends OrmFunctionalTestCase
|
||||
|
||||
$y1ref = $this->_em->getReference(get_class($y1), $y1->id);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $y1ref);
|
||||
self::assertFalse($y1ref->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($y1ref));
|
||||
|
||||
$id = $y1ref->doSomething();
|
||||
|
||||
self::assertTrue($y1ref->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($y1ref));
|
||||
self::assertEquals($this->_em, $y1ref->om);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function assert;
|
||||
@@ -63,16 +62,15 @@ class DDC2306Test extends OrmFunctionalTestCase
|
||||
$address = $this->_em->find(DDC2306Address::class, $address->id);
|
||||
assert($address instanceof DDC2306Address);
|
||||
$user = $address->users->first()->user;
|
||||
assert($user instanceof DDC2306User || $user instanceof Proxy);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $user);
|
||||
$this->assertTrue($this->isUninitializedObject($user));
|
||||
self::assertInstanceOf(DDC2306User::class, $user);
|
||||
|
||||
$userId = $user->id;
|
||||
|
||||
self::assertNotNull($userId);
|
||||
|
||||
$user->__load();
|
||||
$this->_em->getUnitOfWork()->initializeObject($user);
|
||||
|
||||
self::assertEquals(
|
||||
$userId,
|
||||
|
||||
@@ -11,7 +11,6 @@ use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\ORM\Mapping\Table;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function get_class;
|
||||
@@ -50,16 +49,14 @@ class DDC237Test extends OrmFunctionalTestCase
|
||||
$this->_em->clear();
|
||||
|
||||
$x2 = $this->_em->find(get_class($x), $x->id); // proxy injected for Y
|
||||
self::assertInstanceOf(Proxy::class, $x2->y);
|
||||
self::assertFalse($x2->y->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($x2->y));
|
||||
|
||||
// proxy for Y is in identity map
|
||||
|
||||
$z2 = $this->_em->createQuery('select z,y from ' . get_class($z) . ' z join z.y y where z.id = ?1')
|
||||
->setParameter(1, $z->id)
|
||||
->getSingleResult();
|
||||
self::assertInstanceOf(Proxy::class, $z2->y);
|
||||
self::assertTrue($z2->y->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($z2->y));
|
||||
self::assertEquals('Y', $z2->y->data);
|
||||
self::assertEquals($y->id, $z2->y->id);
|
||||
|
||||
@@ -69,7 +66,6 @@ class DDC237Test extends OrmFunctionalTestCase
|
||||
self::assertNotSame($x, $x2);
|
||||
self::assertNotSame($z, $z2);
|
||||
self::assertSame($z2->y, $x2->y);
|
||||
self::assertInstanceOf(Proxy::class, $z2->y);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@@ -15,7 +15,6 @@ use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\ORM\Mapping\Table;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
/**
|
||||
@@ -61,21 +60,20 @@ class DDC2494Test extends OrmFunctionalTestCase
|
||||
|
||||
$this->getQueryLog()->reset()->enable();
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $item->getCurrency());
|
||||
self::assertFalse($item->getCurrency()->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($item->getCurrency()));
|
||||
|
||||
self::assertArrayHasKey('convertToPHPValue', DDC2494TinyIntType::$calls);
|
||||
self::assertCount(1, DDC2494TinyIntType::$calls['convertToPHPValue']);
|
||||
|
||||
self::assertIsInt($item->getCurrency()->getId());
|
||||
self::assertCount(1, DDC2494TinyIntType::$calls['convertToPHPValue']);
|
||||
self::assertFalse($item->getCurrency()->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($item->getCurrency()));
|
||||
|
||||
$this->assertQueryCount(0);
|
||||
|
||||
self::assertIsInt($item->getCurrency()->getTemp());
|
||||
self::assertCount(3, DDC2494TinyIntType::$calls['convertToPHPValue']);
|
||||
self::assertTrue($item->getCurrency()->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($item->getCurrency()));
|
||||
|
||||
$this->assertQueryCount(1);
|
||||
}
|
||||
|
||||
@@ -4,7 +4,6 @@ declare(strict_types=1);
|
||||
|
||||
namespace Doctrine\Tests\ORM\Functional\Ticket;
|
||||
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\Models\Legacy\LegacyUser;
|
||||
use Doctrine\Tests\Models\Legacy\LegacyUserReference;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
@@ -36,15 +35,10 @@ class DDC2519Test extends OrmFunctionalTestCase
|
||||
self::assertInstanceOf(LegacyUser::class, $result[1]->source());
|
||||
self::assertInstanceOf(LegacyUser::class, $result[1]->target());
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $result[0]->source());
|
||||
self::assertInstanceOf(Proxy::class, $result[0]->target());
|
||||
self::assertInstanceOf(Proxy::class, $result[1]->source());
|
||||
self::assertInstanceOf(Proxy::class, $result[1]->target());
|
||||
|
||||
self::assertFalse($result[0]->target()->__isInitialized());
|
||||
self::assertFalse($result[0]->source()->__isInitialized());
|
||||
self::assertFalse($result[1]->target()->__isInitialized());
|
||||
self::assertFalse($result[1]->source()->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($result[0]->target()));
|
||||
self::assertTrue($this->isUninitializedObject($result[0]->source()));
|
||||
self::assertTrue($this->isUninitializedObject($result[1]->target()));
|
||||
self::assertTrue($this->isUninitializedObject($result[1]->source()));
|
||||
|
||||
self::assertNotNull($result[0]->source()->getId());
|
||||
self::assertNotNull($result[0]->target()->getId());
|
||||
|
||||
@@ -14,7 +14,6 @@ use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\ORM\Query;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
/** @group DDC-371 */
|
||||
@@ -51,7 +50,7 @@ class DDC371Test extends OrmFunctionalTestCase
|
||||
->getResult();
|
||||
|
||||
self::assertCount(1, $children);
|
||||
self::assertNotInstanceOf(Proxy::class, $children[0]->parent);
|
||||
self::assertFalse($this->isUninitializedObject($children[0]->parent));
|
||||
self::assertFalse($children[0]->parent->children->isInitialized());
|
||||
self::assertEquals(0, $children[0]->parent->children->unwrap()->count());
|
||||
}
|
||||
|
||||
@@ -10,7 +10,6 @@ use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
use function get_class;
|
||||
@@ -52,7 +51,7 @@ class DDC522Test extends OrmFunctionalTestCase
|
||||
|
||||
self::assertInstanceOf(DDC522Cart::class, $r[0]);
|
||||
self::assertInstanceOf(DDC522Customer::class, $r[0]->customer);
|
||||
self::assertNotInstanceOf(Proxy::class, $r[0]->customer);
|
||||
self::assertFalse($this->isUninitializedObject($r[0]->customer));
|
||||
self::assertEquals('name', $r[0]->customer->name);
|
||||
|
||||
$fkt = new DDC522ForeignKeyTest();
|
||||
@@ -64,8 +63,7 @@ class DDC522Test extends OrmFunctionalTestCase
|
||||
|
||||
$fkt2 = $this->_em->find(get_class($fkt), $fkt->id);
|
||||
self::assertEquals($fkt->cart->id, $fkt2->cartId);
|
||||
self::assertInstanceOf(Proxy::class, $fkt2->cart);
|
||||
self::assertFalse($fkt2->cart->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($fkt2->cart));
|
||||
}
|
||||
|
||||
/**
|
||||
|
||||
@@ -16,7 +16,6 @@ use Doctrine\ORM\Mapping\InheritanceType;
|
||||
use Doctrine\ORM\Mapping\JoinColumn;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\ORM\Mapping\OneToMany;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
class DDC531Test extends OrmFunctionalTestCase
|
||||
@@ -46,7 +45,7 @@ class DDC531Test extends OrmFunctionalTestCase
|
||||
// parent will already be loaded, cannot be lazy because it has mapped subclasses and we would not
|
||||
// know which proxy type to put in.
|
||||
self::assertInstanceOf(DDC531Item::class, $item3->parent);
|
||||
self::assertNotInstanceOf(Proxy::class, $item3->parent);
|
||||
self::assertFalse($this->isUninitializedObject($item3->parent));
|
||||
$item4 = $this->_em->find(DDC531Item::class, $item1->id); // Load parent item (id 1)
|
||||
self::assertNull($item4->parent);
|
||||
self::assertNotNull($item4->getChildren());
|
||||
|
||||
@@ -9,7 +9,6 @@ use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\OneToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
class DDC633Test extends OrmFunctionalTestCase
|
||||
@@ -44,7 +43,7 @@ class DDC633Test extends OrmFunctionalTestCase
|
||||
$eagerAppointment = $this->_em->find(DDC633Appointment::class, $app->id);
|
||||
|
||||
// Eager loading of one to one leads to fetch-join
|
||||
self::assertNotInstanceOf(Proxy::class, $eagerAppointment->patient);
|
||||
self::assertFalse($this->isUninitializedObject($eagerAppointment->patient));
|
||||
self::assertTrue($this->_em->contains($eagerAppointment->patient));
|
||||
}
|
||||
|
||||
@@ -70,8 +69,7 @@ class DDC633Test extends OrmFunctionalTestCase
|
||||
$appointments = $this->_em->createQuery('SELECT a FROM ' . __NAMESPACE__ . '\DDC633Appointment a')->getResult();
|
||||
|
||||
foreach ($appointments as $eagerAppointment) {
|
||||
self::assertInstanceOf(Proxy::class, $eagerAppointment->patient);
|
||||
self::assertTrue($eagerAppointment->patient->__isInitialized(), 'Proxy should already be initialized due to eager loading!');
|
||||
self::assertFalse($this->isUninitializedObject($eagerAppointment->patient), 'Proxy should already be initialized due to eager loading!');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -12,7 +12,6 @@ use Doctrine\ORM\Mapping\Entity;
|
||||
use Doctrine\ORM\Mapping\GeneratedValue;
|
||||
use Doctrine\ORM\Mapping\Id;
|
||||
use Doctrine\ORM\Mapping\ManyToOne;
|
||||
use Doctrine\Persistence\Proxy;
|
||||
use Doctrine\Tests\OrmFunctionalTestCase;
|
||||
|
||||
class DDC6460Test extends OrmFunctionalTestCase
|
||||
@@ -61,11 +60,10 @@ class DDC6460Test extends OrmFunctionalTestCase
|
||||
|
||||
$secondEntityWithLazyParameter = $this->_em->getRepository(DDC6460ParentEntity::class)->findOneById(1);
|
||||
|
||||
self::assertInstanceOf(Proxy::class, $secondEntityWithLazyParameter->lazyLoaded);
|
||||
self::assertInstanceOf(DDC6460Entity::class, $secondEntityWithLazyParameter->lazyLoaded);
|
||||
self::assertFalse($secondEntityWithLazyParameter->lazyLoaded->__isInitialized());
|
||||
self::assertTrue($this->isUninitializedObject($secondEntityWithLazyParameter->lazyLoaded));
|
||||
self::assertEquals($secondEntityWithLazyParameter->lazyLoaded->embedded, $entity->embedded);
|
||||
self::assertTrue($secondEntityWithLazyParameter->lazyLoaded->__isInitialized());
|
||||
self::assertFalse($this->isUninitializedObject($secondEntityWithLazyParameter->lazyLoaded));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user