Skip to content

Commit 7dd7b14

Browse files
authored
Support narrowing a constant array to a list with count
1 parent e169fde commit 7dd7b14

File tree

3 files changed

+23
-7
lines changed

3 files changed

+23
-7
lines changed

src/Analyser/TypeSpecifier.php

+12-5
Original file line numberDiff line numberDiff line change
@@ -1061,10 +1061,11 @@ private function specifyTypesForCountFuncCall(
10611061
$isNormalCount = (new ConstantIntegerType(COUNT_NORMAL))->isSuperTypeOf($mode)->result->or($type->getIterableValueType()->isArray()->negate());
10621062
}
10631063

1064+
$isConstantArray = $type->isConstantArray();
10641065
$isList = $type->isList();
10651066
if (
10661067
!$isNormalCount->yes()
1067-
|| (!$type->isConstantArray()->yes() && !$isList->yes())
1068+
|| (!$isConstantArray->yes() && !$isList->yes())
10681069
|| $type->isIterableAtLeastOnce()->no() // array{} cannot be used for further narrowing
10691070
) {
10701071
return null;
@@ -1082,9 +1083,12 @@ private function specifyTypesForCountFuncCall(
10821083
}
10831084

10841085
if (
1085-
$isList->yes()
1086-
&& $sizeType instanceof ConstantIntegerType
1086+
$sizeType instanceof ConstantIntegerType
10871087
&& $sizeType->getValue() < ConstantArrayTypeBuilder::ARRAY_COUNT_LIMIT
1088+
&& (
1089+
$isList->yes()
1090+
|| $isConstantArray->yes() && $arrayType->getKeyType()->isSuperTypeOf(IntegerRangeType::fromInterval(0, $sizeType->getValue() - 1))->yes()
1091+
)
10881092
) {
10891093
// turn optional offsets non-optional
10901094
$valueTypesBuilder = ConstantArrayTypeBuilder::createEmpty();
@@ -1097,9 +1101,12 @@ private function specifyTypesForCountFuncCall(
10971101
}
10981102

10991103
if (
1100-
$isList->yes()
1101-
&& $sizeType instanceof IntegerRangeType
1104+
$sizeType instanceof IntegerRangeType
11021105
&& $sizeType->getMin() !== null
1106+
&& (
1107+
$isList->yes()
1108+
|| $isConstantArray->yes() && $arrayType->getKeyType()->isSuperTypeOf(IntegerRangeType::fromInterval(0, $sizeType->getMin() - 1))->yes()
1109+
)
11031110
) {
11041111
// turn optional offsets non-optional
11051112
$valueTypesBuilder = ConstantArrayTypeBuilder::createEmpty();

tests/PHPStan/Analyser/nsrt/count-type.php

+10
Original file line numberDiff line numberDiff line change
@@ -87,6 +87,16 @@ public function doBaz(array $arr): void
8787
assertType('1|2', count($arr));
8888
}
8989

90+
public function constantArrayWhichCanBecomeList(string $h): void
91+
{
92+
preg_match('#^([a-z0-9-]+)\..+$#', $h, $matches);
93+
if (count($matches) !== 2) {
94+
return;
95+
}
96+
97+
assertType('array{string, non-empty-string}', $matches);
98+
}
99+
90100
}
91101

92102
/**

tests/PHPStan/Analyser/nsrt/list-count.php

+1-2
Original file line numberDiff line numberDiff line change
@@ -379,9 +379,8 @@ protected function testOptionalKeysInUnionListWithIntRange($row, $twoOrThree, $t
379379
*/
380380
protected function testOptionalKeysInUnionArrayWithIntRange($row, $twoOrThree): void
381381
{
382-
// doesn't narrow because no list
383382
if (count($row) >= $twoOrThree) {
384-
assertType('array{0: int, 1?: string|null, 2?: int|null, 3?: float|null}', $row);
383+
assertType('array{0: int, 1: string|null, 2?: int|null}', $row);
385384
} else {
386385
assertType('array{0: int, 1?: string|null, 2?: int|null, 3?: float|null}|array{string}', $row);
387386
}

0 commit comments

Comments
 (0)