diff --git a/docs/en/reference/architecture.rst b/docs/en/reference/architecture.rst index 00d1c419734..7ba2ef774d7 100644 --- a/docs/en/reference/architecture.rst +++ b/docs/en/reference/architecture.rst @@ -18,7 +18,7 @@ well. Requirements ------------ -Doctrine ORM requires a minimum of PHP 7.1. For greatly improved +Doctrine ORM requires a minimum of PHP 8.1. For greatly improved performance it is also recommended that you use APC with PHP. Doctrine ORM Packages diff --git a/docs/en/reference/dql-doctrine-query-language.rst b/docs/en/reference/dql-doctrine-query-language.rst index 80d41f17002..2ad7788b1ee 100644 --- a/docs/en/reference/dql-doctrine-query-language.rst +++ b/docs/en/reference/dql-doctrine-query-language.rst @@ -976,7 +976,7 @@ The Query class --------------- An instance of the ``Doctrine\ORM\Query`` class represents a DQL -query. You create a Query instance be calling +query. You create a Query instance by calling ``EntityManager#createQuery($dql)``, passing the DQL query string. Alternatively you can create an empty ``Query`` instance and invoke ``Query#setDQL($dql)`` afterwards. Here are some examples: @@ -993,58 +993,146 @@ Alternatively you can create an empty ``Query`` instance and invoke $q = $em->createQuery(); $q->setDQL('select u from MyProject\Model\User u'); -Query Result Formats -~~~~~~~~~~~~~~~~~~~~ +Query Result Formats (Hydration Modes) +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +The way in which the SQL result set of a DQL SELECT query is transformed +to PHP is determined by the so-called "hydration mode". + +``getResult()`` +^^^^^^^^^^^^^^^ + +Retrieves a collection of objects. The result is either a plain collection of objects (pure) or an array +where the objects are nested in the result rows (mixed): + +.. code-block:: php + + createQuery('SELECT u FROM User u'); + $users = $query->getResult(); + // same as: + $users = $query->getResult(AbstractQuery::HYDRATE_OBJECT); + +- Objects fetched in a FROM clause are returned as a Set, that means every + object is only ever included in the resulting array once. This is the case + even when using JOIN or GROUP BY in ways that return the same row for an + object multiple times. If the hydrator sees the same object multiple times, + then it makes sure it is only returned once. + +- If an object is already in memory from a previous query of any kind, then + then the previous object is used, even if the database may contain more + recent data. This even happens if the previous object is still an unloaded proxy. + +``getArrayResult()`` +^^^^^^^^^^^^^^^^^^^^ + +Retrieves an array graph (a nested array) for read-only purposes. + +.. note:: + + An array graph can differ from the corresponding object + graph in certain scenarios due to the difference of the identity + semantics between arrays and objects. + +.. code-block:: php + + getArrayResult(); + // same as: + $users = $query->getResult(AbstractQuery::HYDRATE_ARRAY); + +``getScalarResult()`` +^^^^^^^^^^^^^^^^^^^^^ + +Retrieves a flat/rectangular result set of scalar values that can contain duplicate data. The +pure/mixed distinction does not apply. + +.. code-block:: php + + getScalarResult(); + // same as: + $users = $query->getResult(AbstractQuery::HYDRATE_SCALAR); + +Fields from classes are prefixed by the DQL alias in the result. +A query of the kind `SELECT u.name ...` returns a key `u_name` in the result rows. + +``getSingleScalarResult()`` +^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Retrieves a single scalar value from the result returned by the database. If the result contains +more than a single scalar value, a ``NonUniqueResultException`` is thrown. The pure/mixed distinction does not apply. + +.. code-block:: php + + createQuery('SELECT COUNT(u.id) FROM User u'); + $numUsers = $query->getSingleScalarResult(); + // same as: + $numUsers = $query->getResult(AbstractQuery::HYDRATE_SINGLE_SCALAR); + +``getSingleColumnResult()`` +^^^^^^^^^^^^^^^^^^^^^^^^^^^ + +Retrieves an array from a one-dimensional array of scalar values: + +.. code-block:: php + + createQuery('SELECT a.id FROM User u'); + $ids = $query->getSingleColumnResult(); + // same as: + $ids = $query->getResult(AbstractQuery::HYDRATE_SCALAR_COLUMN); + +``getSingleResult()`` +^^^^^^^^^^^^^^^^^^^^^ + +Retrieves a single object. If the result contains more than one object, a ``NonUniqueResultException`` +is thrown. If the result contains no objects, a ``NoResultException`` is thrown. The pure/mixed distinction does not apply. + +``getOneOrNullResult()`` +^^^^^^^^^^^^^^^^^^^^^^^^ + +Retrieves a single object. If the result contains more than one object, a ``NonUniqueResultException`` +is thrown. If no object is found, ``null`` will be returned. + +Custom Hydration Modes +^^^^^^^^^^^^^^^^^^^^^^ + +You can easily add your own custom hydration modes by first +creating a class which extends ``AbstractHydrator``: + +.. code-block:: php + + _stmt->fetchAllAssociative(); + } + } -The format in which the result of a DQL SELECT query is returned -can be influenced by a so-called ``hydration mode``. A hydration -mode specifies a particular way in which a SQL result set is -transformed. Each hydration mode has its own dedicated method on -the Query class. Here they are: - - -- ``Query#getResult()``: Retrieves a collection of objects. The - result is either a plain collection of objects (pure) or an array - where the objects are nested in the result rows (mixed). -- ``Query#getSingleResult()``: Retrieves a single object. If the - result contains more than one object, an ``NonUniqueResultException`` - is thrown. If the result contains no objects, an ``NoResultException`` - is thrown. The pure/mixed distinction does not apply. -- ``Query#getOneOrNullResult()``: Retrieve a single object. If the - result contains more than one object, a ``NonUniqueResultException`` - is thrown. If no object is found null will be returned. -- ``Query#getArrayResult()``: Retrieves an array graph (a nested - array) that is largely interchangeable with the object graph - generated by ``Query#getResult()`` for read-only purposes. - - .. note:: - - An array graph can differ from the corresponding object - graph in certain scenarios due to the difference of the identity - semantics between arrays and objects. - - - -- ``Query#getScalarResult()``: Retrieves a flat/rectangular result - set of scalar values that can contain duplicate data. The - pure/mixed distinction does not apply. -- ``Query#getSingleScalarResult()``: Retrieves a single scalar - value from the result returned by the dbms. If the result contains - more than a single scalar value, an exception is thrown. The - pure/mixed distinction does not apply. - -Instead of using these methods, you can alternatively use the -general-purpose method -``Query#execute(array $params = [], $hydrationMode = Query::HYDRATE_OBJECT)``. -Using this method you can directly supply the hydration mode as the -second parameter via one of the Query constants. In fact, the -methods mentioned earlier are just convenient shortcuts for the -execute method. For example, the method ``Query#getResult()`` -internally invokes execute, passing in ``Query::HYDRATE_OBJECT`` as -the hydration mode. - -The use of the methods mentioned earlier is generally preferred as -it leads to more concise code. +Next you just need to add the class to the ORM configuration: + +.. code-block:: php + + getConfiguration()->addCustomHydrationMode('CustomHydrator', 'MyProject\Hydrators\CustomHydrator'); + +Now the hydrator is ready to be used in your queries: + +.. code-block:: php + + createQuery('SELECT u FROM CmsUser u'); + $results = $query->getResult('CustomHydrator'); Pure and Mixed Results ~~~~~~~~~~~~~~~~~~~~~~ @@ -1148,165 +1236,6 @@ will return the rows iterating the different top-level entities. [2] => Object (User) [3] => Object (Group) - -Hydration Modes -~~~~~~~~~~~~~~~ - -Each of the Hydration Modes makes assumptions about how the result -is returned to user land. You should know about all the details to -make best use of the different result formats: - -The constants for the different hydration modes are: - - -- ``Query::HYDRATE_OBJECT`` -- ``Query::HYDRATE_ARRAY`` -- ``Query::HYDRATE_SCALAR`` -- ``Query::HYDRATE_SINGLE_SCALAR`` -- ``Query::HYDRATE_SCALAR_COLUMN`` - -Object Hydration -^^^^^^^^^^^^^^^^ - -Object hydration hydrates the result set into the object graph: - -.. code-block:: php - - createQuery('SELECT u FROM CmsUser u'); - $users = $query->getResult(Query::HYDRATE_OBJECT); - -Sometimes the behavior in the object hydrator can be confusing, which is -why we are listing as many of the assumptions here for reference: - -- Objects fetched in a FROM clause are returned as a Set, that means every - object is only ever included in the resulting array once. This is the case - even when using JOIN or GROUP BY in ways that return the same row for an - object multiple times. If the hydrator sees the same object multiple times, - then it makes sure it is only returned once. - -- If an object is already in memory from a previous query of any kind, then - then the previous object is used, even if the database may contain more - recent data. Data from the database is discarded. This even happens if the - previous object is still an unloaded proxy. - -This list might be incomplete. - -Array Hydration -^^^^^^^^^^^^^^^ - -You can run the same query with array hydration and the result set -is hydrated into an array that represents the object graph: - -.. code-block:: php - - createQuery('SELECT u FROM CmsUser u'); - $users = $query->getResult(Query::HYDRATE_ARRAY); - -You can use the ``getArrayResult()`` shortcut as well: - -.. code-block:: php - - getArrayResult(); - -Scalar Hydration -^^^^^^^^^^^^^^^^ - -If you want to return a flat rectangular result set instead of an -object graph you can use scalar hydration: - -.. code-block:: php - - createQuery('SELECT u FROM CmsUser u'); - $users = $query->getResult(Query::HYDRATE_SCALAR); - echo $users[0]['u_id']; - -The following assumptions are made about selected fields using -Scalar Hydration: - - -1. Fields from classes are prefixed by the DQL alias in the result. - A query of the kind 'SELECT u.name ..' returns a key 'u_name' in - the result rows. - -Single Scalar Hydration -^^^^^^^^^^^^^^^^^^^^^^^ - -If you have a query which returns just a single scalar value you can use -single scalar hydration: - -.. code-block:: php - - createQuery('SELECT COUNT(a.id) FROM CmsUser u LEFT JOIN u.articles a WHERE u.username = ?1 GROUP BY u.id'); - $query->setParameter(1, 'jwage'); - $numArticles = $query->getResult(Query::HYDRATE_SINGLE_SCALAR); - -You can use the ``getSingleScalarResult()`` shortcut as well: - -.. code-block:: php - - getSingleScalarResult(); - -Scalar Column Hydration -^^^^^^^^^^^^^^^^^^^^^^^ - -If you have a query which returns a one-dimensional array of scalar values -you can use scalar column hydration: - -.. code-block:: php - - createQuery('SELECT a.id FROM CmsUser u'); - $ids = $query->getResult(Query::HYDRATE_SCALAR_COLUMN); - -You can use the ``getSingleColumnResult()`` shortcut as well: - -.. code-block:: php - - getSingleColumnResult(); - -Custom Hydration Modes -^^^^^^^^^^^^^^^^^^^^^^ - -You can easily add your own custom hydration modes by first -creating a class which extends ``AbstractHydrator``: - -.. code-block:: php - - _stmt->fetchAllAssociative(); - } - } - -Next you just need to add the class to the ORM configuration: - -.. code-block:: php - - getConfiguration()->addCustomHydrationMode('CustomHydrator', 'MyProject\Hydrators\CustomHydrator'); - -Now the hydrator is ready to be used in your queries: - -.. code-block:: php - - createQuery('SELECT u FROM CmsUser u'); - $results = $query->getResult('CustomHydrator'); - Iterating Large Result Sets ~~~~~~~~~~~~~~~~~~~~~~~~~~~ diff --git a/docs/en/tutorials/getting-started.rst b/docs/en/tutorials/getting-started.rst index 5e405618bf4..3550daecf78 100644 --- a/docs/en/tutorials/getting-started.rst +++ b/docs/en/tutorials/getting-started.rst @@ -27,7 +27,7 @@ What is Doctrine? ----------------- Doctrine ORM is an `object-relational mapper (ORM) `_ -for PHP 7.1+ that provides transparent persistence for PHP objects. It uses the Data Mapper +for PHP that provides transparent persistence for PHP objects. It uses the Data Mapper pattern at the heart, aiming for a complete separation of your domain/business logic from the persistence in a relational database management system. diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 75e8f8aae5a..7c8ec81ae84 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -115,11 +115,6 @@ parameters: count: 1 path: src/EntityManagerInterface.php - - - message: "#^Template type T of method Doctrine\\\\ORM\\\\EntityManagerInterface\\:\\:getClassMetadata\\(\\) is not referenced in a parameter\\.$#" - count: 1 - path: src/EntityManagerInterface.php - - message: "#^Method Doctrine\\\\ORM\\\\EntityRepository\\:\\:matching\\(\\) should return Doctrine\\\\Common\\\\Collections\\\\AbstractLazyCollection\\&Doctrine\\\\Common\\\\Collections\\\\Selectable\\ but returns Doctrine\\\\ORM\\\\LazyCriteriaCollection\\<\\(int\\|string\\), object\\>\\.$#" count: 1 diff --git a/phpstan-dbal3.neon b/phpstan-dbal3.neon index a223cd4b792..b9ef942c965 100644 --- a/phpstan-dbal3.neon +++ b/phpstan-dbal3.neon @@ -20,10 +20,6 @@ parameters: message: '~^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:getArrayBindingType\(\) never returns .* so it can be removed from the return type\.$~' path: src/Persisters/Entity/BasicEntityPersister.php - - - message: '~^Unreachable statement \- code above always terminates\.$~' - path: src/Mapping/AssociationMapping.php - - '~^Class Doctrine\\DBAL\\Platforms\\SQLitePlatform not found\.$~' # To be removed in 4.0 diff --git a/phpstan.neon b/phpstan.neon index 4ea0a3630a2..d90ec9fe41f 100644 --- a/phpstan.neon +++ b/phpstan.neon @@ -20,10 +20,6 @@ parameters: message: '~^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:getArrayBindingType\(\) never returns .* so it can be removed from the return type\.$~' path: src/Persisters/Entity/BasicEntityPersister.php - - - message: '~^Unreachable statement \- code above always terminates\.$~' - path: src/Mapping/AssociationMapping.php - # Compatibility with DBAL 3 # See https://github.com/doctrine/dbal/pull/3480 - diff --git a/psalm-baseline.xml b/psalm-baseline.xml index 4faaf8ce91d..8d823650d9d 100644 --- a/psalm-baseline.xml +++ b/psalm-baseline.xml @@ -154,14 +154,6 @@ - - - wrapped->getClassMetadata($className)]]> - - - - - @@ -174,11 +166,9 @@ load($sortedId, null, null, [], $lockMode)]]> loadById($sortedId)]]> metadataFactory]]> - metadataFactory->getMetadataFor($className)]]> - getProxyDir()]]> @@ -251,9 +241,6 @@ - - - @@ -298,6 +285,10 @@ columnNames]]> + + + + diff --git a/src/EntityManagerInterface.php b/src/EntityManagerInterface.php index 3f5535233f9..cf3102bb41b 100644 --- a/src/EntityManagerInterface.php +++ b/src/EntityManagerInterface.php @@ -234,7 +234,7 @@ public function hasFilters(): bool; * * @psalm-param string|class-string $className * - * @psalm-return Mapping\ClassMetadata + * @psalm-return ($className is class-string ? Mapping\ClassMetadata : Mapping\ClassMetadata) * * @psalm-template T of object */ diff --git a/src/LazyCriteriaCollection.php b/src/LazyCriteriaCollection.php index 34dd27e3806..ca67914edec 100644 --- a/src/LazyCriteriaCollection.php +++ b/src/LazyCriteriaCollection.php @@ -64,11 +64,11 @@ public function isEmpty(): bool } /** - * {@inheritDoc} - * * Do an optimized search of an element * - * @template TMaybeContained + * @param mixed $element The element to search for. + * + * @return bool TRUE if the collection contains $element, FALSE otherwise. */ public function contains(mixed $element): bool { diff --git a/src/Mapping/AssociationMapping.php b/src/Mapping/AssociationMapping.php index 0253413e8d8..ce7bdb40ab4 100644 --- a/src/Mapping/AssociationMapping.php +++ b/src/Mapping/AssociationMapping.php @@ -27,7 +27,7 @@ abstract class AssociationMapping implements ArrayAccess /** * The fetching strategy to use for the association, usually defaults to FETCH_LAZY. * - * @var ClassMetadata::FETCH_* + * @var ClassMetadata::FETCH_*|null */ public int|null $fetch = null; @@ -96,13 +96,26 @@ final public function __construct( } /** + * @param mixed[] $mappingArray * @psalm-param array{ * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, * joinTable?: mixed[]|null, * type?: int, - * isOwningSide: bool, ...} $mappingArray + * isOwningSide: bool, + * } $mappingArray */ public static function fromMappingArray(array $mappingArray): static { diff --git a/src/Mapping/ClassMetadata.php b/src/Mapping/ClassMetadata.php index 85079ed049d..f58e00e72fe 100644 --- a/src/Mapping/ClassMetadata.php +++ b/src/Mapping/ClassMetadata.php @@ -1152,7 +1152,7 @@ private function validateAndCompleteTypedAssociationMapping(array $mapping): arr * fieldName?: string, * columnName?: string, * id?: bool, - * generated?: int, + * generated?: self::GENERATED_*, * enumType?: class-string, * } $mapping The field mapping to validate & complete. * diff --git a/src/Mapping/EmbeddedClassMapping.php b/src/Mapping/EmbeddedClassMapping.php index c1d464568a8..8fd02c9232c 100644 --- a/src/Mapping/EmbeddedClassMapping.php +++ b/src/Mapping/EmbeddedClassMapping.php @@ -49,10 +49,12 @@ public function __construct(public string $class) /** * @psalm-param array{ - * class: class-string, - * columnPrefix?: false|string|null, - * declaredField?: string|null, - * originalField?: string|null + * class: class-string, + * columnPrefix?: false|string|null, + * declaredField?: string|null, + * originalField?: string|null, + * inherited?: class-string|null, + * declared?: class-string|null, * } $mappingArray */ public static function fromMappingArray(array $mappingArray): self diff --git a/src/Mapping/FieldMapping.php b/src/Mapping/FieldMapping.php index 3e05eba9ec2..4c09196fd8b 100644 --- a/src/Mapping/FieldMapping.php +++ b/src/Mapping/FieldMapping.php @@ -26,7 +26,7 @@ final class FieldMapping implements ArrayAccess public bool|null $notInsertable = null; public bool|null $notUpdatable = null; public string|null $columnDefinition = null; - /** @psalm-var ClassMetadata::GENERATED_* */ + /** @psalm-var ClassMetadata::GENERATED_*|null */ public int|null $generated = null; /** @var class-string|null */ public string|null $enumType = null; @@ -83,7 +83,34 @@ public function __construct( ) { } - /** @param array{type: string, fieldName: string, columnName: string} $mappingArray */ + /** + * @param array $mappingArray + * @psalm-param array{ + * type: string, + * fieldName: string, + * columnName: string, + * length?: int|null, + * id?: bool|null, + * nullable?: bool|null, + * notInsertable?: bool|null, + * notUpdatable?: bool|null, + * columnDefinition?: string|null, + * generated?: ClassMetadata::GENERATED_*|null, + * enumType?: string|null, + * precision?: int|null, + * scale?: int|null, + * unique?: bool|null, + * inherited?: string|null, + * originalClass?: string|null, + * originalField?: string|null, + * quoted?: bool|null, + * declared?: string|null, + * declaredField?: string|null, + * options?: array|null, + * version?: bool|null, + * default?: string|int|null, + * } $mappingArray + */ public static function fromMappingArray(array $mappingArray): self { $mapping = new self( diff --git a/src/Mapping/JoinColumnMapping.php b/src/Mapping/JoinColumnMapping.php index bc469bb6494..172c25699c9 100644 --- a/src/Mapping/JoinColumnMapping.php +++ b/src/Mapping/JoinColumnMapping.php @@ -31,7 +31,17 @@ public function __construct( /** * @param array $mappingArray - * @psalm-param array{name: string, referencedColumnName: string, ...} $mappingArray + * @psalm-param array{ + * name: string, + * referencedColumnName: string, + * unique?: bool|null, + * quoted?: bool|null, + * fieldName?: string|null, + * onDelete?: string|null, + * columnDefinition?: string|null, + * nullable?: bool|null, + * options?: array|null, + * } $mappingArray */ public static function fromMappingArray(array $mappingArray): self { diff --git a/src/Mapping/JoinTableMapping.php b/src/Mapping/JoinTableMapping.php index df46f9722a9..c8b49681e51 100644 --- a/src/Mapping/JoinTableMapping.php +++ b/src/Mapping/JoinTableMapping.php @@ -35,10 +35,10 @@ public function __construct(public string $name) * @param mixed[] $mappingArray * @psalm-param array{ * name: string, - * quoted?: bool, + * quoted?: bool|null, * joinColumns?: mixed[], * inverseJoinColumns?: mixed[], - * schema?: string, + * schema?: string|null, * options?: array * } $mappingArray */ diff --git a/src/Mapping/ManyToManyOwningSideMapping.php b/src/Mapping/ManyToManyOwningSideMapping.php index c7a0eb4ad58..b09d56c7e30 100644 --- a/src/Mapping/ManyToManyOwningSideMapping.php +++ b/src/Mapping/ManyToManyOwningSideMapping.php @@ -41,9 +41,21 @@ public function toArray(): array * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, * joinTable?: mixed[]|null, * type?: int, - * isOwningSide: bool, ...} $mappingArray + * isOwningSide: bool, + * } $mappingArray */ public static function fromMappingArrayAndNamingStrategy(array $mappingArray, NamingStrategy $namingStrategy): self { diff --git a/src/Mapping/OneToManyAssociationMapping.php b/src/Mapping/OneToManyAssociationMapping.php index 804061a1a1d..786e9812774 100644 --- a/src/Mapping/OneToManyAssociationMapping.php +++ b/src/Mapping/OneToManyAssociationMapping.php @@ -12,9 +12,21 @@ final class OneToManyAssociationMapping extends ToManyInverseSideMapping * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, * joinTable?: mixed[]|null, * type?: int, - * isOwningSide: bool, ...} $mappingArray + * isOwningSide: bool, + * } $mappingArray */ public static function fromMappingArray(array $mappingArray): static { @@ -33,9 +45,21 @@ public static function fromMappingArray(array $mappingArray): static * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, * joinTable?: mixed[]|null, * type?: int, - * isOwningSide: bool, ...} $mappingArray + * isOwningSide: bool, + * } $mappingArray */ public static function fromMappingArrayAndName(array $mappingArray, string $name): static { diff --git a/src/Mapping/ToOneInverseSideMapping.php b/src/Mapping/ToOneInverseSideMapping.php index 232952be438..5be89e6db53 100644 --- a/src/Mapping/ToOneInverseSideMapping.php +++ b/src/Mapping/ToOneInverseSideMapping.php @@ -13,8 +13,21 @@ abstract class ToOneInverseSideMapping extends InverseSideMapping * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, + * joinTable?: mixed[]|null, + * type?: int, * isOwningSide: bool, - * } $mappingArray + * } $mappingArray */ public static function fromMappingArrayAndName( array $mappingArray, diff --git a/src/Mapping/ToOneOwningSideMapping.php b/src/Mapping/ToOneOwningSideMapping.php index d28074bb03e..cb85afbde10 100644 --- a/src/Mapping/ToOneOwningSideMapping.php +++ b/src/Mapping/ToOneOwningSideMapping.php @@ -31,8 +31,22 @@ abstract class ToOneOwningSideMapping extends OwningSideMapping implements ToOne * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, + * joinTable?: mixed[]|null, + * type?: int, + * isOwningSide: bool, * joinColumns?: mixed[]|null, - * isOwningSide: bool, ...} $mappingArray + * } $mappingArray */ public static function fromMappingArray(array $mappingArray): static { @@ -64,8 +78,22 @@ public static function fromMappingArray(array $mappingArray): static * fieldName: string, * sourceEntity: class-string, * targetEntity: class-string, + * cascade?: list<'persist'|'remove'|'detach'|'refresh'|'all'>, + * fetch?: ClassMetadata::FETCH_*|null, + * inherited?: class-string|null, + * declared?: class-string|null, + * cache?: array|null, + * id?: bool|null, + * isOnDeleteCascade?: bool|null, + * originalClass?: class-string|null, + * originalField?: string|null, + * orphanRemoval?: bool, + * unique?: bool|null, + * joinTable?: mixed[]|null, + * type?: int, + * isOwningSide: bool, * joinColumns?: mixed[]|null, - * isOwningSide: bool, ...} $mappingArray + * } $mappingArray */ public static function fromMappingArrayAndName( array $mappingArray, diff --git a/src/PersistentCollection.php b/src/PersistentCollection.php index 24cdef0a5de..d54d3d1b997 100644 --- a/src/PersistentCollection.php +++ b/src/PersistentCollection.php @@ -350,11 +350,6 @@ public function containsKey(mixed $key): bool return parent::containsKey($key); } - /** - * {@inheritDoc} - * - * @template TMaybeContained - */ public function contains(mixed $element): bool { if (! $this->initialized && $this->getMapping()->fetch === ClassMetadata::FETCH_EXTRA_LAZY) { diff --git a/src/Tools/SchemaValidator.php b/src/Tools/SchemaValidator.php index 6ebe991876c..fdfc00390c8 100644 --- a/src/Tools/SchemaValidator.php +++ b/src/Tools/SchemaValidator.php @@ -52,18 +52,18 @@ class SchemaValidator * It maps built-in Doctrine types to PHP types */ private const BUILTIN_TYPES_MAP = [ - AsciiStringType::class => 'string', - BigIntType::class => 'string', - BooleanType::class => 'bool', - DecimalType::class => 'string', - FloatType::class => 'float', - GuidType::class => 'string', - IntegerType::class => 'int', - JsonType::class => 'array', - SimpleArrayType::class => 'array', - SmallIntType::class => 'int', - StringType::class => 'string', - TextType::class => 'string', + AsciiStringType::class => ['string'], + BigIntType::class => ['int', 'string'], + BooleanType::class => ['bool'], + DecimalType::class => ['string'], + FloatType::class => ['float'], + GuidType::class => ['string'], + IntegerType::class => ['int'], + JsonType::class => ['array'], + SimpleArrayType::class => ['array'], + SmallIntType::class => ['int'], + StringType::class => ['string'], + TextType::class => ['string'], ]; public function __construct( @@ -353,21 +353,21 @@ function (FieldMapping $fieldMapping) use ($class): string|null { $propertyType = $propertyType->getName(); // If the property type is the same as the metadata field type, we are ok - if ($propertyType === $metadataFieldType) { + if (in_array($propertyType, $metadataFieldType, true)) { return null; } if (is_a($propertyType, BackedEnum::class, true)) { $backingType = (string) (new ReflectionEnum($propertyType))->getBackingType(); - if ($metadataFieldType !== $backingType) { + if (! in_array($backingType, $metadataFieldType, true)) { return sprintf( "The field '%s#%s' has the property type '%s' with a backing type of '%s' that differs from the metadata field type '%s'.", $class->name, $fieldName, $propertyType, $backingType, - $metadataFieldType, + implode('|', $metadataFieldType), ); } @@ -392,7 +392,7 @@ function (FieldMapping $fieldMapping) use ($class): string|null { ) { $backingType = (string) (new ReflectionEnum($fieldMapping->enumType))->getBackingType(); - if ($metadataFieldType === $backingType) { + if (in_array($backingType, $metadataFieldType, true)) { return null; } @@ -402,7 +402,7 @@ function (FieldMapping $fieldMapping) use ($class): string|null { $fieldName, $fieldMapping->enumType, $backingType, - $metadataFieldType, + implode('|', $metadataFieldType), ); } @@ -418,7 +418,7 @@ function (FieldMapping $fieldMapping) use ($class): string|null { $class->name, $fieldName, $propertyType, - $metadataFieldType, + implode('|', $metadataFieldType), $fieldMapping->type, ); }, @@ -431,8 +431,10 @@ function (FieldMapping $fieldMapping) use ($class): string|null { /** * The exact DBAL type must be used (no subclasses), since consumers of doctrine/orm may have their own * customization around field types. + * + * @return list|null */ - private function findBuiltInType(Type $type): string|null + private function findBuiltInType(Type $type): array|null { $typeName = $type::class; diff --git a/tests/Tests/Models/BigIntegers/BigIntegers.php b/tests/Tests/Models/BigIntegers/BigIntegers.php new file mode 100644 index 00000000000..a79f6c2b05d --- /dev/null +++ b/tests/Tests/Models/BigIntegers/BigIntegers.php @@ -0,0 +1,26 @@ +em->getClassMetadata(BigIntegers::class); + + self::assertSame( + ['The field \'Doctrine\Tests\Models\BigIntegers\BigIntegers#three\' has the property type \'float\' that differs from the metadata field type \'int|string\' returned by the \'bigint\' DBAL type.'], + $this->validator->validateClass($class), + ); + } } #[MappedSuperclass]