Skip to content

Commit f4eb0bc

Browse files
committed
Fix analysis on array_map with named arguments
1 parent 068be33 commit f4eb0bc

File tree

3 files changed

+71
-12
lines changed

3 files changed

+71
-12
lines changed

src/Parser/ArrayMapArgVisitor.php

+34-12
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44

55
use PhpParser\Node;
66
use PhpParser\NodeVisitorAbstract;
7-
use function array_slice;
7+
use function array_splice;
88
use function count;
99

1010
final class ArrayMapArgVisitor extends NodeVisitorAbstract
@@ -14,19 +14,41 @@ final class ArrayMapArgVisitor extends NodeVisitorAbstract
1414

1515
public function enterNode(Node $node): ?Node
1616
{
17-
if ($node instanceof Node\Expr\FuncCall && $node->name instanceof Node\Name && !$node->isFirstClassCallable()) {
18-
$functionName = $node->name->toLowerString();
19-
if ($functionName === 'array_map') {
20-
$args = $node->getArgs();
21-
if (isset($args[0])) {
22-
$slicedArgs = array_slice($args, 1);
23-
if (count($slicedArgs) > 0) {
24-
$args[0]->value->setAttribute(self::ATTRIBUTE_NAME, $slicedArgs);
25-
}
26-
}
27-
}
17+
if (!$this->isArrayMapCall($node)) {
18+
return null;
2819
}
20+
21+
$args = $node->getArgs();
22+
if (count($args) < 2) {
23+
return null;
24+
}
25+
26+
$callbackPos = 0;
27+
if ($args[1]->name !== null && $args[1]->name->name === 'callback') {
28+
$callbackPos = 1;
29+
}
30+
[$callback] = array_splice($args, $callbackPos, 1);
31+
$callback->value->setAttribute(self::ATTRIBUTE_NAME, $args);
32+
2933
return null;
3034
}
3135

36+
/**
37+
* @phpstan-assert-if-true Node\Expr\FuncCall $node
38+
*/
39+
private function isArrayMapCall(Node $node): bool
40+
{
41+
if (!$node instanceof Node\Expr\FuncCall) {
42+
return false;
43+
}
44+
if (!$node->name instanceof Node\Name) {
45+
return false;
46+
}
47+
if ($node->isFirstClassCallable()) {
48+
return false;
49+
}
50+
51+
return $node->name->toLowerString() === 'array_map';
52+
}
53+
3254
}

tests/PHPStan/Rules/Functions/CallToFunctionParametersRuleTest.php

+11
Original file line numberDiff line numberDiff line change
@@ -1936,4 +1936,15 @@ public function testBug12051(): void
19361936
$this->analyse([__DIR__ . '/data/bug-12051.php'], []);
19371937
}
19381938

1939+
public function testBug12317(): void
1940+
{
1941+
if (PHP_VERSION_ID < 80000) {
1942+
$this->markTestSkipped('Test requires PHP 8.0.');
1943+
}
1944+
1945+
$this->checkExplicitMixed = true;
1946+
$this->checkImplicitMixed = true;
1947+
$this->analyse([__DIR__ . '/data/bug-12317.php'], []);
1948+
}
1949+
19391950
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace Bug12317;
4+
5+
class Uuid {
6+
public function __construct(public string $uuid) {}
7+
public function __toString() { return $this->uuid; }
8+
}
9+
10+
class HelloWorld
11+
{
12+
/**
13+
* @param list<Uuid> $a
14+
*
15+
* @return list<string>
16+
*/
17+
public function sayHello(array $a): array
18+
{
19+
$b = array_map(
20+
array: $a,
21+
callback: static fn(Uuid $c): string => (string) $c,
22+
);
23+
24+
return $b;
25+
}
26+
}

0 commit comments

Comments
 (0)