Skip to content

Commit 9bf407f

Browse files
committed
Fix IN/NOT IN expression handling and support enums when matching on to-many-collections
This fixes that using a `Criteria` with an `IN` or `NIN` expression on a to-many collection currently leads to an SQL error (doctrine#6173). The `ManyToMany` persister needs to know about the slightly different SQL syntax for `[NOT] IN ()`. In the case of `[NOT] IN` expressions, the value will be an array, which also required me to change (I guess "fix") the parameter type handling. I have pulled the necessary code from the `BasicEntityPersister` and placed it as static helper methods in `PersisterHelper`. This is somewhat inspired by doctrine#11516, which aims at fixing doctrine#11481: By re-using the parameter type handling code, it also fixes using backed enums in `EQ`, `IN` and `NIN` expressions within `Criteria` when `matching()` on one-to-many and many-to-many collections.
1 parent bd4a053 commit 9bf407f

10 files changed

+378
-164
lines changed

phpstan-baseline.neon

+18-36
Original file line numberDiff line numberDiff line change
@@ -2742,18 +2742,6 @@ parameters:
27422742
count: 1
27432743
path: src/Persisters/Entity/BasicEntityPersister.php
27442744

2745-
-
2746-
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:expandCriteriaParameters\(\) should return array\{list\<mixed\>, list\<int\|string\|null\>\} but returns array\{array\<mixed\>, list\<int\|string\|null\>\}\.$#'
2747-
identifier: return.type
2748-
count: 1
2749-
path: src/Persisters/Entity/BasicEntityPersister.php
2750-
2751-
-
2752-
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:expandParameters\(\) should return array\{list\<mixed\>, list\<int\|string\|null\>\} but returns array\{array\<mixed\>, list\<int\|string\|null\>\}\.$#'
2753-
identifier: return.type
2754-
count: 1
2755-
path: src/Persisters/Entity/BasicEntityPersister.php
2756-
27572745
-
27582746
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:expandToManyParameters\(\) return type has no value type specified in iterable type array\.$#'
27592747
identifier: missingType.iterableValue
@@ -2790,12 +2778,6 @@ parameters:
27902778
count: 1
27912779
path: src/Persisters/Entity/BasicEntityPersister.php
27922780

2793-
-
2794-
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:getIndividualValue\(\) should return list\<mixed\> but returns array\<mixed\>\.$#'
2795-
identifier: return.type
2796-
count: 1
2797-
path: src/Persisters/Entity/BasicEntityPersister.php
2798-
27992781
-
28002782
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:getManyToManyCollection\(\) has parameter \$assoc with no value type specified in iterable type array\.$#'
28012783
identifier: missingType.iterableValue
@@ -2856,12 +2838,6 @@ parameters:
28562838
count: 5
28572839
path: src/Persisters/Entity/BasicEntityPersister.php
28582840

2859-
-
2860-
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:getTypes\(\) has parameter \$class with generic class Doctrine\\ORM\\Mapping\\ClassMetadata but does not specify its types\: T$#'
2861-
identifier: missingType.generics
2862-
count: 1
2863-
path: src/Persisters/Entity/BasicEntityPersister.php
2864-
28652841
-
28662842
message: '#^Method Doctrine\\ORM\\Persisters\\Entity\\BasicEntityPersister\:\:load\(\) has parameter \$assoc with no value type specified in iterable type array\.$#'
28672843
identifier: missingType.iterableValue
@@ -2922,18 +2898,6 @@ parameters:
29222898
count: 8
29232899
path: src/Persisters/Entity/BasicEntityPersister.php
29242900

2925-
-
2926-
message: '#^Offset ''relationToTargetKeyColumns'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
2927-
identifier: offsetAccess.notFound
2928-
count: 1
2929-
path: src/Persisters/Entity/BasicEntityPersister.php
2930-
2931-
-
2932-
message: '#^Offset ''sourceToTargetKeyColumns'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
2933-
identifier: offsetAccess.notFound
2934-
count: 1
2935-
path: src/Persisters/Entity/BasicEntityPersister.php
2936-
29372901
-
29382902
message: '#^Offset ''targetToSourceKeyColumns'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
29392903
identifier: offsetAccess.notFound
@@ -5589,8 +5553,26 @@ parameters:
55895553
count: 1
55905554
path: src/Utility/PersisterHelper.php
55915555

5556+
-
5557+
message: '#^Method Doctrine\\ORM\\Utility\\PersisterHelper\:\:inferParameterTypes\(\) has parameter \$class with generic class Doctrine\\ORM\\Mapping\\ClassMetadata but does not specify its types\: T$#'
5558+
identifier: missingType.generics
5559+
count: 1
5560+
path: src/Utility/PersisterHelper.php
5561+
55925562
-
55935563
message: '#^Offset ''joinTable'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
55945564
identifier: offsetAccess.notFound
55955565
count: 1
55965566
path: src/Utility/PersisterHelper.php
5567+
5568+
-
5569+
message: '#^Offset ''relationToTargetKeyColumns'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
5570+
identifier: offsetAccess.notFound
5571+
count: 1
5572+
path: src/Utility/PersisterHelper.php
5573+
5574+
-
5575+
message: '#^Offset ''sourceToTargetKeyColumns'' might not exist on array\{cache\?\: array, cascade\: array\<string\>, declared\?\: class\-string, fetch\: mixed, fieldName\: string, id\?\: bool, inherited\?\: class\-string, indexBy\?\: string, \.\.\.\}\.$#'
5576+
identifier: offsetAccess.notFound
5577+
count: 1
5578+
path: src/Utility/PersisterHelper.php

src/Persisters/Collection/ManyToManyPersister.php

+11-3
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
use Doctrine\ORM\Utility\PersisterHelper;
1717

1818
use function array_fill;
19+
use function array_merge;
1920
use function array_pop;
2021
use function count;
2122
use function get_class;
@@ -257,9 +258,16 @@ public function loadCriteria(PersistentCollection $collection, Criteria $criteri
257258
if ($value === null && ($operator === Comparison::EQ || $operator === Comparison::NEQ)) {
258259
$whereClauses[] = sprintf('te.%s %s NULL', $field, $operator === Comparison::EQ ? 'IS' : 'IS NOT');
259260
} else {
260-
$whereClauses[] = sprintf('te.%s %s ?', $field, $operator);
261-
$params[] = $value;
262-
$paramTypes[] = PersisterHelper::getTypeOfField($name, $targetClass, $this->em)[0];
261+
if ($operator === Comparison::IN) {
262+
$whereClauses[] = sprintf('te.%s IN (?)', $field);
263+
} elseif ($operator === Comparison::NIN) {
264+
$whereClauses[] = sprintf('te.%s NOT IN (?)', $field);
265+
} else {
266+
$whereClauses[] = sprintf('te.%s %s ?', $field, $operator);
267+
}
268+
269+
$params = array_merge($params, PersisterHelper::convertToParameterValue($value, $this->em));
270+
$paramTypes = array_merge($paramTypes, PersisterHelper::inferParameterTypes($name, $value, $targetClass, $this->em));
263271
}
264272
}
265273

src/Persisters/Entity/BasicEntityPersister.php

+7-125
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44

55
namespace Doctrine\ORM\Persisters\Entity;
66

7-
use BackedEnum;
87
use Doctrine\Common\Collections\Criteria;
98
use Doctrine\Common\Collections\Expr\Comparison;
109
use Doctrine\DBAL\Connection;
@@ -26,9 +25,7 @@
2625
use Doctrine\ORM\Persisters\Exception\UnrecognizedField;
2726
use Doctrine\ORM\Persisters\SqlExpressionVisitor;
2827
use Doctrine\ORM\Persisters\SqlValueVisitor;
29-
use Doctrine\ORM\Proxy\DefaultProxyClassNameResolver;
3028
use Doctrine\ORM\Query;
31-
use Doctrine\ORM\Query\QueryException;
3229
use Doctrine\ORM\Repository\Exception\InvalidFindByCall;
3330
use Doctrine\ORM\UnitOfWork;
3431
use Doctrine\ORM\Utility\IdentifierFlattener;
@@ -47,7 +44,6 @@
4744
use function count;
4845
use function implode;
4946
use function is_array;
50-
use function is_object;
5147
use function reset;
5248
use function spl_object_id;
5349
use function sprintf;
@@ -392,7 +388,7 @@ final protected function extractIdentifierTypes(array $id, ClassMetadata $versio
392388
$types = [];
393389

394390
foreach ($id as $field => $value) {
395-
$types = array_merge($types, $this->getTypes($field, $value, $versionedClass));
391+
$types = array_merge($types, PersisterHelper::inferParameterTypes($field, $value, $versionedClass, $this->em));
396392
}
397393

398394
return $types;
@@ -953,8 +949,8 @@ public function expandCriteriaParameters(Criteria $criteria)
953949
continue;
954950
}
955951

956-
$sqlParams = array_merge($sqlParams, $this->getValues($value));
957-
$sqlTypes = array_merge($sqlTypes, $this->getTypes($field, $value, $this->class));
952+
$sqlParams = array_merge($sqlParams, PersisterHelper::convertToParameterValue($value, $this->em));
953+
$sqlTypes = array_merge($sqlTypes, PersisterHelper::inferParameterTypes($field, $value, $this->class, $this->em));
958954
}
959955

