-
Notifications
You must be signed in to change notification settings - Fork 506
/
Copy pathStrlenTypeSpecifyingExtension.php
72 lines (59 loc) · 2.31 KB
/
StrlenTypeSpecifyingExtension.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
<?php declare(strict_types = 1);
namespace PHPStan\Type\Php;
use PhpParser\Node\Expr\FuncCall;
use PHPStan\Analyser\Scope;
use PHPStan\Analyser\SpecifiedTypes;
use PHPStan\Analyser\TypeSpecifier;
use PHPStan\Analyser\TypeSpecifierAwareExtension;
use PHPStan\Analyser\TypeSpecifierContext;
use PHPStan\Reflection\FunctionReflection;
use PHPStan\Type\Accessory\AccessoryNonEmptyStringType;
use PHPStan\Type\Accessory\AccessoryNonFalsyStringType;
use PHPStan\Type\Constant\ConstantIntegerType;
use PHPStan\Type\FunctionTypeSpecifyingExtension;
use PHPStan\Type\IntegerRangeType;
use PHPStan\Type\NeverType;
use function count;
use function in_array;
use function strtolower;
final class StrlenTypeSpecifyingExtension implements FunctionTypeSpecifyingExtension, TypeSpecifierAwareExtension
{
private TypeSpecifier $typeSpecifier;
public function setTypeSpecifier(TypeSpecifier $typeSpecifier): void
{
$this->typeSpecifier = $typeSpecifier;
}
public function isFunctionSupported(FunctionReflection $functionReflection, FuncCall $node, TypeSpecifierContext $context): bool
{
return in_array(strtolower($functionReflection->getName()), ['strlen', 'mb_strlen'], true) && !$context->null();
}
public function specifyTypes(FunctionReflection $functionReflection, FuncCall $node, Scope $scope, TypeSpecifierContext $context): SpecifiedTypes
{
$args = $node->getArgs();
if (
count($args) < 1
|| $context->getReturnType() === null
) {
return new SpecifiedTypes();
}
$argType = $scope->getType($args[0]->value);
if (!$argType->isString()->yes()) {
return new SpecifiedTypes();
}
$returnType = $context->getReturnType();
if ($returnType instanceof NeverType) {
return $this->typeSpecifier->create($args[0]->value, $returnType, $context->negate(), $scope);
}
if (
$context->true() && IntegerRangeType::createAllGreaterThanOrEqualTo(1)->isSuperTypeOf($returnType)->yes()
|| ($context->false() && (new ConstantIntegerType(0))->isSuperTypeOf($returnType)->yes())
) {
$accessory = new AccessoryNonEmptyStringType();
if (IntegerRangeType::createAllGreaterThanOrEqualTo(2)->isSuperTypeOf($returnType)->yes()) {
$accessory = new AccessoryNonFalsyStringType();
}
return $this->typeSpecifier->create($args[0]->value, $accessory, $context, $scope)->setRootExpr($node);
}
return new SpecifiedTypes();
}
}