Skip to content

Commit ea7e0ac

Browse files
committed
Fixed inferring return type coming from array_map
1 parent d97ddee commit ea7e0ac

File tree

5 files changed

+68
-3
lines changed

5 files changed

+68
-3
lines changed

src/Analyser/MutatingScope.php

+20-1
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,7 @@
3737
use PHPStan\Reflection\ParametersAcceptor;
3838
use PHPStan\Reflection\ParametersAcceptorSelector;
3939
use PHPStan\Reflection\PassedByReference;
40+
use PHPStan\Reflection\Php\DummyParameter;
4041
use PHPStan\Reflection\Php\PhpFunctionFromParserNodeReflection;
4142
use PHPStan\Reflection\Php\PhpMethodFromParserNodeReflection;
4243
use PHPStan\Reflection\PropertyReflection;
@@ -1281,7 +1282,25 @@ private function resolveType(Expr $node): Type
12811282
$returnType = TypehintHelper::decideType($this->getFunctionType($node->returnType, false, false), $returnType);
12821283
}
12831284
} else {
1284-
$closureScope = $this->enterAnonymousFunctionWithoutReflection($node);
1285+
$callableParameters = null;
1286+
$arg = $node->getAttribute('parent');
1287+
if ($arg instanceof Arg) {
1288+
$funcCall = $arg->getAttribute('parent');
1289+
$argOrder = $arg->getAttribute('expressionOrder');
1290+
if ($funcCall instanceof FuncCall && $funcCall->name instanceof Name) {
1291+
$functionName = $this->reflectionProvider->resolveFunctionName($funcCall->name, $this);
1292+
if (
1293+
$functionName === 'array_map'
1294+
&& $argOrder === 0
1295+
&& isset($funcCall->args[1])
1296+
) {
1297+
$callableParameters = [
1298+
new DummyParameter('item', $this->getType($funcCall->args[1]->value)->getIterableValueType(), false, PassedByReference::createNo(), false, null),
1299+
];
1300+
}
1301+
}
1302+
}
1303+
$closureScope = $this->enterAnonymousFunctionWithoutReflection($node, $callableParameters);
12851304
$closureReturnStatements = [];
12861305
$closureYieldStatements = [];
12871306
$closureExecutionEnds = [];

src/NodeVisitor/StatementOrderVisitor.php

+4-1
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,10 @@ public function enterNode(Node $node)
4040
$node->setAttribute('statementOrder', $order);
4141
$node->setAttribute('statementDepth', $this->depth);
4242

43-
if ($node instanceof Node\Expr && count($this->expressionOrderStack) > 0) {
43+
if (
44+
($node instanceof Node\Expr || $node instanceof Node\Arg)
45+
&& count($this->expressionOrderStack) > 0
46+
) {
4447
$expressionOrder = $this->expressionOrderStack[count($this->expressionOrderStack) - 1];
4548
$node->setAttribute('expressionOrder', $expressionOrder);
4649
$node->setAttribute('expressionDepth', $this->expressionDepth);

tests/PHPStan/Analyser/NodeScopeResolverTest.php

+6
Original file line numberDiff line numberDiff line change
@@ -10923,6 +10923,11 @@ public function dataBug4498(): array
1092310923
return $this->gatherAssertTypes(__DIR__ . '/data/bug-4498.php');
1092410924
}
1092510925

10926+
public function dataBug4587(): array
10927+
{
10928+
return $this->gatherAssertTypes(__DIR__ . '/data/bug-4587.php');
10929+
}
10930+
1092610931
/**
1092710932
* @param string $file
1092810933
* @return array<string, mixed[]>
@@ -11169,6 +11174,7 @@ private function gatherAssertTypes(string $file): array
1116911174
* @dataProvider dataBug3769
1117011175
* @dataProvider dataBugInstanceOfClassString
1117111176
* @dataProvider dataBug4498
11177+
* @dataProvider dataBug4587
1117211178
* @param string $assertType
1117311179
* @param string $file
1117411180
* @param mixed ...$args
+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
<?php
2+
3+
namespace Bug4587;
4+
5+
use function PHPStan\Analyser\assertType;
6+
7+
class HelloWorld
8+
{
9+
public function a(): void
10+
{
11+
/** @var list<array{a: int}> $results */
12+
$results = [];
13+
14+
$type = array_map(static function (array $result): array {
15+
assertType('array(\'a\' => int)', $result);
16+
return $result;
17+
}, $results);
18+
19+
assertType('array<int, array(\'a\' => int)>', $type);
20+
}
21+
22+
public function b(): void
23+
{
24+
/** @var list<array{a: int}> $results */
25+
$results = [];
26+
27+
$type = array_map(static function (array $result): array {
28+
assertType('array(\'a\' => int)', $result);
29+
$result['a'] = (string) $result['a'];
30+
assertType('array(\'a\' => string&numeric)', $result);
31+
32+
return $result;
33+
}, $results);
34+
35+
assertType('array<int, array(\'a\' => string&numeric)>', $type);
36+
}
37+
}

tests/PHPStan/Analyser/data/generics.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -218,7 +218,7 @@ function testArrayMap(array $listOfIntegers)
218218

219219
return (string) $int;
220220
}, $listOfIntegers);
221-
assertType('array<string>', $strings);
221+
assertType('array<string&numeric>', $strings);
222222
}
223223

224224
/**

0 commit comments

Comments
 (0)