960956
return [$sqlParams, $sqlTypes];
@@ -1947,8 +1943,8 @@ public function expandParameters($criteria)
19471943
continue; // skip null values.
19481944
}
19491945

1950-
$types = array_merge($types, $this->getTypes($field, $value, $this->class));
1951-
$params = array_merge($params, $this->getValues($value));
1946+
$types = array_merge($types, PersisterHelper::inferParameterTypes($field, $value, $this->class, $this->em));
1947+
$params = array_merge($params, PersisterHelper::convertToParameterValue($value, $this->em));
19521948
}
19531949

19541950
return [$params, $types];
@@ -1976,127 +1972,13 @@ private function expandToManyParameters(array $criteria): array
19761972
continue; // skip null values.
19771973
}
19781974

1979-
$types = array_merge($types, $this->getTypes($criterion['field'], $criterion['value'], $criterion['class']));
1980-
$params = array_merge($params, $this->getValues($criterion['value']));
1975+
$types = array_merge($types, PersisterHelper::inferParameterTypes($criterion['field'], $criterion['value'], $criterion['class'], $this->em));
1976+
$params = array_merge($params, PersisterHelper::convertToParameterValue($criterion['value'], $this->em));
19811977
}
19821978

19831979
return [$params, $types];
19841980
}
19851981

