Skip to content

Commit c50b71f

Browse files
committed
Do not report missing implementation abstract method from trait when it's implicitly implemented by enum
1 parent 1cba4ba commit c50b71f

File tree

5 files changed

+119
-15
lines changed

5 files changed

+119
-15
lines changed

src/Rules/Classes/EnumSanityRule.php

-13
Original file line numberDiff line numberDiff line change
@@ -47,20 +47,7 @@ public function processNode(Node $node, Scope $scope): array
4747
$errors = [];
4848

4949
foreach ($enumNode->getMethods() as $methodNode) {
50-
if ($methodNode->isAbstract()) {
51-
$errors[] = RuleErrorBuilder::message(sprintf(
52-
'Enum %s contains abstract method %s().',
53-
$classReflection->getDisplayName(),
54-
$methodNode->name->name,
55-
))
56-
->identifier('enum.abstractMethod')
57-
->line($methodNode->getStartLine())
58-
->nonIgnorable()
59-
->build();
60-
}
61-
6250
$lowercasedMethodName = $methodNode->name->toLowerString();
63-
6451
if ($methodNode->isMagic()) {
6552
if ($lowercasedMethodName === '__construct') {
6653
$errors[] = RuleErrorBuilder::message(sprintf(

src/Rules/Methods/AbstractMethodInNonAbstractClassRule.php

+13
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@
77
use PHPStan\Rules\Rule;
88
use PHPStan\Rules\RuleErrorBuilder;
99
use PHPStan\ShouldNotHappenException;
10+
use function in_array;
1011
use function sprintf;
1112

1213
/**
@@ -29,6 +30,18 @@ public function processNode(Node $node, Scope $scope): array
2930
$class = $scope->getClassReflection();
3031

3132
if (!$class->isAbstract() && $node->isAbstract()) {
33+
if ($class->isEnum()) {
34+
$lowercasedMethodName = $node->name->toLowerString();
35+
if ($lowercasedMethodName === 'cases') {
36+
return [];
37+
}
38+
if ($class->isBackedEnum()) {
39+
if (in_array($lowercasedMethodName, ['from', 'tryfrom'], true)) {
40+
return [];
41+
}
42+
}
43+
}
44+
3245
$description = $class->getClassTypeDescription();
3346
return [
3447
RuleErrorBuilder::message(sprintf(

tests/PHPStan/Rules/Classes/EnumSanityRuleTest.php

+29-2
Original file line numberDiff line numberDiff line change
@@ -24,10 +24,11 @@ public function testRule(): void
2424
}
2525

2626
$expected = [
27-
[
27+
/*[
28+
// reported by AbstractMethodInNonAbstractClassRule
2829
'Enum EnumSanity\EnumWithAbstractMethod contains abstract method foo().',
2930
7,
30-
],
31+
],*/
3132
[
3233
'Enum EnumSanity\EnumWithConstructorAndDestructor contains constructor.',
3334
12,
@@ -123,4 +124,30 @@ public function testBug9402(): void
123124
]);
124125
}
125126

127+
public function testBug11592(): void
128+
{
129+
if (PHP_VERSION_ID < 80100) {
130+
$this->markTestSkipped('Test requires PHP 8.1');
131+
}
132+
133+
$this->analyse([__DIR__ . '/data/bug-11592.php'], [
134+
[
135+
'Enum Bug11592\Test2 cannot redeclare native method cases().',
136+
22,
137+
],
138+
[
139+
'Enum Bug11592\BackedTest2 cannot redeclare native method cases().',
140+
37,
141+
],
142+
[
143+
'Enum Bug11592\BackedTest2 cannot redeclare native method from().',
144+
39,
145+
],
146+
[
147+
'Enum Bug11592\BackedTest2 cannot redeclare native method tryFrom().',
148+
41,
149+
],
150+
]);
151+
}
152+
126153
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
<?php // lint >= 8.1
2+
3+
namespace Bug11592;
4+
5+
trait HelloWorld
6+
{
7+
abstract public static function cases(): array;
8+
9+
abstract public static function from(): self;
10+
11+
abstract public static function tryFrom(): ?self;
12+
}
13+
14+
enum Test
15+
{
16+
use HelloWorld;
17+
}
18+
19+
enum Test2
20+
{
21+
22+
abstract public static function cases(): array;
23+
24+
abstract public static function from(): self;
25+
26+
abstract public static function tryFrom(): ?self;
27+
28+
}
29+
30+
enum BackedTest: int
31+
{
32+
use HelloWorld;
33+
}
34+
35+
enum BackedTest2: int
36+
{
37+
abstract public static function cases(): array;
38+
39+
abstract public static function from(): self;
40+
41+
abstract public static function tryFrom(): ?self;
42+
}
43+
44+
enum EnumWithAbstractMethod
45+
{
46+
abstract function foo();
47+
}

tests/PHPStan/Rules/Methods/AbstractMethodInNonAbstractClassRuleTest.php

+30
Original file line numberDiff line numberDiff line change
@@ -89,4 +89,34 @@ public function testEnum(): void
8989
]);
9090
}
9191

92+
public function testBug11592(): void
93+
{
94+
if (PHP_VERSION_ID < 80100) {
95+
$this->markTestSkipped('Test requires PHP 8.1');
96+
}
97+
98+
$this->analyse([__DIR__ . '/../Classes/data/bug-11592.php'], [
99+
[
100+
'Enum Bug11592\Test contains abstract method from().',
101+
9,
102+
],
103+
[
104+
'Enum Bug11592\Test contains abstract method tryFrom().',
105+
11,
106+
],
107+
[
108+
'Enum Bug11592\Test2 contains abstract method from().',
109+
24,
110+
],
111+
[
112+
'Enum Bug11592\Test2 contains abstract method tryFrom().',
113+
26,
114+
],
115+
[
116+
'Enum Bug11592\EnumWithAbstractMethod contains abstract method foo().',
117+
46,
118+
],
119+
]);
120+
}
121+
92122
}

0 commit comments

Comments
 (0)