Skip to content

Commit 5bd227d

Browse files
committed
Fix return type of QueryInterface::execute()
The execute mehtod has a boolean argument that defines if a raw result should be returned. This means that no object mapping is done but an array containing the result values is returned. It is now also supported to not just pass constant boolean values but a variable containing a boolean to the execute function. In this case it's not clear if a raw result or objects should be returned so we do now return a union type containing both.
1 parent a7549f5 commit 5bd227d

File tree

3 files changed

+31
-15
lines changed

3 files changed

+31
-15
lines changed

src/Type/QueryInterfaceDynamicReturnTypeExtension.php

+15-5
Original file line numberDiff line numberDiff line change
@@ -6,14 +6,17 @@
66
use PHPStan\Analyser\Scope;
77
use PHPStan\Reflection\MethodReflection;
88
use PHPStan\Type\ArrayType;
9+
use PHPStan\Type\BooleanType;
910
use PHPStan\Type\Constant\ConstantBooleanType;
1011
use PHPStan\Type\DynamicMethodReturnTypeExtension;
1112
use PHPStan\Type\ErrorType;
1213
use PHPStan\Type\Generic\GenericObjectType;
1314
use PHPStan\Type\IntegerType;
1415
use PHPStan\Type\MixedType;
1516
use PHPStan\Type\ObjectType;
17+
use PHPStan\Type\StringType;
1618
use PHPStan\Type\Type;
19+
use PHPStan\Type\TypeCombinator;
1720
use SaschaEgerer\PhpstanTypo3\Helpers\Typo3ClassNamingUtilityTrait;
1821
use TYPO3\CMS\Extbase\Persistence\Generic\QueryResult;
1922
use TYPO3\CMS\Extbase\Persistence\QueryInterface;
@@ -45,7 +48,6 @@ public function getTypeFromMethodCall(
4548
$argument = $methodCall->getArgs()[0] ?? null;
4649

4750
$classReflection = $scope->getClassReflection();
48-
4951
$queryType = $scope->getType($methodCall->var);
5052
if ($queryType instanceof GenericObjectType) {
5153
$modelType = $queryType->getTypes();
@@ -64,15 +66,23 @@ public function getTypeFromMethodCall(
6466
}
6567
}
6668

69+
$returnType = new GenericObjectType(QueryResult::class, $modelType);
70+
$rawReturnType = new ArrayType(new IntegerType(), new ArrayType(new StringType(), new MixedType()));
71+
6772
if ($argument !== null) {
6873
$argType = $scope->getType($argument->value);
6974

70-
if ($classReflection !== null && $argType instanceof ConstantBooleanType && $argType->getValue() === true) {
71-
return new ArrayType(new IntegerType(), $modelType[0]);
75+
if ($argType instanceof ConstantBooleanType) {
76+
if ($argType->getValue() === true) {
77+
// A static boolean value with "true" has been given
78+
return $rawReturnType;
79+
}
80+
} elseif ($argType instanceof BooleanType) {
81+
// A variable with a boolean value has been given but we don't know it's value
82+
return TypeCombinator::union($rawReturnType, $returnType);
7283
}
7384
}
74-
75-
return new GenericObjectType(QueryResult::class, $modelType);
85+
return $returnType;
7686
}
7787

7888
}

stubs/QueryInterface.stub

-6
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,6 @@ namespace TYPO3\CMS\Extbase\Persistence;
66
*/
77
interface QueryInterface
88
{
9-
/**
10-
* @param bool $returnRawQueryResult
11-
* @return \TYPO3\CMS\Extbase\Persistence\QueryResultInterface<ModelType>|array<string, mixed>
12-
*/
13-
public function execute($returnRawQueryResult = false);
14-
159
/**
1610
* @param mixed $constraint
1711
* @return \TYPO3\CMS\Extbase\Persistence\QueryInterface<ModelType>

tests/Unit/Type/data/custom-query-type.php

+16-4
Original file line numberDiff line numberDiff line change
@@ -28,19 +28,31 @@ class SomeOtherModel extends AbstractEntity
2828
class MyModelRepository extends Repository
2929
{
3030

31-
public function findBySomething(): void
31+
public function findBySomething(bool $booleanParameter = false): void
3232
{
3333
/** @var QueryInterface<SomeOtherModel> $query */
3434
$query = $this->persistenceManager->createQueryForType(SomeOtherModel::class);
3535

3636
$result = $query->execute();
3737
assertType(
38-
'TYPO3\CMS\Extbase\Persistence\QueryInterface<CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>',
39-
$query
38+
'TYPO3\CMS\Extbase\Persistence\Generic\QueryResult<CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>',
39+
$result
40+
);
41+
42+
$result = $query->execute(false);
43+
assertType(
44+
'TYPO3\CMS\Extbase\Persistence\Generic\QueryResult<CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>',
45+
$result
46+
);
47+
48+
$result = $query->execute($booleanParameter);
49+
assertType(
50+
'array<int, array<string, mixed>>|TYPO3\CMS\Extbase\Persistence\Generic\QueryResult<CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>',
51+
$result
4052
);
4153

4254
$rawResult = $query->execute(true);
43-
assertType('array<int, CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>', $rawResult);
55+
assertType('array<int, array<string, mixed>>', $rawResult);
4456

4557
$array = $result->toArray();
4658
assertType('array<int, CustomQueryType\My\Test\Extension\Domain\Model\SomeOtherModel>', $array);

0 commit comments

Comments
 (0)