Skip to content

Commit 2f2e842

Browse files
committed
isset() narrows string-key in int-keyed-array to numeric-string
1 parent 9843dd8 commit 2f2e842

File tree

3 files changed

+43
-17
lines changed

3 files changed

+43
-17
lines changed

src/Analyser/TypeSpecifier.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -821,7 +821,7 @@ public function specifyTypesInCondition(
821821
);
822822
} else {
823823
$varType = $scope->getType($var->var);
824-
$narrowedKey = AllowedArrayKeysTypes::narrowOffsetKeyType($varType);
824+
$narrowedKey = AllowedArrayKeysTypes::narrowOffsetKeyType($varType, $dimType);
825825
if ($narrowedKey !== null) {
826826
$types = $types->unionWith(
827827
$this->create(

src/Rules/Arrays/AllowedArrayKeysTypes.php

+15-11
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,8 @@ public static function getType(): Type
3232
]);
3333
}
3434

35-
public static function narrowOffsetKeyType(Type $varType): ?Type {
35+
public static function narrowOffsetKeyType(Type $varType, Type $keyType): ?Type
36+
{
3637
if (!$varType->isArray()->yes() || $varType->isIterableAtLeastOnce()->no()) {
3738
return null;
3839
}
@@ -66,17 +67,20 @@ public static function narrowOffsetKeyType(Type $varType): ?Type {
6667
if (!$varIterableKeyType->isNumericString()->no() || !$varIterableKeyType->isInteger()->no()) {
6768
$narrowedKey = TypeCombinator::union($narrowedKey, new FloatType());
6869
}
69-
} else {
70-
$narrowedKey = new MixedType(
71-
false,
72-
new UnionType([
73-
new ArrayType(new MixedType(), new MixedType()),
74-
new ObjectWithoutClassType(),
75-
new ResourceType(),
76-
]),
77-
);
70+
71+
return $narrowedKey;
72+
} elseif ($varIterableKeyType->isInteger()->yes() && $keyType->isString()->yes()) {
73+
return TypeCombinator::intersect($varIterableKeyType->toString(), $keyType);
7874
}
7975

80-
return $narrowedKey;
76+
return new MixedType(
77+
false,
78+
new UnionType([
79+
new ArrayType(new MixedType(), new MixedType()),
80+
new ObjectWithoutClassType(),
81+
new ResourceType(),
82+
]),
83+
);
8184
}
85+
8286
}

tests/PHPStan/Analyser/nsrt/bug-11716.php

+27-5
Original file line numberDiff line numberDiff line change
@@ -35,9 +35,10 @@ public function parse(string $glue): string
3535
}
3636

3737
/**
38-
* @param array<int, string> $arr
38+
* @param array<int, string> $intKeyedArr
39+
* @param array<string, string> $stringKeyedArr
3940
*/
40-
function narrowKey($mixed, string $s, int $i, array $generalArr, array $arr): void {
41+
function narrowKey($mixed, string $s, int $i, array $generalArr, array $intKeyedArr, array $stringKeyedArr): void {
4142
if (isset($generalArr[$mixed])) {
4243
assertType('mixed~(array|object|resource)', $mixed);
4344
} else {
@@ -59,21 +60,42 @@ function narrowKey($mixed, string $s, int $i, array $generalArr, array $arr): vo
5960
}
6061
assertType('string', $s);
6162

62-
if (isset($arr[$mixed])) {
63+
if (isset($intKeyedArr[$mixed])) {
6364
assertType('mixed~(array|object|resource)', $mixed);
6465
} else {
6566
assertType('mixed', $mixed);
6667
}
6768
assertType('mixed', $mixed);
6869

69-
if (isset($arr[$i])) {
70+
if (isset($intKeyedArr[$i])) {
7071
assertType('int', $i);
7172
} else {
7273
assertType('int', $i);
7374
}
7475
assertType('int', $i);
7576

76-
if (isset($arr[$s])) {
77+
if (isset($intKeyedArr[$s])) {
78+
assertType("numeric-string", $s);
79+
} else {
80+
assertType('string', $s);
81+
}
82+
assertType('string', $s);
83+
84+
if (isset($stringKeyedArr[$mixed])) {
85+
assertType('mixed~(array|object|resource)', $mixed);
86+
} else {
87+
assertType('mixed', $mixed);
88+
}
89+
assertType('mixed', $mixed);
90+
91+
if (isset($stringKeyedArr[$i])) {
92+
assertType('int', $i);
93+
} else {
94+
assertType('int', $i);
95+
}
96+
assertType('int', $i);
97+
98+
if (isset($stringKeyedArr[$s])) {
7799
assertType('string', $s);
78100
} else {
79101
assertType('string', $s);

0 commit comments

Comments
 (0)