Skip to content

Commit deef919

Browse files
committed
Avoid new HasOffsetValueType being intersected with oversized array
1 parent 16f63b3 commit deef919

File tree

3 files changed

+37
-0
lines changed

3 files changed

+37
-0
lines changed

phpstan-baseline.neon

+10
Original file line numberDiff line numberDiff line change
@@ -1170,6 +1170,11 @@ parameters:
11701170
count: 3
11711171
path: src/Type/IntersectionType.php
11721172

1173+
-
1174+
message: "#^Doing instanceof PHPStan\\\\Type\\\\ArrayType is error\\-prone and deprecated\\. Use Type\\:\\:isArray\\(\\) or Type\\:\\:getArrays\\(\\) instead\\.$#"
1175+
count: 1
1176+
path: src/Type/IntersectionType.php
1177+
11731178
-
11741179
message: "#^Doing instanceof PHPStan\\\\Type\\\\BooleanType is error\\-prone and deprecated\\. Use Type\\:\\:isBoolean\\(\\) instead\\.$#"
11751180
count: 1
@@ -1180,6 +1185,11 @@ parameters:
11801185
count: 2
11811186
path: src/Type/IntersectionType.php
11821187

1188+
-
1189+
message: "#^Doing instanceof PHPStan\\\\Type\\\\Constant\\\\ConstantStringType is error\\-prone and deprecated\\. Use Type\\:\\:getConstantStrings\\(\\) instead\\.$#"
1190+
count: 1
1191+
path: src/Type/IntersectionType.php
1192+
11831193
-
11841194
message: "#^Doing instanceof PHPStan\\\\Type\\\\IntersectionType is error\\-prone and deprecated\\.$#"
11851195
count: 4

src/Type/Constant/ConstantArrayTypeBuilder.php

+2
Original file line numberDiff line numberDiff line change
@@ -128,6 +128,7 @@ public function setOffsetValueType(?Type $offsetType, Type $valueType, bool $opt
128128

129129
if (count($this->keyTypes) > self::ARRAY_COUNT_LIMIT) {
130130
$this->degradeToGeneralArray = true;
131+
$this->oversized = true;
131132
}
132133

133134
return;
@@ -194,6 +195,7 @@ public function setOffsetValueType(?Type $offsetType, Type $valueType, bool $opt
194195

195196
if (count($this->keyTypes) > self::ARRAY_COUNT_LIMIT) {
196197
$this->degradeToGeneralArray = true;
198+
$this->oversized = true;
197199
}
198200

199201
return;

src/Type/IntersectionType.php

+25
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,7 @@
2828
use PHPStan\Type\Accessory\AccessoryType;
2929
use PHPStan\Type\Accessory\NonEmptyArrayType;
3030
use PHPStan\Type\Constant\ConstantIntegerType;
31+
use PHPStan\Type\Constant\ConstantStringType;
3132
use PHPStan\Type\Generic\TemplateType;
3233
use PHPStan\Type\Generic\TemplateTypeMap;
3334
use PHPStan\Type\Generic\TemplateTypeVariance;
@@ -722,6 +723,30 @@ public function getOffsetValueType(Type $offsetType): Type
722723

723724
public function setOffsetValueType(?Type $offsetType, Type $valueType, bool $unionValues = true): Type
724725
{
726+
if ($this->isOversizedArray()->yes()) {
727+
return $this->intersectTypes(static function (Type $type) use ($offsetType, $valueType, $unionValues): Type {
728+
// avoid new HasOffsetValueType being intersected with oversized array
729+
if (!$type instanceof ArrayType) {
730+
return $type->setOffsetValueType($offsetType, $valueType, $unionValues);
731+
}
732+
733+
if (!$offsetType instanceof ConstantStringType && !$offsetType instanceof ConstantIntegerType) {
734+
return $type->setOffsetValueType($offsetType, $valueType, $unionValues);
735+
}
736+
737+
if (!$offsetType->isSuperTypeOf($type->getKeyType())->yes()) {
738+
return $type->setOffsetValueType($offsetType, $valueType, $unionValues);
739+
}
740+
741+
return TypeCombinator::intersect(
742+
new ArrayType(
743+
TypeCombinator::union($type->getKeyType(), $offsetType),
744+
TypeCombinator::union($type->getItemType(), $valueType),
745+
),
746+
new NonEmptyArrayType(),
747+
);
748+
});
749+
}
725750
return $this->intersectTypes(static fn (Type $type): Type => $type->setOffsetValueType($offsetType, $valueType, $unionValues));
726751
}
727752

0 commit comments

Comments
 (0)