11
11
use PhpParser \Node \Expr \NullsafeMethodCall ;
12
12
use PhpParser \Node \Expr \StaticCall ;
13
13
use PHPStan \Analyser \Scope ;
14
- use PHPStan \Reflection \ClassReflection ;
15
14
use ReflectionClass ;
16
15
use ShipMonk \PHPStan \DeadCode \Graph \ClassConstantRef ;
17
16
use ShipMonk \PHPStan \DeadCode \Graph \ClassConstantUsage ;
@@ -68,14 +67,18 @@ private function processMethodCall(MethodCall $node, Scope $scope): array
68
67
// ideally, we should check if T is covariant (marks children as used) or invariant (should not mark children as used)
69
68
// the default changed in PHP 8.4, see: https://github.com/phpstan/phpstan/issues/12459#issuecomment-2607123277
70
69
foreach ($ reflection ->getActiveTemplateTypeMap ()->getTypes () as $ genericType ) {
71
- foreach ($ genericType ->getObjectClassReflections () as $ genericReflection ) {
70
+ $ genericClassNames = $ genericType ->getObjectClassNames () === []
71
+ ? [null ] // call over ReflectionClass without specifying the generic type
72
+ : $ genericType ->getObjectClassNames ();
73
+
74
+ foreach ($ genericClassNames as $ genericClassName ) {
72
75
$ usedConstants = [
73
76
...$ usedConstants ,
74
- ...$ this ->extractConstantsUsedByReflection ($ methodName , $ genericReflection , $ node ->getArgs (), $ node , $ scope ),
77
+ ...$ this ->extractConstantsUsedByReflection ($ genericClassName , $ methodName , $ node ->getArgs (), $ node , $ scope ),
75
78
];
76
79
$ usedMethods = [
77
80
...$ usedMethods ,
78
- ...$ this ->extractMethodsUsedByReflection ($ methodName , $ genericReflection , $ node ->getArgs (), $ node , $ scope ),
81
+ ...$ this ->extractMethodsUsedByReflection ($ genericClassName , $ methodName , $ node ->getArgs (), $ node , $ scope ),
79
82
];
80
83
}
81
84
}
@@ -93,8 +96,8 @@ private function processMethodCall(MethodCall $node, Scope $scope): array
93
96
* @return list<ClassConstantUsage>
94
97
*/
95
98
private function extractConstantsUsedByReflection (
99
+ ?string $ genericClassName ,
96
100
string $ methodName ,
97
- ClassReflection $ genericReflection ,
98
101
array $ args ,
99
102
Node $ node ,
100
103
Scope $ scope
@@ -103,14 +106,14 @@ private function extractConstantsUsedByReflection(
103
106
$ usedConstants = [];
104
107
105
108
if ($ methodName === 'getConstants ' || $ methodName === 'getReflectionConstants ' ) {
106
- $ usedConstants [] = $ this ->createConstantUsage ($ node , $ scope , $ genericReflection -> getName () , null );
109
+ $ usedConstants [] = $ this ->createConstantUsage ($ node , $ scope , $ genericClassName , null );
107
110
}
108
111
109
112
if (($ methodName === 'getConstant ' || $ methodName === 'getReflectionConstant ' ) && count ($ args ) === 1 ) {
110
113
$ firstArg = $ args [array_key_first ($ args )];
111
114
112
115
foreach ($ scope ->getType ($ firstArg ->value )->getConstantStrings () as $ constantString ) {
113
- $ usedConstants [] = $ this ->createConstantUsage ($ node , $ scope , $ genericReflection -> getName () , $ constantString ->getValue ());
116
+ $ usedConstants [] = $ this ->createConstantUsage ($ node , $ scope , $ genericClassName , $ constantString ->getValue ());
114
117
}
115
118
}
116
119
@@ -122,8 +125,8 @@ private function extractConstantsUsedByReflection(
122
125
* @return list<ClassMethodUsage>
123
126
*/
124
127
private function extractMethodsUsedByReflection (
128
+ ?string $ genericClassName ,
125
129
string $ methodName ,
126
- ClassReflection $ genericReflection ,
127
130
array $ args ,
128
131
Node $ node ,
129
132
Scope $ scope
@@ -132,23 +135,19 @@ private function extractMethodsUsedByReflection(
132
135
$ usedMethods = [];
133
136
134
137
if ($ methodName === 'getMethods ' ) {
135
- $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ genericReflection -> getName () , null );
138
+ $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ genericClassName , null );
136
139
}
137
140
138
141
if ($ methodName === 'getMethod ' && count ($ args ) === 1 ) {
139
142
$ firstArg = $ args [array_key_first ($ args )];
140
143
141
144
foreach ($ scope ->getType ($ firstArg ->value )->getConstantStrings () as $ constantString ) {
142
- $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ genericReflection -> getName () , $ constantString ->getValue ());
145
+ $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ genericClassName , $ constantString ->getValue ());
143
146
}
144
147
}
145
148
146
149
if (in_array ($ methodName , ['getConstructor ' , 'newInstance ' , 'newInstanceArgs ' ], true )) {
147
- $ constructor = $ genericReflection ->getNativeReflection ()->getConstructor ();
148
-
149
- if ($ constructor !== null ) {
150
- $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ constructor ->getDeclaringClass ()->getName (), '__construct ' );
151
- }
150
+ $ usedMethods [] = $ this ->createMethodUsage ($ node , $ scope , $ genericClassName , '__construct ' );
152
151
}
153
152
154
153
return $ usedMethods ;
@@ -180,7 +179,7 @@ private function getMethodNames(CallLike $call, Scope $scope): array
180
179
private function createConstantUsage (
181
180
Node $ node ,
182
181
Scope $ scope ,
183
- string $ className ,
182
+ ? string $ className ,
184
183
?string $ constantName
185
184
): ClassConstantUsage
186
185
{
@@ -197,7 +196,7 @@ private function createConstantUsage(
197
196
private function createMethodUsage (
198
197
Node $ node ,
199
198
Scope $ scope ,
200
- string $ className ,
199
+ ? string $ className ,
201
200
?string $ methodName
202
201
): ClassMethodUsage
203
202
{
0 commit comments