Skip to content

Commit e42ff8a

Browse files
committed
Support ReflectionClass calls without generic type specified
1 parent 33cf2c2 commit e42ff8a

File tree

4 files changed

+60
-17
lines changed

4 files changed

+60
-17
lines changed

src/Provider/ReflectionUsageProvider.php

+16-17
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,6 @@
1111
use PhpParser\Node\Expr\NullsafeMethodCall;
1212
use PhpParser\Node\Expr\StaticCall;
1313
use PHPStan\Analyser\Scope;
14-
use PHPStan\Reflection\ClassReflection;
1514
use ReflectionClass;
1615
use ShipMonk\PHPStan\DeadCode\Graph\ClassConstantRef;
1716
use ShipMonk\PHPStan\DeadCode\Graph\ClassConstantUsage;
@@ -68,14 +67,18 @@ private function processMethodCall(MethodCall $node, Scope $scope): array
6867
// ideally, we should check if T is covariant (marks children as used) or invariant (should not mark children as used)
6968
// the default changed in PHP 8.4, see: https://github.com/phpstan/phpstan/issues/12459#issuecomment-2607123277
7069
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) {
7275
$usedConstants = [
7376
...$usedConstants,
74-
...$this->extractConstantsUsedByReflection($methodName, $genericReflection, $node->getArgs(), $node, $scope),
77+
...$this->extractConstantsUsedByReflection($genericClassName, $methodName, $node->getArgs(), $node, $scope),
7578
];
7679
$usedMethods = [
7780
...$usedMethods,
78-
...$this->extractMethodsUsedByReflection($methodName, $genericReflection, $node->getArgs(), $node, $scope),
81+
...$this->extractMethodsUsedByReflection($genericClassName, $methodName, $node->getArgs(), $node, $scope),
7982
];
8083
}
8184
}
@@ -93,8 +96,8 @@ private function processMethodCall(MethodCall $node, Scope $scope): array
9396
* @return list<ClassConstantUsage>
9497
*/
9598
private function extractConstantsUsedByReflection(
99+
?string $genericClassName,
96100
string $methodName,
97-
ClassReflection $genericReflection,
98101
array $args,
99102
Node $node,
100103
Scope $scope
@@ -103,14 +106,14 @@ private function extractConstantsUsedByReflection(
103106
$usedConstants = [];
104107

105108
if ($methodName === 'getConstants' || $methodName === 'getReflectionConstants') {
106-
$usedConstants[] = $this->createConstantUsage($node, $scope, $genericReflection->getName(), null);
109+
$usedConstants[] = $this->createConstantUsage($node, $scope, $genericClassName, null);
107110
}
108111

109112
if (($methodName === 'getConstant' || $methodName === 'getReflectionConstant') && count($args) === 1) {
110113
$firstArg = $args[array_key_first($args)];
111114

112115
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());
114117
}
115118
}
116119

@@ -122,8 +125,8 @@ private function extractConstantsUsedByReflection(
122125
* @return list<ClassMethodUsage>
123126
*/
124127
private function extractMethodsUsedByReflection(
128+
?string $genericClassName,
125129
string $methodName,
126-
ClassReflection $genericReflection,
127130
array $args,
128131
Node $node,
129132
Scope $scope
@@ -132,23 +135,19 @@ private function extractMethodsUsedByReflection(
132135
$usedMethods = [];
133136

134137
if ($methodName === 'getMethods') {
135-
$usedMethods[] = $this->createMethodUsage($node, $scope, $genericReflection->getName(), null);
138+
$usedMethods[] = $this->createMethodUsage($node, $scope, $genericClassName, null);
136139
}
137140

138141
if ($methodName === 'getMethod' && count($args) === 1) {
139142
$firstArg = $args[array_key_first($args)];
140143

141144
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());
143146
}
144147
}
145148

146149
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');
152151
}
153152

154153
return $usedMethods;
@@ -180,7 +179,7 @@ private function getMethodNames(CallLike $call, Scope $scope): array
180179
private function createConstantUsage(
181180
Node $node,
182181
Scope $scope,
183-
string $className,
182+
?string $className,
184183
?string $constantName
185184
): ClassConstantUsage
186185
{
@@ -197,7 +196,7 @@ private function createConstantUsage(
197196
private function createMethodUsage(
198197
Node $node,
199198
Scope $scope,
200-
string $className,
199+
?string $className,
201200
?string $methodName
202201
): ClassMethodUsage
203202
{

tests/Rule/DeadCodeRuleTest.php

+1
Original file line numberDiff line numberDiff line change
@@ -678,6 +678,7 @@ public static function provideFiles(): Traversable
678678
// providers
679679
yield 'provider-vendor' => [__DIR__ . '/data/providers/vendor.php'];
680680
yield 'provider-reflection' => [__DIR__ . '/data/providers/reflection.php', self::requiresPhp(8_01_00)];
681+
yield 'provider-reflection-no-t' => [__DIR__ . '/data/providers/reflection-no-generics.php'];
681682
yield 'provider-symfony' => [__DIR__ . '/data/providers/symfony.php', self::requiresPhp(8_00_00)];
682683
yield 'provider-symfony-7.1' => [__DIR__ . '/data/providers/symfony-gte71.php', self::requiresPhp(8_00_00) && self::requiresPackage('symfony/dependency-injection', '>= 7.1')];
683684
yield 'provider-phpunit' => [__DIR__ . '/data/providers/phpunit.php', self::requiresPhp(8_00_00)];
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace ReflectionNoGenerics;
4+
5+
interface MyParent
6+
{
7+
const CONST1 = 1;
8+
9+
public function bar();
10+
}
11+
12+
class Holder2
13+
{
14+
const CONST1 = 1;
15+
const CONST2 = 2; // error: Unused ReflectionNoGenerics\Holder2::CONST2
16+
17+
public function foo() {} // error: Unused ReflectionNoGenerics\Holder2::foo
18+
}
19+
20+
class Holder3
21+
{
22+
const CONST1 = 1;
23+
const CONST2 = 2; // error: Unused ReflectionNoGenerics\Holder3::CONST2
24+
25+
public function bar() {}
26+
}
27+
28+
function testNoGenericTypeKnown(\ReflectionClass $reflection) {
29+
echo $reflection->getConstant('CONST1'); // marks all constants named CONST1 as used
30+
echo $reflection->getMethod('bar'); // marks all methods named bar as used
31+
echo $reflection->getMethods(); // we ignore mixed over mixed calls
32+
}

tests/Rule/data/providers/reflection.php

+11
Original file line numberDiff line numberDiff line change
@@ -127,3 +127,14 @@ function testMemberOnlyInDescendant(string $fqn) {
127127
echo $classReflection->getConstant('NOT_IN_PARENT');
128128
}
129129
}
130+
131+
/**
132+
* @param class-string<HolderParent> $fqn
133+
*/
134+
function testNoGenericTypeKnown(string $fqn) {
135+
$classReflection = new \ReflectionClass($fqn);
136+
137+
if ($classReflection->hasConstant('NOT_IN_PARENT')) {
138+
echo $classReflection->getConstant('NOT_IN_PARENT');
139+
}
140+
}

0 commit comments

Comments
 (0)