/
ResolvedFunctionVariant.php
102 lines (80 loc) · 2.44 KB
/
ResolvedFunctionVariant.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
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
<?php declare(strict_types = 1);
namespace PHPStan\Reflection;
use PHPStan\Reflection\Php\DummyParameter;
use PHPStan\Type\ConditionalTypeForParameter;
use PHPStan\Type\Generic\TemplateTypeHelper;
use PHPStan\Type\Generic\TemplateTypeMap;
use PHPStan\Type\Type;
use PHPStan\Type\TypeTraverser;
use function array_key_exists;
use function array_map;
class ResolvedFunctionVariant implements ParametersAcceptor
{
/** @var ParameterReflection[]|null */
private ?array $parameters = null;
private ?Type $returnType = null;
/**
* @param array<string, Type> $passedArgs
*/
public function __construct(
private ParametersAcceptor $parametersAcceptor,
private TemplateTypeMap $resolvedTemplateTypeMap,
private array $passedArgs,
)
{
}
public function getOriginalParametersAcceptor(): ParametersAcceptor
{
return $this->parametersAcceptor;
}
public function getTemplateTypeMap(): TemplateTypeMap
{
return $this->parametersAcceptor->getTemplateTypeMap();
}
public function getResolvedTemplateTypeMap(): TemplateTypeMap
{
return $this->resolvedTemplateTypeMap;
}
public function getParameters(): array
{
$parameters = $this->parameters;
if ($parameters === null) {
$parameters = array_map(fn (ParameterReflection $param): ParameterReflection => new DummyParameter(
$param->getName(),
TemplateTypeHelper::resolveTemplateTypes($param->getType(), $this->resolvedTemplateTypeMap),
$param->isOptional(),
$param->passedByReference(),
$param->isVariadic(),
$param->getDefaultValue(),
), $this->parametersAcceptor->getParameters());
$this->parameters = $parameters;
}
return $parameters;
}
public function isVariadic(): bool
{
return $this->parametersAcceptor->isVariadic();
}
public function getReturnType(): Type
{
$type = $this->returnType;
if ($type === null) {
$type = TemplateTypeHelper::resolveTemplateTypes(
$this->parametersAcceptor->getReturnType(),
$this->resolvedTemplateTypeMap,
);
$type = $this->resolveConditionalTypes($type);
$this->returnType = $type;
}
return $type;
}
private function resolveConditionalTypes(Type $type): Type
{
return TypeTraverser::map($type, function (Type $type, callable $traverse): Type {
while ($type instanceof ConditionalTypeForParameter && array_key_exists($type->getParameterName(), $this->passedArgs)) {
$type = $type->toConditional($this->passedArgs[$type->getParameterName()]);
}
return $traverse($type);
});
}
}