Skip to content

Commit d17f8d0

Browse files
committed
Fix for handling non-stringable types in dynamic class constant access
1 parent 11ee70e commit d17f8d0

File tree

3 files changed

+70
-3
lines changed

3 files changed

+70
-3
lines changed

src/Rules/Classes/ClassConstantRule.php

+26-3
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
use PHPStan\Reflection\ReflectionProvider;
1212
use PHPStan\Rules\ClassNameCheck;
1313
use PHPStan\Rules\ClassNameNodePair;
14+
use PHPStan\Rules\IdentifierRuleError;
1415
use PHPStan\Rules\Rule;
1516
use PHPStan\Rules\RuleErrorBuilder;
1617
use PHPStan\Rules\RuleLevelHelper;
@@ -20,6 +21,7 @@
2021
use PHPStan\Type\Type;
2122
use PHPStan\Type\TypeCombinator;
2223
use PHPStan\Type\VerbosityLevel;
24+
use function array_map;
2325
use function array_merge;
2426
use function in_array;
2527
use function sprintf;
@@ -47,11 +49,32 @@ public function getNodeType(): string
4749

4850
public function processNode(Node $node, Scope $scope): array
4951
{
50-
if (!$node->name instanceof Node\Identifier) {
51-
return [];
52+
$errors = [];
53+
if ($node->name instanceof Node\Identifier) {
54+
$constantNames = [$node->name->name];
55+
} else {
56+
$fetchType = $scope->getType($node->name);
57+
$constantNames = array_map(static fn ($type): string => $type->getValue(), $fetchType->getConstantStrings());
58+
$fetchStringType = $fetchType->toString();
59+
if (!$fetchStringType->isString()->yes()) {
60+
$errors[] = RuleErrorBuilder::message(sprintf('Cannot fetch class constant with a non-stringable type %s.', $fetchType->describe(VerbosityLevel::typeOnly())))
61+
->identifier('classConstant.fetchInvalidExpression')
62+
->build();
63+
}
64+
}
65+
66+
foreach ($constantNames as $constantName) {
67+
$errors = array_merge($errors, $this->processSingleClassConstFetch($scope, $node, $constantName));
5268
}
53-
$constantName = $node->name->name;
5469

70+
return $errors;
71+
}
72+
73+
/**
74+
* @return list<IdentifierRuleError>
75+
*/
76+
private function processSingleClassConstFetch(Scope $scope, ClassConstFetch $node, string $constantName): array
77+
{
5578
$class = $node->class;
5679
$messages = [];
5780
if ($class instanceof Node\Name) {

tests/PHPStan/Rules/Classes/ClassConstantRuleTest.php

+24
Original file line numberDiff line numberDiff line change
@@ -435,4 +435,28 @@ public function testClassConstantAccessedOnTrait(): void
435435
]);
436436
}
437437

438+
public function testBug9475(): void
439+
{
440+
if (PHP_VERSION_ID < 80200) {
441+
$this->markTestSkipped('Test requires PHP 8.2.');
442+
}
443+
444+
$this->phpVersion = PHP_VERSION_ID;
445+
446+
$this->analyse([__DIR__ . '/data/bug-9475.php'], [
447+
[
448+
'Cannot fetch class constant with a non-stringable type $this(Bug9475\Classes).',
449+
12,
450+
],
451+
[
452+
'Cannot fetch class constant with a non-stringable type object.',
453+
13,
454+
],
455+
[
456+
'Cannot fetch class constant with a non-stringable type array.',
457+
14,
458+
],
459+
]);
460+
}
461+
438462
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
<?php // lint >=8.3
2+
3+
namespace Bug9475;
4+
5+
use Stringable;
6+
7+
final class Classes
8+
{
9+
10+
public function testStaticMethods(string $name, Stringable $stringable, object $object, array $array): void
11+
{
12+
echo 'Hello, ' . self::{$this};
13+
echo 'Hello, ' . self::{$object};
14+
echo 'Hello, ' . self::{$array};
15+
16+
echo 'Hello, ' . self::{$name}; // valid
17+
echo 'Hello, ' . self::{$stringable}; // valid
18+
}
19+
20+
}

0 commit comments

Comments
 (0)