Skip to content

Commit ef9ef4f

Browse files
committed
[symfony] Add RouteGenerateControllerClassRequireNameRule
1 parent c7ed282 commit ef9ef4f

14 files changed

+375
-0
lines changed
Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
<?php
2+
3+
namespace Symplify\PHPStanRules\Reflection;
4+
5+
use PHPStan\Reflection\ClassReflection;
6+
use ReflectionMethod;
7+
8+
final class InvokeClassMethodResolver
9+
{
10+
public static function resolve(ClassReflection $controllerClassReflection): null|ReflectionMethod
11+
{
12+
if (! $controllerClassReflection->hasMethod('__invoke')) {
13+
return null;
14+
}
15+
16+
$nativeReflectionClass = $controllerClassReflection->getNativeReflection();
17+
return $nativeReflectionClass->getMethod('__invoke');
18+
}
19+
}
Lines changed: 156 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,156 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Rules\Symfony;
6+
7+
use PhpParser\Node;
8+
use PhpParser\Node\Expr\MethodCall;
9+
use PhpParser\Node\Identifier;
10+
use PHPStan\Analyser\Scope;
11+
use PHPStan\Reflection\ClassReflection;
12+
use PHPStan\Reflection\ReflectionProvider;
13+
use PHPStan\Rules\Rule;
14+
use PHPStan\Rules\RuleError;
15+
use PHPStan\Rules\RuleErrorBuilder;
16+
use PHPStan\Type\Constant\ConstantStringType;
17+
use PHPStan\Type\ObjectType;
18+
use ReflectionAttribute;
19+
use ReflectionMethod;
20+
use Symfony\Component\Routing\Generator\UrlGeneratorInterface;
21+
use Symfony\Component\Routing\Route;
22+
23+
/**
24+
* To pass a controller class in $this->router->generate(SomeController::class),
25+
* the controller must be present #[Route(name:: self::class)
26+
*
27+
* @see https://symfony.com/blog/new-in-symfony-6-4-fqcn-based-routes
28+
*
29+
* @implements Rule<MethodCall>
30+
*
31+
* @see \Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\RouteGenerateControllerClassRequireNameRuleTest
32+
*/
33+
final readonly class RouteGenerateControllerClassRequireNameRule implements Rule
34+
{
35+
/**
36+
* @api
37+
* @var string
38+
*/
39+
public const ERROR_MESSAGE = 'To pass a controller class to generate() method, the controller must have "#[Route(name: self::class)]" above the __invoke() method';
40+
41+
/**
42+
* @var string
43+
*/
44+
private const IDENTIFIER = 'be5.routeGenerateControllerName';
45+
46+
public function __construct(
47+
private ReflectionProvider $reflectionProvider,
48+
) {
49+
}
50+
51+
public function getNodeType(): string
52+
{
53+
return MethodCall::class;
54+
}
55+
56+
/**
57+
* @param MethodCall $node
58+
* @return RuleError[]
59+
*/
60+
public function processNode(Node $node, Scope $scope): array
61+
{
62+
if (! $this->isRouterGenerateMethodCall($node, $scope)) {
63+
return [];
64+
}
65+
66+
$controllerClassReflection = $this->matchControllerFirstArgClassReflection($node, $scope);
67+
68+
// the used argument is not a class reference
69+
if (! $controllerClassReflection instanceof ClassReflection) {
70+
return [];
71+
}
72+
73+
$invokeClassMethodReflection = \Symplify\PHPStanRules\Reflection\InvokeClassMethodResolver::resolve($controllerClassReflection);
74+
75+
// there must be __invoke() method
76+
if (! $invokeClassMethodReflection instanceof ReflectionMethod) {
77+
return [RuleErrorBuilder::message(self::ERROR_MESSAGE)->identifier(self::IDENTIFIER)->build()];
78+
}
79+
80+
$routeAttributes = $this->findRouteAttributes($invokeClassMethodReflection);
81+
if ($this->hasAtLeastOneRouteWithSelfClassName($routeAttributes, $controllerClassReflection)) {
82+
return [];
83+
}
84+
85+
return [RuleErrorBuilder::message(self::ERROR_MESSAGE)->identifier(self::IDENTIFIER)->build()];
86+
}
87+
88+
private function isRouterGenerateMethodCall(MethodCall $methodCall, Scope $scope): bool
89+
{
90+
if ($methodCall->isFirstClassCallable()) {
91+
return false;
92+
}
93+
94+
if (! $methodCall->name instanceof Identifier) {
95+
return false;
96+
}
97+
98+
if ($methodCall->name->toString() !== 'generate') {
99+
return false;
100+
}
101+
102+
$callerType = $scope->getType($methodCall->var);
103+
if (! $callerType instanceof ObjectType) {
104+
return false;
105+
}
106+
107+
return $callerType->isInstanceOf(UrlGeneratorInterface::class)->yes();
108+
}
109+
110+
private function matchControllerFirstArgClassReflection(MethodCall $methodCall, Scope $scope): ?ClassReflection
111+
{
112+
$firstArg = $methodCall->getArgs()[0];
113+
$argType = $scope->getType($firstArg->value);
114+
115+
// we look for a controller class reference
116+
if (! $argType instanceof ConstantStringType) {
117+
return null;
118+
}
119+
120+
$controllerClass = $argType->getValue();
121+
if (! $this->reflectionProvider->hasClass($controllerClass)) {
122+
return null;
123+
}
124+
125+
return $this->reflectionProvider->getClass($controllerClass);
126+
}
127+
128+
/**
129+
* @return ReflectionAttribute[]
130+
*/
131+
private function findRouteAttributes(ReflectionMethod $reflectionMethod): array
132+
{
133+
return array_merge(
134+
$reflectionMethod->getAttributes(\Symfony\Component\Routing\Attribute\Route::class),
135+
$reflectionMethod->getAttributes(Route::class),
136+
$reflectionMethod->getAttributes(\Symfony\Component\Routing\Annotation\Route::class)
137+
);
138+
}
139+
140+
/**
141+
* @param ReflectionAttribute[] $routeAttributes
142+
*/
143+
private function hasAtLeastOneRouteWithSelfClassName(array $routeAttributes, ClassReflection $classReflection): bool
144+
{
145+
foreach ($routeAttributes as $routeAttribute) {
146+
$routeName = $routeAttribute->getArguments()['name'] ?? null;
147+
148+
// name must be same as current controller class
149+
if ($routeName === $classReflection->getName()) {
150+
return true;
151+
}
152+
}
153+
154+
return false;
155+
}
156+
}
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Fixture;
6+
7+
use Symfony\Component\Routing\RouterInterface;
8+
use Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source\SomeControllerWithStringNameClass;
9+
10+
final class CallingControllerWithWrongString
11+
{
12+
public function run(RouterInterface $router)
13+
{
14+
$url = $router->generate(SomeControllerWithStringNameClass::class);
15+
}
16+
}
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Fixture;
6+
7+
use Symfony\Component\Routing\RouterInterface;
8+
use Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source\WithoutInvokeController;
9+
10+
final class CallingControllerWithoutInvoke
11+
{
12+
public function run(RouterInterface $router)
13+
{
14+
$url = $router->generate(WithoutInvokeController::class);
15+
}
16+
}
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Fixture;
6+
7+
use Symfony\Component\Routing\RouterInterface;
8+
use Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source\SomeControllerWithRouteClass;
9+
10+
final class CallingCorrectController
11+
{
12+
public function run(RouterInterface $router)
13+
{
14+
$url = $router->generate(SomeControllerWithRouteClass::class);
15+
}
16+
}
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Fixture;
6+
7+
use Symfony\Component\Routing\RouterInterface;
8+
use Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source\SomeControllerWIthoutRouteClass;
9+
10+
final class CallingWrongController
11+
{
12+
public function run(RouterInterface $router)
13+
{
14+
$url = $router->generate(SomeControllerWIthoutRouteClass::class);
15+
}
16+
}
Lines changed: 16 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Fixture;
6+
7+
use Symfony\Component\Routing\RouterInterface;
8+
use Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source\TwoRoutesController;
9+
10+
final class TwoRoutes
11+
{
12+
public function run(RouterInterface $router)
13+
{
14+
$url = $router->generate(TwoRoutesController::class);
15+
}
16+
}
Lines changed: 52 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,52 @@
1+
<?php
2+
3+
declare(strict_types=1);
4+
5+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule;
6+
7+
use Iterator;
8+
use PHPStan\Reflection\ReflectionProvider;
9+
use PHPStan\Rules\Rule;
10+
use PHPStan\Testing\RuleTestCase;
11+
use PHPUnit\Framework\Attributes\DataProvider;
12+
use Symplify\PHPStanRules\Rules\Symfony\RouteGenerateControllerClassRequireNameRule;
13+
14+
/**
15+
* @extends RuleTestCase<RouteGenerateControllerClassRequireNameRule>
16+
*/
17+
final class RouteGenerateControllerClassRequireNameRuleTest extends RuleTestCase
18+
{
19+
/**
20+
* @param mixed[] $expectedErrorMessagesWithLines
21+
*/
22+
#[DataProvider('provideData')]
23+
public function testRule(string $filePath, array $expectedErrorMessagesWithLines): void
24+
{
25+
$this->analyse([$filePath], $expectedErrorMessagesWithLines);
26+
}
27+
28+
public static function provideData(): Iterator
29+
{
30+
yield [__DIR__ . '/Fixture/CallingCorrectController.php', []];
31+
yield [__DIR__ . '/Fixture/TwoRoutes.php', []];
32+
33+
yield [__DIR__ . '/Fixture/CallingWrongController.php', [
34+
[RouteGenerateControllerClassRequireNameRule::ERROR_MESSAGE, 14],
35+
]];
36+
37+
yield [__DIR__ . '/Fixture/CallingControllerWithoutInvoke.php', [
38+
[RouteGenerateControllerClassRequireNameRule::ERROR_MESSAGE, 14],
39+
]];
40+
41+
yield [__DIR__ . '/Fixture/CallingControllerWithWrongString.php', [
42+
[RouteGenerateControllerClassRequireNameRule::ERROR_MESSAGE, 14],
43+
]];
44+
}
45+
46+
protected function getRule(): Rule
47+
{
48+
$reflectionProvider = self::getContainer()->getByType(ReflectionProvider::class);
49+
50+
return new RouteGenerateControllerClassRequireNameRule($reflectionProvider);
51+
}
52+
}
Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<?php
2+
3+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source;
4+
5+
use Symfony\Component\Routing\Attribute\Route;
6+
7+
class SomeControllerWIthoutRouteClass
8+
{
9+
#[Route()]
10+
public function __invoke()
11+
{
12+
}
13+
}
Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
<?php
2+
3+
namespace Symplify\PHPStanRules\Tests\Rules\Symfony\RouteGenerateControllerClassRequireNameRule\Source;
4+
5+
use Symfony\Component\Routing\Attribute\Route;
6+
7+
class SomeControllerWithRouteClass
8+
{
9+
#[Route(name: self::class)]
10+
public function __invoke()
11+
{
12+
}
13+
}

0 commit comments

Comments
 (0)