Skip to content

Commit 00184f0

Browse files
committed
More precise array_keys return type
1 parent 68d01a5 commit 00184f0

File tree

2 files changed

+72
-2
lines changed

2 files changed

+72
-2
lines changed

src/Type/Php/ArrayKeysFunctionDynamicReturnTypeExtension.php

+21-2
Original file line numberDiff line numberDiff line change
@@ -6,10 +6,14 @@
66
use PHPStan\Analyser\Scope;
77
use PHPStan\Php\PhpVersion;
88
use PHPStan\Reflection\FunctionReflection;
9+
use PHPStan\Type\Accessory\AccessoryArrayListType;
10+
use PHPStan\Type\ArrayType;
911
use PHPStan\Type\DynamicFunctionReturnTypeExtension;
12+
use PHPStan\Type\GeneralizePrecision;
1013
use PHPStan\Type\NeverType;
1114
use PHPStan\Type\NullType;
1215
use PHPStan\Type\Type;
16+
use PHPStan\Type\TypeCombinator;
1317
use function count;
1418
use function strtolower;
1519

@@ -27,7 +31,7 @@ public function isFunctionSupported(FunctionReflection $functionReflection): boo
2731

2832
public function getTypeFromFunctionCall(FunctionReflection $functionReflection, FuncCall $functionCall, Scope $scope): ?Type
2933
{
30-
if (count($functionCall->getArgs()) !== 1) {
34+
if (count($functionCall->getArgs()) < 1) {
3135
return null;
3236
}
3337

@@ -36,7 +40,22 @@ public function getTypeFromFunctionCall(FunctionReflection $functionReflection,
3640
return $this->phpVersion->arrayFunctionsReturnNullWithNonArray() ? new NullType() : new NeverType();
3741
}
3842

39-
return $arrayType->getKeysArray();
43+
$keysArray = $arrayType->getKeysArray();
44+
if (count($functionCall->getArgs()) === 1) {
45+
return $keysArray;
46+
}
47+
48+
$newArrayType = $keysArray;
49+
if (!$keysArray->isConstantArray()->no()) {
50+
$newArrayType = new ArrayType(
51+
$keysArray->getIterableKeyType()->generalize(GeneralizePrecision::lessSpecific()),
52+
$keysArray->getIterableValueType()->generalize(GeneralizePrecision::lessSpecific()),
53+
);
54+
}
55+
if ($keysArray->isList()->yes()) {
56+
$newArrayType = TypeCombinator::intersect($newArrayType, new AccessoryArrayListType());
57+
}
58+
return $newArrayType;
4059
}
4160

4261
}
+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
<?php
2+
3+
namespace Bug11928;
4+
5+
use function PHPStan\Testing\assertType;
6+
7+
function doFoo()
8+
{
9+
$a = [2 => 1, 3 => 2, 4 => 1];
10+
11+
$keys = array_keys($a, 1); // returns [2, 4]
12+
assertType('list<int>', $keys);
13+
14+
$keys = array_keys($a); // returns [2, 3, 4]
15+
assertType('array{2, 3, 4}', $keys);
16+
}
17+
18+
function doFooStrings() {
19+
$a = [2 => 'hi', 3 => '123', 'xy' => 5];
20+
$keys = array_keys($a, 1);
21+
assertType('list<int|string>', $keys);
22+
23+
$keys = array_keys($a);
24+
assertType("array{2, 3, 'xy'}", $keys);
25+
}
26+
27+
/**
28+
* @param array<int, int> $array
29+
* @param list<int> $list
30+
* @param array<string, string> $strings
31+
* @return void
32+
*/
33+
function doFooBar(array $array, array $list, array $strings) {
34+
$keys = array_keys($strings, "a", true);
35+
assertType('list<string>', $keys);
36+
37+
$keys = array_keys($strings, "a", false);
38+
assertType('list<string>', $keys);
39+
40+
$keys = array_keys($array, 1, true);
41+
assertType('list<int>', $keys);
42+
43+
$keys = array_keys($array, 1, false);
44+
assertType('list<int>', $keys);
45+
46+
$keys = array_keys($list, 1, true);
47+
assertType('list<int<0, max>>', $keys);
48+
49+
$keys = array_keys($list, 1, true);
50+
assertType('list<int<0, max>>', $keys);
51+
}

0 commit comments

Comments
 (0)