Skip to content

Commit 676d440

Browse files
committed
fix wrong inClassLike after stack pop
1 parent f3fb80a commit 676d440

File tree

3 files changed

+27
-2
lines changed

3 files changed

+27
-2
lines changed

src/Parser/VariadicMethodsVisitor.php

+12-2
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,11 @@
1010
use PHPStan\TrinaryLogic;
1111
use function array_key_exists;
1212
use function array_pop;
13+
use function count;
1314
use function implode;
1415
use function in_array;
1516
use function sprintf;
17+
use function str_contains;
1618

1719
final class VariadicMethodsVisitor extends NodeVisitorAbstract
1820
{
@@ -33,6 +35,8 @@ final class VariadicMethodsVisitor extends NodeVisitorAbstract
3335

3436
public const ATTRIBUTE_NAME = 'variadicMethods';
3537

38+
private const ANONYMOUS_CLASS_PREFIX = 'class@anonymous';
39+
3640
public function beforeTraverse(array $nodes): ?array
3741
{
3842
$this->topNode = null;
@@ -57,7 +61,7 @@ public function enterNode(Node $node): ?Node
5761

5862
if ($node instanceof Node\Stmt\ClassLike) {
5963
if (!$node->name instanceof Node\Identifier) {
60-
$className = sprintf('class@anonymous:%s:%s', $node->getStartLine(), $node->getEndLine());
64+
$className = sprintf('%s:%s:%s', self::ANONYMOUS_CLASS_PREFIX, $node->getStartLine(), $node->getEndLine());
6165
$this->classStack[] = $className;
6266
$this->inClassLike = $className; // anonymous classes are in global namespace
6367
} else {
@@ -105,7 +109,13 @@ public function leaveNode(Node $node): ?Node
105109
array_pop($this->classStack);
106110

107111
if ($this->classStack !== []) {
108-
$this->inClassLike = $this->inNamespace !== null ? $this->inNamespace . '\\' . implode('\\', $this->classStack) : implode('\\', $this->classStack);
112+
$lastClass = $this->classStack[count($this->classStack) - 1];
113+
114+
if (str_contains($lastClass, self::ANONYMOUS_CLASS_PREFIX)) {
115+
$this->inClassLike = $lastClass;
116+
} else {
117+
$this->inClassLike = $this->inNamespace !== null ? $this->inNamespace . '\\' . implode('\\', $this->classStack) : implode('\\', $this->classStack);
118+
}
109119
} else {
110120
$this->inClassLike = null;
111121
}

tests/PHPStan/Parser/ParserTest.php

+5
Original file line numberDiff line numberDiff line change
@@ -53,6 +53,11 @@ public function dataVariadicCallLikes(): iterable
5353
'implicit_variadic_fn' => TrinaryLogic::createYes(),
5454
],
5555
'class@anonymous:54:54' => [],
56+
'class@anonymous:61:68' => [
57+
'nestedClass' => TrinaryLogic::createNo(),
58+
'implicit_variadic_fn' => TrinaryLogic::createYes(),
59+
],
60+
'class@anonymous:63:63' => [],
5661
],
5762
];
5863

tests/PHPStan/Parser/data/variadic-methods.php

+10
Original file line numberDiff line numberDiff line change
@@ -56,3 +56,13 @@ function implicit_variadic_fn() {
5656
$args = func_get_args();
5757
}
5858
};
59+
60+
61+
$c = new class () {
62+
function nestedClass() {
63+
$nested = new class () {};
64+
}
65+
function implicit_variadic_fn() {
66+
$args = func_get_args();
67+
}
68+
};

0 commit comments

Comments
 (0)