1986-
/**
1987-
* Infers field types to be used by parameter type casting.
1988-
*
1989-
* @param mixed $value
1990-
*
1991-
* @return int[]|null[]|string[]
1992-
* @phpstan-return list<int|string|null>
1993-
*
1994-
* @throws QueryException
1995-
*/
1996-
private function getTypes(string $field, $value, ClassMetadata $class): array
1997-
{
1998-
$types = [];
1999-
2000-
switch (true) {
2001-
case isset($class->fieldMappings[$field]):
2002-
$types = array_merge($types, [$class->fieldMappings[$field]['type']]);
2003-
break;
2004-
2005-
case isset($class->associationMappings[$field]):
2006-
$assoc = $class->associationMappings[$field];
2007-
$class = $this->em->getClassMetadata($assoc['targetEntity']);
2008-
2009-
if (! $assoc['isOwningSide']) {
2010-
$assoc = $class->associationMappings[$assoc['mappedBy']];
2011-
$class = $this->em->getClassMetadata($assoc['targetEntity']);
2012-
}
2013-
2014-
$columns = $assoc['type'] === ClassMetadata::MANY_TO_MANY
2015-
? $assoc['relationToTargetKeyColumns']
2016-
: $assoc['sourceToTargetKeyColumns'];
2017-
2018-
foreach ($columns as $column) {
2019-
$types[] = PersisterHelper::getTypeOfColumn($column, $class, $this->em);
2020-
}
2021-
2022-
break;
2023-
2024-
default:
2025-
$types[] = null;
2026-
break;
2027-
}
2028-
2029-
if (is_array($value)) {
2030-
return array_map(static function ($type) {
2031-
$type = Type::getType($type);
2032-
2033-
return $type->getBindingType() + Connection::ARRAY_PARAM_OFFSET;
2034-
}, $types);
2035-
}
2036-
2037-
return $types;
2038-
}
2039-
2040-
/**
2041-
* Retrieves the parameters that identifies a value.
2042-
*
2043-
* @param mixed $value
2044-
*
2045-
* @return mixed[]
2046-
*/
2047-
private function getValues($value): array
2048-
{
2049-
if (is_array($value)) {
2050-
$newValue = [];
2051-
2052-
foreach ($value as $itemValue) {
2053-
$newValue = array_merge($newValue, $this->getValues($itemValue));
2054-
}
2055-
2056-
return [$newValue];
2057-
}
2058-
2059-
return $this->getIndividualValue($value);
2060-
}
2061-
2062-
/**
2063-
* Retrieves an individual parameter value.
2064-
*
2065-
* @param mixed $value
2066-
*
2067-
* @phpstan-return list<mixed>
2068-
*/
2069-
private function getIndividualValue($value): array
2070-
{
2071-
if (! is_object($value)) {
2072-
return [$value];
2073-
}
2074-
2075-
if ($value instanceof BackedEnum) {
2076-
return [$value->value];
2077-
}
2078-
2079-
$valueClass = DefaultProxyClassNameResolver::getClass($value);
2080-
2081-
if ($this->em->getMetadataFactory()->isTransient($valueClass)) {
2082-
return [$value];
2083-
}
2084-
2085-
$class = $this->em->getClassMetadata($valueClass);
2086-
2087-
if ($class->isIdentifierComposite) {
2088-
$newValue = [];
2089-
2090-
foreach ($class->getIdentifierValues($value) as $innerValue) {
2091-
$newValue = array_merge($newValue, $this->getValues($innerValue));
2092-
}
2093-
2094-
return $newValue;
2095-
}
2096-
2097-
return [$this->em->getUnitOfWork()->getSingleIdentifierValue($value)];
2098-
}
2099-
21001982
/**
21011983
* {@inheritDoc}
21021984
*/

0 commit comments

Comments
 (0)