2
2
3
3
namespace PHPStan \PhpDoc ;
4
4
5
+ use PHPStan \Analyser \NameScope ;
5
6
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprArrayNode ;
6
7
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprFalseNode ;
7
8
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprFloatNode ;
10
11
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprNullNode ;
11
12
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprStringNode ;
12
13
use PHPStan \PhpDocParser \Ast \ConstExpr \ConstExprTrueNode ;
14
+ use PHPStan \PhpDocParser \Ast \ConstExpr \ConstFetchNode ;
15
+ use PHPStan \Reflection \InitializerExprContext ;
16
+ use PHPStan \Reflection \InitializerExprTypeResolver ;
17
+ use PHPStan \Reflection \ReflectionProvider ;
13
18
use PHPStan \Type \Constant \ConstantArrayTypeBuilder ;
14
19
use PHPStan \Type \Constant \ConstantBooleanType ;
15
20
use PHPStan \Type \Constant \ConstantFloatType ;
16
21
use PHPStan \Type \Constant \ConstantIntegerType ;
17
22
use PHPStan \Type \Constant \ConstantStringType ;
18
- use PHPStan \Type \MixedType ;
23
+ use PHPStan \Type \Enum \EnumCaseObjectType ;
24
+ use PHPStan \Type \ErrorType ;
19
25
use PHPStan \Type \NullType ;
20
26
use PHPStan \Type \Type ;
27
+ use function strtolower ;
21
28
22
29
final class ConstExprNodeResolver
23
30
{
24
31
25
- public function resolve (ConstExprNode $ node ): Type
32
+ public function __construct (
33
+ private ReflectionProvider \ReflectionProviderProvider $ reflectionProviderProvider ,
34
+ private InitializerExprTypeResolver $ initializerExprTypeResolver ,
35
+ )
36
+ {
37
+ }
38
+
39
+ public function resolve (ConstExprNode $ node , NameScope $ nameScope ): Type
26
40
{
27
41
if ($ node instanceof ConstExprArrayNode) {
28
- return $ this ->resolveArrayNode ($ node );
42
+ return $ this ->resolveArrayNode ($ node, $ nameScope );
29
43
}
30
44
31
45
if ($ node instanceof ConstExprFalseNode) {
@@ -52,22 +66,74 @@ public function resolve(ConstExprNode $node): Type
52
66
return new ConstantStringType ($ node ->value );
53
67
}
54
68
55
- return new MixedType ();
69
+ if ($ node instanceof ConstFetchNode) {
70
+ if ($ nameScope ->getClassName () !== null ) {
71
+ switch (strtolower ($ node ->className )) {
72
+ case 'static ' :
73
+ case 'self ' :
74
+ $ className = $ nameScope ->getClassName ();
75
+ break ;
76
+
77
+ case 'parent ' :
78
+ if ($ this ->getReflectionProvider ()->hasClass ($ nameScope ->getClassName ())) {
79
+ $ classReflection = $ this ->getReflectionProvider ()->getClass ($ nameScope ->getClassName ());
80
+ if ($ classReflection ->getParentClass () === null ) {
81
+ return new ErrorType ();
82
+
83
+ }
84
+
85
+ $ className = $ classReflection ->getParentClass ()->getName ();
86
+ }
87
+ break ;
88
+ }
89
+ }
90
+ if (!isset ($ className )) {
91
+ $ className = $ nameScope ->resolveStringName ($ node ->className );
92
+ }
93
+ if (!$ this ->getReflectionProvider ()->hasClass ($ className )) {
94
+ return new ErrorType ();
95
+ }
96
+ $ classReflection = $ this ->getReflectionProvider ()->getClass ($ className );
97
+ if (!$ classReflection ->hasConstant ($ node ->name )) {
98
+ return new ErrorType ();
99
+ }
100
+ if ($ classReflection ->isEnum () && $ classReflection ->hasEnumCase ($ node ->name )) {
101
+ return new EnumCaseObjectType ($ classReflection ->getName (), $ node ->name );
102
+ }
103
+
104
+ $ reflectionConstant = $ classReflection ->getNativeReflection ()->getReflectionConstant ($ node ->name );
105
+ if ($ reflectionConstant === false ) {
106
+ return new ErrorType ();
107
+ }
108
+ $ declaringClass = $ reflectionConstant ->getDeclaringClass ();
109
+
110
+ return $ this ->initializerExprTypeResolver ->getType (
111
+ $ reflectionConstant ->getValueExpression (),
112
+ InitializerExprContext::fromClass ($ declaringClass ->getName (), $ declaringClass ->getFileName () ?: null ),
113
+ );
114
+ }
115
+
116
+ return new ErrorType ();
56
117
}
57
118
58
- private function resolveArrayNode (ConstExprArrayNode $ node ): Type
119
+ private function resolveArrayNode (ConstExprArrayNode $ node, NameScope $ nameScope ): Type
59
120
{
60
121
$ arrayBuilder = ConstantArrayTypeBuilder::createEmpty ();
61
122
foreach ($ node ->items as $ item ) {
62
123
if ($ item ->key === null ) {
63
124
$ key = null ;
64
125
} else {
65
- $ key = $ this ->resolve ($ item ->key );
126
+ $ key = $ this ->resolve ($ item ->key , $ nameScope );
66
127
}
67
- $ arrayBuilder ->setOffsetValueType ($ key , $ this ->resolve ($ item ->value ));
128
+ $ arrayBuilder ->setOffsetValueType ($ key , $ this ->resolve ($ item ->value , $ nameScope ));
68
129
}
69
130
70
131
return $ arrayBuilder ->getArray ();
71
132
}
72
133
134
+ private function getReflectionProvider (): ReflectionProvider
135
+ {
136
+ return $ this ->reflectionProviderProvider ->getReflectionProvider ();
137
+ }
138
+
73
139
}
0 commit comments