-
Notifications
You must be signed in to change notification settings - Fork 51
/
Copy pathDynamicCallOnStaticMethodsRule.php
67 lines (55 loc) · 1.51 KB
/
DynamicCallOnStaticMethodsRule.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
<?php declare(strict_types = 1);
namespace PHPStan\Rules\StrictCalls;
use PhpParser\Node;
use PhpParser\Node\Expr\MethodCall;
use PHPStan\Analyser\Scope;
use PHPStan\Rules\Rule;
use PHPStan\Rules\RuleLevelHelper;
use PHPStan\TrinaryLogic;
use PHPStan\Type\ErrorType;
use PHPStan\Type\Type;
use function sprintf;
class DynamicCallOnStaticMethodsRule implements Rule
{
/** @var RuleLevelHelper */
private $ruleLevelHelper;
public function __construct(RuleLevelHelper $ruleLevelHelper)
{
$this->ruleLevelHelper = $ruleLevelHelper;
}
public function getNodeType(): string
{
return MethodCall::class;
}
/**
* @param MethodCall $node
* @return string[]
*/
public function processNode(Node $node, Scope $scope): array
{
if (!$node->name instanceof Node\Identifier) {
return [];
}
$name = $node->name->name;
$type = $this->ruleLevelHelper->findTypeToCheck(
$scope,
$node->var,
'',
static function (Type $type) use ($name): bool {
return $type->canCallMethods()->yes() && $type->hasMethod($name)->yes();
}
)->getType();
if ($type instanceof ErrorType || !$type->canCallMethods()->yes() || !$type->hasMethod($name)->yes()) {
return [];
}
$methodReflection = $type->getMethod($name, $scope);
if ($methodReflection->isStatic() && $type->hasMethod('__callStatic') !== TrinaryLogic::createYes()) {
return [sprintf(
'Dynamic call to static method %s::%s().',
$methodReflection->getDeclaringClass()->getDisplayName(),
$methodReflection->getName()
)];
}
return [];
}
}