/
ContainerArrayAccessDynamicMethodReturnTypeExtension.php
83 lines (63 loc) · 2 KB
/
ContainerArrayAccessDynamicMethodReturnTypeExtension.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
73
74
75
76
77
78
79
80
81
82
83
<?php
declare(strict_types=1);
namespace NunoMaduro\Larastan\ReturnTypes;
use NunoMaduro\Larastan\Concerns\HasContainer;
use PhpParser\Node\Expr\MethodCall;
use PHPStan\Analyser\Scope;
use PHPStan\Reflection\MethodReflection;
use PHPStan\Type\DynamicMethodReturnTypeExtension;
use PHPStan\Type\ErrorType;
use PHPStan\Type\ObjectType;
use PHPStan\Type\Type;
use PHPStan\Type\TypeCombinator;
use PHPStan\Type\TypeUtils;
class ContainerArrayAccessDynamicMethodReturnTypeExtension implements DynamicMethodReturnTypeExtension
{
use HasContainer;
/**
* @var string
*/
private $className;
public function __construct(string $className)
{
$this->className = $className;
}
public function getClass(): string
{
return $this->className;
}
public function isMethodSupported(MethodReflection $methodReflection): bool
{
return $methodReflection->getName() === 'offsetGet';
}
public function getTypeFromMethodCall(
MethodReflection $methodReflection,
MethodCall $methodCall,
Scope $scope
): ?Type {
$args = $methodCall->getArgs();
if (count($args) === 0) {
return null;
}
$argType = $scope->getType($args[0]->value);
$argStrings = TypeUtils::getConstantStrings($argType);
if ($argStrings === []) {
return null;
}
$argTypes = [];
foreach ($argStrings as $argString) {
$resolvedValue = $this->resolve($argString->getValue());
if ($resolvedValue === null) {
$argTypes[] = new ErrorType();
continue;
}
if (is_object($resolvedValue)) {
$class = get_class($resolvedValue);
$argTypes[] = new ObjectType($class);
continue;
}
$argTypes[] = $scope->getTypeFromValue($resolvedValue);
}
return count($argTypes) > 1 ? TypeCombinator::union(...$argTypes) : $argTypes[0];
}
}