From 90376a64311ea30dc47f804233a6c035cb22194f Mon Sep 17 00:00:00 2001 From: Timothy Choi Date: Mon, 22 Apr 2024 15:27:13 +0200 Subject: [PATCH 01/15] fix(docs): typo --- docs/en/cookbook/validation-of-entities.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/en/cookbook/validation-of-entities.rst b/docs/en/cookbook/validation-of-entities.rst index a1e39a5e51b..a33cb983240 100644 --- a/docs/en/cookbook/validation-of-entities.rst +++ b/docs/en/cookbook/validation-of-entities.rst @@ -11,7 +11,7 @@ What we offer are hooks to execute any kind of validation. .. note:: You don't need to validate your entities in the lifecycle - events. Its only one of many options. Of course you can also + events. It is only one of many options. Of course you can also perform validations in value setters or any other method of your entities that are used in your code. From fb4578406f495393955ac3d373f07f02d86159be Mon Sep 17 00:00:00 2001 From: Tomasz Ryba Date: Tue, 23 Apr 2024 23:31:19 +0200 Subject: [PATCH 02/15] Respect orderBy for EAGER fetch mode EAGER fetch mode ignores orderBy as of changes introduced with #8391 Fixes #11163 Fixes #11381 --- src/UnitOfWork.php | 10 +- .../ORM/Functional/Ticket/GH11163Test.php | 135 ++++++++++++++++++ 2 files changed, 143 insertions(+), 2 deletions(-) create mode 100644 tests/Tests/ORM/Functional/Ticket/GH11163Test.php diff --git a/src/UnitOfWork.php b/src/UnitOfWork.php index 25b88221583..609bea433f8 100644 --- a/src/UnitOfWork.php +++ b/src/UnitOfWork.php @@ -3224,7 +3224,13 @@ public function triggerEagerLoads() * * @param PersistentCollection[] $collections * @param array $mapping - * @psalm-param array{targetEntity: class-string, sourceEntity: class-string, mappedBy: string, indexBy: string|null} $mapping + * @psalm-param array{ + * targetEntity: class-string, + * sourceEntity: class-string, + * mappedBy: string, + * indexBy: string|null, + * orderBy: array|null + * } $mapping */ private function eagerLoadCollections(array $collections, array $mapping): void { @@ -3241,7 +3247,7 @@ private function eagerLoadCollections(array $collections, array $mapping): void $entities[] = $collection->getOwner(); } - $found = $this->getEntityPersister($targetEntity)->loadAll([$mappedBy => $entities]); + $found = $this->getEntityPersister($targetEntity)->loadAll([$mappedBy => $entities], $mapping['orderBy'] ?? null); $targetClass = $this->em->getClassMetadata($targetEntity); $targetProperty = $targetClass->getReflectionProperty($mappedBy); diff --git a/tests/Tests/ORM/Functional/Ticket/GH11163Test.php b/tests/Tests/ORM/Functional/Ticket/GH11163Test.php new file mode 100644 index 00000000000..927457840c5 --- /dev/null +++ b/tests/Tests/ORM/Functional/Ticket/GH11163Test.php @@ -0,0 +1,135 @@ +setUpEntitySchema([ + GH11163Bucket::class, + GH11163BucketItem::class, + ]); + } + + public function tearDown(): void + { + parent::tearDown(); + + $conn = static::$sharedConn; + $conn->executeStatement('DELETE FROM GH11163BucketItem'); + $conn->executeStatement('DELETE FROM GH11163Bucket'); + } + + public function testFetchEagerModeWithOrderBy(): void + { + // Load entities into database + $this->_em->persist($bucket = new GH11163Bucket(11163)); + $this->_em->persist(new GH11163BucketItem(1, $bucket, 2)); + $this->_em->persist(new GH11163BucketItem(2, $bucket, 3)); + $this->_em->persist(new GH11163BucketItem(3, $bucket, 1)); + $this->_em->flush(); + $this->_em->clear(); + + // Fetch entity from database + $dql = 'SELECT bucket FROM ' . GH11163Bucket::class . ' bucket WHERE bucket.id = :id'; + $bucket = $this->_em->createQuery($dql) + ->setParameter('id', 11163) + ->getSingleResult(); + + // Assert associated entity is loaded eagerly + static::assertInstanceOf(GH11163Bucket::class, $bucket); + static::assertInstanceOf(PersistentCollection::class, $bucket->items); + static::assertTrue($bucket->items->isInitialized()); + + static::assertCount(3, $bucket->items); + + // Assert order of entities + static::assertSame(1, $bucket->items[0]->position); + static::assertSame(3, $bucket->items[0]->id); + + static::assertSame(2, $bucket->items[1]->position); + static::assertSame(1, $bucket->items[1]->id); + + static::assertSame(3, $bucket->items[2]->position); + static::assertSame(2, $bucket->items[2]->id); + } +} + +/** + * @ORM\Entity + */ +class GH11163Bucket +{ + /** + * @ORM\Id + * @ORM\Column(type="integer") + * + * @var int + */ + private $id; + + /** + * @ORM\OneToMany( + * targetEntity=GH11163BucketItem::class, + * mappedBy="bucket", + * fetch="EAGER" + * ) + * @ORM\OrderBy({"position" = "ASC"}) + * + * @var Collection + */ + public $items; + + public function __construct(int $id) + { + $this->id = $id; + $this->items = new ArrayCollection(); + } +} + +/** + * @ORM\Entity + */ +class GH11163BucketItem +{ + /** + * @ORM\ManyToOne(targetEntity=GH11163Bucket::class, inversedBy="items") + * @ORM\JoinColumn(nullable=false) + * + * @var GH11163Bucket + */ + private $bucket; + + /** + * @ORM\Id + * @ORM\Column(type="integer") + * + * @var int + */ + public $id; + + /** + * @ORM\Column(type="integer", nullable=false) + * + * @var int + */ + public $position; + + public function __construct(int $id, GH11163Bucket $bucket, int $position) + { + $this->id = $id; + $this->bucket = $bucket; + $this->position = $position; + } +} From cbec236e8b1168c7b60e6c2a223333bebbae594c Mon Sep 17 00:00:00 2001 From: Simon Podlipsky Date: Thu, 25 Apr 2024 10:32:40 +0200 Subject: [PATCH 03/15] fix: always cleanup in `AbstractHydrator::toIterable()` (#11101) Previously it didn't cleanup anything as long as the iteration hasn't reached the final row. Co-authored-by: Oleg Andreyev --- src/Internal/Hydration/AbstractHydrator.php | 34 ++++++++++--------- .../ORM/Hydration/AbstractHydratorTest.php | 30 ++++++++++++++++ 2 files changed, 48 insertions(+), 16 deletions(-) diff --git a/src/Internal/Hydration/AbstractHydrator.php b/src/Internal/Hydration/AbstractHydrator.php index e9782e6d5fa..d7071cf7f68 100644 --- a/src/Internal/Hydration/AbstractHydrator.php +++ b/src/Internal/Hydration/AbstractHydrator.php @@ -182,29 +182,31 @@ public function toIterable($stmt, ResultSetMapping $resultSetMapping, array $hin $this->prepare(); - while (true) { - $row = $this->statement()->fetchAssociative(); - - if ($row === false) { - $this->cleanup(); + try { + while (true) { + $row = $this->statement()->fetchAssociative(); - break; - } + if ($row === false) { + break; + } - $result = []; + $result = []; - $this->hydrateRowData($row, $result); + $this->hydrateRowData($row, $result); - $this->cleanupAfterRowIteration(); - if (count($result) === 1) { - if (count($resultSetMapping->indexByMap) === 0) { - yield end($result); + $this->cleanupAfterRowIteration(); + if (count($result) === 1) { + if (count($resultSetMapping->indexByMap) === 0) { + yield end($result); + } else { + yield from $result; + } } else { - yield from $result; + yield $result; } - } else { - yield $result; } + } finally { + $this->cleanup(); } } diff --git a/tests/Tests/ORM/Hydration/AbstractHydratorTest.php b/tests/Tests/ORM/Hydration/AbstractHydratorTest.php index 24ababac507..c08c24c0e5c 100644 --- a/tests/Tests/ORM/Hydration/AbstractHydratorTest.php +++ b/tests/Tests/ORM/Hydration/AbstractHydratorTest.php @@ -12,6 +12,7 @@ use Doctrine\ORM\Internal\Hydration\AbstractHydrator; use Doctrine\ORM\ORMException; use Doctrine\ORM\Query\ResultSetMapping; +use Doctrine\Tests\Models\Hydration\SimpleEntity; use Doctrine\Tests\OrmFunctionalTestCase; use PHPUnit\Framework\MockObject\MockObject; @@ -154,4 +155,33 @@ public function testHydrateAllClearsAllAttachedListenersEvenOnError(): void $this->expectException(ORMException::class); $this->hydrator->hydrateAll($this->mockResult, $this->mockResultMapping); } + + public function testToIterableIfYieldAndBreakBeforeFinishAlwaysCleansUp(): void + { + $this->setUpEntitySchema([SimpleEntity::class]); + + $entity1 = new SimpleEntity(); + $this->_em->persist($entity1); + $entity2 = new SimpleEntity(); + $this->_em->persist($entity2); + + $this->_em->flush(); + $this->_em->clear(); + + $evm = $this->_em->getEventManager(); + + $q = $this->_em->createQuery('SELECT e.id FROM ' . SimpleEntity::class . ' e'); + + // select two entities, but do no iterate + $q->toIterable(); + self::assertCount(0, $evm->getListeners(Events::onClear)); + + // select two entities, but abort after first record + foreach ($q->toIterable() as $result) { + self::assertCount(1, $evm->getListeners(Events::onClear)); + break; + } + + self::assertCount(0, $evm->getListeners(Events::onClear)); + } } From 7d3b3f28e9adfda8b03ff896441ba4d12fa9e653 Mon Sep 17 00:00:00 2001 From: Nasimi Mammadov Date: Thu, 25 Apr 2024 16:12:11 -0700 Subject: [PATCH 04/15] Update association-mapping.rst Changed capitalized column names to lowercase for consistency. Other occurances of column names mentioned as lowercase several times at this same page. --- docs/en/reference/association-mapping.rst | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/docs/en/reference/association-mapping.rst b/docs/en/reference/association-mapping.rst index 6280eee0619..55dfe57671e 100644 --- a/docs/en/reference/association-mapping.rst +++ b/docs/en/reference/association-mapping.rst @@ -1300,8 +1300,8 @@ This is essentially the same as the following, more verbose, mapping: * @var Collection */ #[JoinTable(name: 'User_Group')] - #[JoinColumn(name: 'User_id', referencedColumnName: 'id')] - #[InverseJoinColumn(name: 'Group_id', referencedColumnName: 'id')] + #[JoinColumn(name: 'user_id', referencedColumnName: 'id')] + #[InverseJoinColumn(name: 'group_id', referencedColumnName: 'id')] #[ManyToMany(targetEntity: Group::class)] private Collection $groups; // ... @@ -1333,10 +1333,10 @@ This is essentially the same as the following, more verbose, mapping: - + - + From bb36d49b38e1b3f7f198f0672ce4afaf9a07855b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Petr=20Mor=C3=A1vek?= Date: Fri, 26 Apr 2024 14:24:55 +0200 Subject: [PATCH 05/15] Keep entities in identity map until the scheduled deletions are executed. If the entity gets reloaded from database before the deletions are executed UnitOfWork needs to be able to return the original instance in REMOVED state. --- src/UnitOfWork.php | 4 +- .../ORM/Functional/Ticket/GH6123Test.php | 80 +++++++++++++++++++ tests/Tests/ORM/UnitOfWorkTest.php | 8 +- 3 files changed, 89 insertions(+), 3 deletions(-) create mode 100644 tests/Tests/ORM/Functional/Ticket/GH6123Test.php diff --git a/src/UnitOfWork.php b/src/UnitOfWork.php index 609bea433f8..f1affcf7ebc 100644 --- a/src/UnitOfWork.php +++ b/src/UnitOfWork.php @@ -1292,6 +1292,8 @@ private function executeDeletions(): void $eventsToDispatch = []; foreach ($entities as $entity) { + $this->removeFromIdentityMap($entity); + $oid = spl_object_id($entity); $class = $this->em->getClassMetadata(get_class($entity)); $persister = $this->getEntityPersister($class->name); @@ -1667,8 +1669,6 @@ public function scheduleForDelete($entity) return; } - $this->removeFromIdentityMap($entity); - unset($this->entityUpdates[$oid]); if (! isset($this->entityDeletions[$oid])) { diff --git a/tests/Tests/ORM/Functional/Ticket/GH6123Test.php b/tests/Tests/ORM/Functional/Ticket/GH6123Test.php new file mode 100644 index 00000000000..7a5f14980f0 --- /dev/null +++ b/tests/Tests/ORM/Functional/Ticket/GH6123Test.php @@ -0,0 +1,80 @@ +createSchemaForModels( + GH6123Entity::class, + ); + } + + public function testLoadingRemovedEntityFromDatabaseDoesNotCreateNewManagedEntityInstance(): void + { + $entity = new GH6123Entity(); + $this->_em->persist($entity); + $this->_em->flush(); + + self::assertSame(UnitOfWork::STATE_MANAGED, $this->_em->getUnitOfWork()->getEntityState($entity)); + self::assertFalse($this->_em->getUnitOfWork()->isScheduledForDelete($entity)); + + $this->_em->remove($entity); + + $freshEntity = $this->loadEntityFromDatabase($entity->id); + self::assertSame($entity, $freshEntity); + + self::assertSame(UnitOfWork::STATE_REMOVED, $this->_em->getUnitOfWork()->getEntityState($freshEntity)); + self::assertTrue($this->_em->getUnitOfWork()->isScheduledForDelete($freshEntity)); + } + + public function testRemovedEntityCanBePersistedAgain(): void + { + $entity = new GH6123Entity(); + $this->_em->persist($entity); + $this->_em->flush(); + + $this->_em->remove($entity); + self::assertSame(UnitOfWork::STATE_REMOVED, $this->_em->getUnitOfWork()->getEntityState($entity)); + self::assertTrue($this->_em->getUnitOfWork()->isScheduledForDelete($entity)); + + $this->loadEntityFromDatabase($entity->id); + + $this->_em->persist($entity); + self::assertSame(UnitOfWork::STATE_MANAGED, $this->_em->getUnitOfWork()->getEntityState($entity)); + self::assertFalse($this->_em->getUnitOfWork()->isScheduledForDelete($entity)); + + $this->_em->flush(); + } + + private function loadEntityFromDatabase(int $id): GH6123Entity|null + { + return $this->_em->createQueryBuilder() + ->select('e') + ->from(GH6123Entity::class, 'e') + ->where('e.id = :id') + ->setParameter('id', $id) + ->getQuery() + ->getOneOrNullResult(); + } +} + +#[ORM\Entity] +class GH6123Entity +{ + #[ORM\Id] + #[ORM\GeneratedValue] + #[Column(type: Types::INTEGER, nullable: false)] + public int $id; +} diff --git a/tests/Tests/ORM/UnitOfWorkTest.php b/tests/Tests/ORM/UnitOfWorkTest.php index 0569bfd06dd..ee475e729d0 100644 --- a/tests/Tests/ORM/UnitOfWorkTest.php +++ b/tests/Tests/ORM/UnitOfWorkTest.php @@ -413,12 +413,18 @@ public function testRemovedAndRePersistedEntitiesAreInTheIdentityMapAndAreNotGar $entity->id = 123; $this->_unitOfWork->registerManaged($entity, ['id' => 123], []); + self::assertSame(UnitOfWork::STATE_MANAGED, $this->_unitOfWork->getEntityState($entity)); + self::assertFalse($this->_unitOfWork->isScheduledForDelete($entity)); self::assertTrue($this->_unitOfWork->isInIdentityMap($entity)); $this->_unitOfWork->remove($entity); - self::assertFalse($this->_unitOfWork->isInIdentityMap($entity)); + self::assertSame(UnitOfWork::STATE_REMOVED, $this->_unitOfWork->getEntityState($entity)); + self::assertTrue($this->_unitOfWork->isScheduledForDelete($entity)); + self::assertTrue($this->_unitOfWork->isInIdentityMap($entity)); $this->_unitOfWork->persist($entity); + self::assertSame(UnitOfWork::STATE_MANAGED, $this->_unitOfWork->getEntityState($entity)); + self::assertFalse($this->_unitOfWork->isScheduledForDelete($entity)); self::assertTrue($this->_unitOfWork->isInIdentityMap($entity)); } From 4e335f404413a09bdfb33c3d5771005162d22dfa Mon Sep 17 00:00:00 2001 From: Claudio Zizza Date: Sat, 27 Apr 2024 00:04:43 +0200 Subject: [PATCH 06/15] Remove unused test group --- tests/Tests/ORM/Functional/QueryDqlFunctionTest.php | 1 - 1 file changed, 1 deletion(-) diff --git a/tests/Tests/ORM/Functional/QueryDqlFunctionTest.php b/tests/Tests/ORM/Functional/QueryDqlFunctionTest.php index e1ad89ecc2d..f5dead66a31 100644 --- a/tests/Tests/ORM/Functional/QueryDqlFunctionTest.php +++ b/tests/Tests/ORM/Functional/QueryDqlFunctionTest.php @@ -249,7 +249,6 @@ public function testOperatorMultiply(): void self::assertEquals(1600000, $result[3]['op']); } - /** @group test */ public function testOperatorDiv(): void { $result = $this->_em->createQuery('SELECT m, (m.salary/0.5) AS op FROM Doctrine\Tests\Models\Company\CompanyManager m ORDER BY m.salary ASC') From c363f55ad1eca11ade4becebeda5b260c690a685 Mon Sep 17 00:00:00 2001 From: W0rma Date: Mon, 29 Apr 2024 14:35:45 +0200 Subject: [PATCH 07/15] Fix deprecation layer --- src/ORMException.php | 94 +++++++++++++++++++++++--------------------- 1 file changed, 49 insertions(+), 45 deletions(-) diff --git a/src/ORMException.php b/src/ORMException.php index eb5aa6c7b40..898c20ecb0c 100644 --- a/src/ORMException.php +++ b/src/ORMException.php @@ -5,11 +5,31 @@ namespace Doctrine\ORM; use Doctrine\Common\Cache\Cache as CacheDriver; -use Doctrine\Persistence\ObjectRepository; +use Doctrine\ORM\Cache\Exception\InvalidResultCacheDriver; +use Doctrine\ORM\Cache\Exception\MetadataCacheNotConfigured; +use Doctrine\ORM\Cache\Exception\MetadataCacheUsesNonPersistentCache; +use Doctrine\ORM\Cache\Exception\QueryCacheNotConfigured; +use Doctrine\ORM\Cache\Exception\QueryCacheUsesNonPersistentCache; +use Doctrine\ORM\Exception\EntityManagerClosed; +use Doctrine\ORM\Exception\InvalidEntityRepository; +use Doctrine\ORM\Exception\InvalidHydrationMode; +use Doctrine\ORM\Exception\MismatchedEventManager; +use Doctrine\ORM\Exception\MissingIdentifierField; +use Doctrine\ORM\Exception\MissingMappingDriverImplementation; +use Doctrine\ORM\Exception\NamedNativeQueryNotFound; +use Doctrine\ORM\Exception\NamedQueryNotFound; +use Doctrine\ORM\Exception\ProxyClassesAlwaysRegenerating; +use Doctrine\ORM\Exception\UnexpectedAssociationValue; +use Doctrine\ORM\Exception\UnknownEntityNamespace; +use Doctrine\ORM\Exception\UnrecognizedIdentifierFields; +use Doctrine\ORM\Persisters\Exception\CantUseInOperatorOnCompositeKeys; +use Doctrine\ORM\Persisters\Exception\InvalidOrientation; +use Doctrine\ORM\Persisters\Exception\UnrecognizedField; +use Doctrine\ORM\Repository\Exception\InvalidFindByCall; +use Doctrine\ORM\Repository\Exception\InvalidMagicMethodCall; +use Doctrine\ORM\Tools\Exception\NotSupported; use Exception; -use function get_debug_type; -use function implode; use function sprintf; /** @@ -26,8 +46,7 @@ class ORMException extends Exception */ public static function missingMappingDriverImpl() { - return new self("It's a requirement to specify a Metadata Driver and pass it " . - 'to Doctrine\\ORM\\Configuration::setMetadataDriverImpl().'); + return MissingMappingDriverImplementation::create(); } /** @@ -39,11 +58,11 @@ public static function missingMappingDriverImpl() */ public static function namedQueryNotFound($queryName) { - return new self('Could not find a named query by the name "' . $queryName . '"'); + return NamedQueryNotFound::fromName($queryName); } /** - * @deprecated Use Doctrine\ORM\Exception\NamedQueryNotFound + * @deprecated Use Doctrine\ORM\Exception\NamedNativeQueryNotFound * * @param string $nativeQueryName * @@ -51,7 +70,7 @@ public static function namedQueryNotFound($queryName) */ public static function namedNativeQueryNotFound($nativeQueryName) { - return new self('Could not find a named native query by the name "' . $nativeQueryName . '"'); + return NamedNativeQueryNotFound::fromName($nativeQueryName); } /** @@ -63,7 +82,7 @@ public static function namedNativeQueryNotFound($nativeQueryName) */ public static function unrecognizedField($field) { - return new self(sprintf('Unrecognized field: %s', $field)); + return new UnrecognizedField(sprintf('Unrecognized field: %s', $field)); } /** @@ -78,7 +97,7 @@ public static function unrecognizedField($field) */ public static function unexpectedAssociationValue($class, $association, $given, $expected) { - return new self(sprintf('Found entity of type %s on association %s#%s, but expecting %s', $given, $class, $association, $expected)); + return UnexpectedAssociationValue::create($class, $association, $given, $expected); } /** @@ -91,7 +110,7 @@ public static function unexpectedAssociationValue($class, $association, $given, */ public static function invalidOrientation($className, $field) { - return new self('Invalid order by orientation specified for ' . $className . '#' . $field); + return InvalidOrientation::fromClassNameAndField($className, $field); } /** @@ -101,7 +120,7 @@ public static function invalidOrientation($className, $field) */ public static function entityManagerClosed() { - return new self('The EntityManager is closed.'); + return EntityManagerClosed::create(); } /** @@ -113,7 +132,7 @@ public static function entityManagerClosed() */ public static function invalidHydrationMode($mode) { - return new self(sprintf("'%s' is an invalid hydration mode.", $mode)); + return InvalidHydrationMode::fromMode($mode); } /** @@ -123,7 +142,7 @@ public static function invalidHydrationMode($mode) */ public static function mismatchedEventManager() { - return new self('Cannot use different EventManager instances for EntityManager and Connection.'); + return MismatchedEventManager::create(); } /** @@ -135,11 +154,11 @@ public static function mismatchedEventManager() */ public static function findByRequiresParameter($methodName) { - return new self("You need to pass a parameter to '" . $methodName . "'"); + return InvalidMagicMethodCall::onMissingParameter($methodName); } /** - * @deprecated Doctrine\ORM\Repository\Exception\InvalidFindByCall + * @deprecated Doctrine\ORM\Repository\Exception\InvalidMagicMethodCall::becauseFieldNotFoundIn() * * @param string $entityName * @param string $fieldName @@ -149,10 +168,7 @@ public static function findByRequiresParameter($methodName) */ public static function invalidMagicCall($entityName, $fieldName, $method) { - return new self( - "Entity '" . $entityName . "' has no field '" . $fieldName . "'. " . - "You can therefore not call '" . $method . "' on the entities' repository" - ); + return InvalidMagicMethodCall::becauseFieldNotFoundIn($entityName, $fieldName, $method); } /** @@ -165,10 +181,7 @@ public static function invalidMagicCall($entityName, $fieldName, $method) */ public static function invalidFindByInverseAssociation($entityName, $associationFieldName) { - return new self( - "You cannot search for the association field '" . $entityName . '#' . $associationFieldName . "', " . - 'because it is the inverse side of an association. Find methods only work on owning side associations.' - ); + return InvalidFindByCall::fromInverseSideUsage($entityName, $associationFieldName); } /** @@ -178,7 +191,7 @@ public static function invalidFindByInverseAssociation($entityName, $association */ public static function invalidResultCacheDriver() { - return new self('Invalid result cache driver; it must implement Doctrine\\Common\\Cache\\Cache.'); + return InvalidResultCacheDriver::create(); } /** @@ -188,7 +201,7 @@ public static function invalidResultCacheDriver() */ public static function notSupported() { - return new self('This behaviour is (currently) not supported by Doctrine 2'); + return NotSupported::create(); } /** @@ -198,7 +211,7 @@ public static function notSupported() */ public static function queryCacheNotConfigured() { - return new self('Query Cache is not configured.'); + return QueryCacheNotConfigured::create(); } /** @@ -208,7 +221,7 @@ public static function queryCacheNotConfigured() */ public static function metadataCacheNotConfigured() { - return new self('Class Metadata Cache is not configured.'); + return MetadataCacheNotConfigured::create(); } /** @@ -218,7 +231,7 @@ public static function metadataCacheNotConfigured() */ public static function queryCacheUsesNonPersistentCache(CacheDriver $cache) { - return new self('Query Cache uses a non-persistent cache driver, ' . get_debug_type($cache) . '.'); + return QueryCacheUsesNonPersistentCache::fromDriver($cache); } /** @@ -228,7 +241,7 @@ public static function queryCacheUsesNonPersistentCache(CacheDriver $cache) */ public static function metadataCacheUsesNonPersistentCache(CacheDriver $cache) { - return new self('Metadata Cache uses a non-persistent cache driver, ' . get_debug_type($cache) . '.'); + return MetadataCacheUsesNonPersistentCache::fromDriver($cache); } /** @@ -238,7 +251,7 @@ public static function metadataCacheUsesNonPersistentCache(CacheDriver $cache) */ public static function proxyClassesAlwaysRegenerating() { - return new self('Proxy Classes are always regenerating.'); + return ProxyClassesAlwaysRegenerating::create(); } /** @@ -250,9 +263,7 @@ public static function proxyClassesAlwaysRegenerating() */ public static function unknownEntityNamespace($entityNamespaceAlias) { - return new self( - sprintf("Unknown Entity namespace alias '%s'.", $entityNamespaceAlias) - ); + return UnknownEntityNamespace::fromNamespaceAlias($entityNamespaceAlias); } /** @@ -264,11 +275,7 @@ public static function unknownEntityNamespace($entityNamespaceAlias) */ public static function invalidEntityRepository($className) { - return new self(sprintf( - "Invalid repository class '%s'. It must be a %s.", - $className, - ObjectRepository::class - )); + return InvalidEntityRepository::fromClassName($className); } /** @@ -281,7 +288,7 @@ public static function invalidEntityRepository($className) */ public static function missingIdentifierField($className, $fieldName) { - return new self(sprintf('The identifier %s is missing for a query of %s', $fieldName, $className)); + return MissingIdentifierField::fromFieldAndClass($fieldName, $className); } /** @@ -294,10 +301,7 @@ public static function missingIdentifierField($className, $fieldName) */ public static function unrecognizedIdentifierFields($className, $fieldNames) { - return new self( - "Unrecognized identifier fields: '" . implode("', '", $fieldNames) . "' " . - "are not present on class '" . $className . "'." - ); + return UnrecognizedIdentifierFields::fromClassAndFieldNames($className, $fieldNames); } /** @@ -307,6 +311,6 @@ public static function unrecognizedIdentifierFields($className, $fieldNames) */ public static function cantUseInOperatorOnCompositeKeys() { - return new self("Can't use IN operator on entities that have composite keys."); + return CantUseInOperatorOnCompositeKeys::create(); } } From ad5c8e4bdc93e422a284c1cdc0bf6fa6d8f4d7b3 Mon Sep 17 00:00:00 2001 From: W0rma Date: Mon, 29 Apr 2024 14:52:02 +0200 Subject: [PATCH 08/15] Make test compatible with PHP 7.1 --- .../Tests/ORM/Functional/Ticket/GH6123Test.php | 18 +++++++++++++----- 1 file changed, 13 insertions(+), 5 deletions(-) diff --git a/tests/Tests/ORM/Functional/Ticket/GH6123Test.php b/tests/Tests/ORM/Functional/Ticket/GH6123Test.php index 7a5f14980f0..05a1a765f27 100644 --- a/tests/Tests/ORM/Functional/Ticket/GH6123Test.php +++ b/tests/Tests/ORM/Functional/Ticket/GH6123Test.php @@ -6,7 +6,6 @@ use Doctrine\DBAL\Types\Types; use Doctrine\ORM\Mapping as ORM; -use Doctrine\ORM\Mapping\Column; use Doctrine\ORM\UnitOfWork; use Doctrine\Tests\OrmFunctionalTestCase; @@ -17,7 +16,7 @@ protected function setUp(): void parent::setUp(); $this->createSchemaForModels( - GH6123Entity::class, + GH6123Entity::class ); } @@ -58,7 +57,7 @@ public function testRemovedEntityCanBePersistedAgain(): void $this->_em->flush(); } - private function loadEntityFromDatabase(int $id): GH6123Entity|null + private function loadEntityFromDatabase(int $id): ?GH6123Entity { return $this->_em->createQueryBuilder() ->select('e') @@ -70,11 +69,20 @@ private function loadEntityFromDatabase(int $id): GH6123Entity|null } } +/** + * @ORM\Entity + */ #[ORM\Entity] class GH6123Entity { + /** + * @ORM\Id + * @ORM\GeneratedValue + * @ORM\Column(type="integer") + * @var int + */ #[ORM\Id] #[ORM\GeneratedValue] - #[Column(type: Types::INTEGER, nullable: false)] - public int $id; + #[ORM\Column(type: Types::INTEGER)] + public $id; } From 0e26e3ed503e773b05e445557b80508538d9b215 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Gr=C3=A9goire=20Paris?= Date: Sun, 5 May 2024 22:39:45 +0200 Subject: [PATCH 09/15] Setup Dependabot Targeting 2.19.x, since we want the updates to bubble up. Since Dependabot has had no effect on doctrine/dbal yet, I suppose that means that "dependabot.yml" must be present on the default branch. --- .github/dependabot.yml | 9 +++++++++ 1 file changed, 9 insertions(+) create mode 100644 .github/dependabot.yml diff --git a/.github/dependabot.yml b/.github/dependabot.yml new file mode 100644 index 00000000000..c704d7d05b4 --- /dev/null +++ b/.github/dependabot.yml @@ -0,0 +1,9 @@ +version: 2 +updates: + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "weekly" + labels: + - "CI" + target-branch: "2.19.x" From e6bb4ef20e2b0e8bdc97cdbc473d1f016cee437a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Gr=C3=A9goire=20Paris?= Date: Sun, 5 May 2024 22:43:51 +0200 Subject: [PATCH 10/15] Upgrade codecov/codecov-action --- .github/workflows/continuous-integration.yml | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/.github/workflows/continuous-integration.yml b/.github/workflows/continuous-integration.yml index ab98943cc3d..1639b09cd9a 100644 --- a/.github/workflows/continuous-integration.yml +++ b/.github/workflows/continuous-integration.yml @@ -377,6 +377,8 @@ jobs: path: "reports" - name: "Upload to Codecov" - uses: "codecov/codecov-action@v3" + uses: "codecov/codecov-action@v4" with: directory: reports + env: + CODECOV_TOKEN: "${{ secrets.CODECOV_TOKEN }}" From f26b3b9cf9af9c0866a0916d35d0ee1c5372a264 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sun, 5 May 2024 21:17:24 +0000 Subject: [PATCH 11/15] Bump doctrine/.github from 3.0.0 to 5.0.1 Bumps [doctrine/.github](https://github.com/doctrine/.github) from 3.0.0 to 5.0.1. - [Release notes](https://github.com/doctrine/.github/releases) - [Commits](https://github.com/doctrine/.github/compare/3.0.0...5.0.1) --- updated-dependencies: - dependency-name: doctrine/.github dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/coding-standards.yml | 2 +- .github/workflows/release-on-milestone-closed.yml | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/.github/workflows/coding-standards.yml b/.github/workflows/coding-standards.yml index ac2788b39a4..659da17bac6 100644 --- a/.github/workflows/coding-standards.yml +++ b/.github/workflows/coding-standards.yml @@ -24,4 +24,4 @@ on: jobs: coding-standards: - uses: "doctrine/.github/.github/workflows/coding-standards.yml@3.0.0" + uses: "doctrine/.github/.github/workflows/coding-standards.yml@5.0.1" diff --git a/.github/workflows/release-on-milestone-closed.yml b/.github/workflows/release-on-milestone-closed.yml index d46dc4c36bb..89d4fe8bf1c 100644 --- a/.github/workflows/release-on-milestone-closed.yml +++ b/.github/workflows/release-on-milestone-closed.yml @@ -7,7 +7,7 @@ on: jobs: release: - uses: "doctrine/.github/.github/workflows/release-on-milestone-closed.yml@4.0.0" + uses: "doctrine/.github/.github/workflows/release-on-milestone-closed.yml@5.0.1" secrets: GIT_AUTHOR_EMAIL: ${{ secrets.GIT_AUTHOR_EMAIL }} GIT_AUTHOR_NAME: ${{ secrets.GIT_AUTHOR_NAME }} From 029ca611f0d06c9eccacc2874d3b138c005fb2ba Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Gr=C3=A9goire=20Paris?= Date: Sun, 5 May 2024 23:38:41 +0200 Subject: [PATCH 12/15] Use ramsey/composer-install in PHPBench workflow (#11444) It will handle caching for us. --- .github/workflows/phpbench.yml | 11 ++--------- 1 file changed, 2 insertions(+), 9 deletions(-) diff --git a/.github/workflows/phpbench.yml b/.github/workflows/phpbench.yml index d98e7fa2158..2a09ec3b18f 100644 --- a/.github/workflows/phpbench.yml +++ b/.github/workflows/phpbench.yml @@ -47,15 +47,8 @@ jobs: coverage: "pcov" ini-values: "zend.assertions=1, apc.enable_cli=1" - - name: "Cache dependencies installed with composer" - uses: "actions/cache@v3" - with: - path: "~/.composer/cache" - key: "php-${{ matrix.php-version }}-composer-locked-${{ hashFiles('composer.lock') }}" - restore-keys: "php-${{ matrix.php-version }}-composer-locked-" - - - name: "Install dependencies with composer" - run: "composer update --no-interaction --no-progress" + - name: "Install dependencies with Composer" + uses: "ramsey/composer-install@v3" - name: "Run PHPBench" run: "vendor/bin/phpbench run --report=default" From c5291b4de8b3d256433c9f247e8b47baa690ac9e Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Sun, 5 May 2024 23:47:43 +0200 Subject: [PATCH 13/15] Bump ramsey/composer-install from 2 to 3 (#11442) Bumps [ramsey/composer-install](https://github.com/ramsey/composer-install) from 2 to 3. - [Release notes](https://github.com/ramsey/composer-install/releases) - [Commits](https://github.com/ramsey/composer-install/compare/v2...v3) --- updated-dependencies: - dependency-name: ramsey/composer-install dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/continuous-integration.yml | 10 +++++----- .github/workflows/documentation.yml | 2 +- .github/workflows/static-analysis.yml | 4 ++-- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/.github/workflows/continuous-integration.yml b/.github/workflows/continuous-integration.yml index 1639b09cd9a..47453fdae52 100644 --- a/.github/workflows/continuous-integration.yml +++ b/.github/workflows/continuous-integration.yml @@ -80,7 +80,7 @@ jobs: if: "${{ matrix.dbal-version != 'default' }}" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: composer-options: "--ignore-platform-req=php+" @@ -162,7 +162,7 @@ jobs: if: "${{ matrix.dbal-version != 'default' }}" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: composer-options: "--ignore-platform-req=php+" @@ -232,7 +232,7 @@ jobs: extensions: "${{ matrix.extension }}" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: composer-options: "--ignore-platform-req=php+" @@ -302,7 +302,7 @@ jobs: if: "${{ matrix.dbal-version != 'default' }}" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: composer-options: "--ignore-platform-req=php+" @@ -348,7 +348,7 @@ jobs: ini-values: "zend.assertions=1, apc.enable_cli=1" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: dependency-versions: "${{ matrix.deps }}" diff --git a/.github/workflows/documentation.yml b/.github/workflows/documentation.yml index ef8053a211c..65cbad613b2 100644 --- a/.github/workflows/documentation.yml +++ b/.github/workflows/documentation.yml @@ -36,7 +36,7 @@ jobs: run: "composer require --dev phpdocumentor/guides-cli --no-update" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: dependency-versions: "highest" diff --git a/.github/workflows/static-analysis.yml b/.github/workflows/static-analysis.yml index a8765e23299..6ed391c70ba 100644 --- a/.github/workflows/static-analysis.yml +++ b/.github/workflows/static-analysis.yml @@ -58,7 +58,7 @@ jobs: run: "composer require doctrine/persistence ^$([ ${{ matrix.persistence-version }} = default ] && echo '3.1' || echo ${{ matrix.persistence-version }}) --no-update" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: dependency-versions: "highest" @@ -95,7 +95,7 @@ jobs: run: "composer require doctrine/persistence ^3.1 --no-update" - name: "Install dependencies with Composer" - uses: "ramsey/composer-install@v2" + uses: "ramsey/composer-install@v3" with: dependency-versions: "highest" From e83d8a80baa917a2018efa93f5fc70f4529c8a7e Mon Sep 17 00:00:00 2001 From: Alexey Prohorov Date: Wed, 15 May 2024 10:42:04 +0300 Subject: [PATCH 14/15] Using an integer as discriminator value with ORM v3 This fixes a bug that occurred when configuring integers as discriminator values. Doctrine throws a type error whenever the application generates queries. --- src/Query/SqlWalker.php | 9 +- .../ORM/Functional/Ticket/GH11341Test.php | 131 ++++++++++++++++++ 2 files changed, 138 insertions(+), 2 deletions(-) create mode 100644 tests/Tests/ORM/Functional/Ticket/GH11341Test.php diff --git a/src/Query/SqlWalker.php b/src/Query/SqlWalker.php index 018c2455e49..7fdc9a6f9c1 100644 --- a/src/Query/SqlWalker.php +++ b/src/Query/SqlWalker.php @@ -30,6 +30,7 @@ use function implode; use function is_array; use function is_float; +use function is_int; use function is_numeric; use function is_string; use function preg_match; @@ -384,7 +385,9 @@ private function generateDiscriminatorColumnConditionSQL(array $dqlAliases): str $values = []; if ($class->discriminatorValue !== null) { // discriminators can be 0 - $values[] = $conn->quote($class->discriminatorValue); + $values[] = $class->getDiscriminatorColumn()->type === 'integer' && is_int($class->discriminatorValue) + ? $class->discriminatorValue + : $conn->quote((string) $class->discriminatorValue); } foreach ($class->subClasses as $subclassName) { @@ -396,7 +399,9 @@ private function generateDiscriminatorColumnConditionSQL(array $dqlAliases): str continue; } - $values[] = $conn->quote((string) $subclassMetadata->discriminatorValue); + $values[] = $subclassMetadata->getDiscriminatorColumn()->type === 'integer' && is_int($subclassMetadata->discriminatorValue) + ? $subclassMetadata->discriminatorValue + : $conn->quote((string) $subclassMetadata->discriminatorValue); } if ($values !== []) { diff --git a/tests/Tests/ORM/Functional/Ticket/GH11341Test.php b/tests/Tests/ORM/Functional/Ticket/GH11341Test.php new file mode 100644 index 00000000000..16853418ccf --- /dev/null +++ b/tests/Tests/ORM/Functional/Ticket/GH11341Test.php @@ -0,0 +1,131 @@ +setUpEntitySchema([ + IntegerBaseClass::class, + IntegerFooEntity::class, + IntegerBarEntity::class, + StringAsIntBaseClass::class, + StringAsIntFooEntity::class, + StringAsIntBarEntity::class, + StringBaseClass::class, + StringFooEntity::class, + StringBarEntity::class, + ]); + } + + public static function dqlStatements(): Generator + { + yield ['SELECT e FROM ' . IntegerBaseClass::class . ' e', '/WHERE [a-z]0_.type IN \(1, 2\)$/']; + yield ['SELECT e FROM ' . IntegerFooEntity::class . ' e', '/WHERE [a-z]0_.type IN \(1\)$/']; + yield ['SELECT e FROM ' . IntegerBarEntity::class . ' e', '/WHERE [a-z]0_.type IN \(2\)$/']; + yield ['SELECT e FROM ' . StringAsIntBaseClass::class . ' e', '/WHERE [a-z]0_.type IN \(\'1\', \'2\'\)$/']; + yield ['SELECT e FROM ' . StringAsIntFooEntity::class . ' e', '/WHERE [a-z]0_.type IN \(\'1\'\)$/']; + yield ['SELECT e FROM ' . StringAsIntBarEntity::class . ' e', '/WHERE [a-z]0_.type IN \(\'2\'\)$/']; + yield ['SELECT e FROM ' . StringBaseClass::class . ' e', '/WHERE [a-z]0_.type IN \(\'1\', \'2\'\)$/']; + yield ['SELECT e FROM ' . StringFooEntity::class . ' e', '/WHERE [a-z]0_.type IN \(\'1\'\)$/']; + yield ['SELECT e FROM ' . StringBarEntity::class . ' e', '/WHERE [a-z]0_.type IN \(\'2\'\)$/']; + } + + #[DataProvider('dqlStatements')] + public function testDiscriminatorValue(string $dql, string $expectedDiscriminatorValues): void + { + $query = $this->_em->createQuery($dql); + $sql = $query->getSQL(); + + self::assertMatchesRegularExpression($expectedDiscriminatorValues, $sql); + } +} + +#[ORM\Entity] +#[ORM\Table(name: 'integer_discriminator')] +#[ORM\InheritanceType('SINGLE_TABLE')] +#[ORM\DiscriminatorColumn(name: 'type', type: 'integer')] +#[ORM\DiscriminatorMap([ + 1 => IntegerFooEntity::class, + 2 => IntegerBarEntity::class, +])] +class IntegerBaseClass +{ + #[ORM\Id] + #[ORM\GeneratedValue(strategy: 'IDENTITY')] + #[ORM\Column(type: 'integer')] + private int|null $id = null; +} + +#[ORM\Entity] +class IntegerFooEntity extends IntegerBaseClass +{ +} + +#[ORM\Entity] +class IntegerBarEntity extends IntegerBaseClass +{ +} + +#[ORM\Entity] +#[ORM\Table(name: 'string_as_int_discriminator')] +#[ORM\InheritanceType('SINGLE_TABLE')] +#[ORM\DiscriminatorColumn(name: 'type', type: 'string')] +#[ORM\DiscriminatorMap([ + 1 => StringAsIntFooEntity::class, + 2 => StringAsIntBarEntity::class, +])] +class StringAsIntBaseClass +{ + #[ORM\Id] + #[ORM\GeneratedValue(strategy: 'IDENTITY')] + #[ORM\Column(type: 'integer')] + private int|null $id = null; +} + +#[ORM\Entity] +class StringAsIntFooEntity extends StringAsIntBaseClass +{ +} + +#[ORM\Entity] +class StringAsIntBarEntity extends StringAsIntBaseClass +{ +} + + +#[ORM\Entity] +#[ORM\Table(name: 'string_discriminator')] +#[ORM\InheritanceType('SINGLE_TABLE')] +#[ORM\DiscriminatorColumn(name: 'type', type: 'string')] +#[ORM\DiscriminatorMap([ + '1' => StringFooEntity::class, + '2' => StringBarEntity::class, +])] +class StringBaseClass +{ + #[ORM\Id] + #[ORM\GeneratedValue(strategy: 'IDENTITY')] + #[ORM\Column(type: 'integer')] + private int|null $id = null; +} + +#[ORM\Entity] +class StringFooEntity extends StringBaseClass +{ +} + +#[ORM\Entity] +class StringBarEntity extends StringBaseClass +{ +} From 2b04cc2e3f7a1ca8a2e8fa4701c3b3d9b5cfc942 Mon Sep 17 00:00:00 2001 From: Alexey Prohorov Date: Thu, 16 May 2024 11:53:29 +0300 Subject: [PATCH 15/15] Using an integer as discriminator value with ORM v3 This fixes a bug that occurred when configuring integers as discriminator values and using DQL instanceOf function in the queries. Doctrine throws a type error whenever the application generates these queries. --- src/Query/SqlWalker.php | 6 +++-- .../ORM/Functional/Ticket/GH11341Test.php | 26 +++++++++++++++++++ 2 files changed, 30 insertions(+), 2 deletions(-) diff --git a/src/Query/SqlWalker.php b/src/Query/SqlWalker.php index 7fdc9a6f9c1..004d29e773c 100644 --- a/src/Query/SqlWalker.php +++ b/src/Query/SqlWalker.php @@ -2251,8 +2251,10 @@ private function getChildDiscriminatorsFromClassMetadata( $discriminators += HierarchyDiscriminatorResolver::resolveDiscriminatorsForClass($metadata, $this->em); } - foreach (array_keys($discriminators) as $dis) { - $sqlParameterList[] = $this->conn->quote($dis); + foreach (array_keys($discriminators) as $discriminatorValue) { + $sqlParameterList[] = $rootClass->getDiscriminatorColumn()->type === 'integer' && is_int($discriminatorValue) + ? $discriminatorValue + : $this->conn->quote((string) $discriminatorValue); } return '(' . implode(', ', $sqlParameterList) . ')'; diff --git a/tests/Tests/ORM/Functional/Ticket/GH11341Test.php b/tests/Tests/ORM/Functional/Ticket/GH11341Test.php index 16853418ccf..5c35dfe86c3 100644 --- a/tests/Tests/ORM/Functional/Ticket/GH11341Test.php +++ b/tests/Tests/ORM/Functional/Ticket/GH11341Test.php @@ -49,6 +49,32 @@ public function testDiscriminatorValue(string $dql, string $expectedDiscriminato self::assertMatchesRegularExpression($expectedDiscriminatorValues, $sql); } + + public static function dqlStatementsForInstanceOf(): Generator + { + yield [IntegerBaseClass::class, IntegerFooEntity::class]; + yield [StringBaseClass::class, StringFooEntity::class]; + yield [StringAsIntBaseClass::class, StringAsIntFooEntity::class]; + } + + /** + * @psalm-param class-string $baseClass + * @psalm-param class-string $inheritedClass + */ + #[DataProvider('dqlStatementsForInstanceOf')] + public function testInstanceOf(string $baseClass, string $inheritedClass): void + { + $this->_em->persist(new $inheritedClass()); + $this->_em->flush(); + + $dql = 'SELECT p FROM ' . $baseClass . ' p WHERE p INSTANCE OF ' . $baseClass; + + $query = $this->_em->createQuery($dql); + $result = $query->getResult(); + + self::assertCount(1, $result); + self::assertContainsOnlyInstancesOf($baseClass, $result); + } } #[ORM\Entity]