Skip to content

Commit ae5562f

Browse files
committed
Fixed false positive about undefined method guarded with method_exists
1 parent 3d52930 commit ae5562f

File tree

6 files changed

+61
-3
lines changed

6 files changed

+61
-3
lines changed

src/Rules/Methods/CallMethodsRule.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -58,7 +58,7 @@ public function processNode(Node $node, Scope $scope): array
5858
*/
5959
private function processSingleMethodCall(Scope $scope, MethodCall $node, string $methodName): array
6060
{
61-
[$errors, $methodReflection] = $this->methodCallCheck->check($scope, $methodName, $node->var);
61+
[$errors, $methodReflection] = $this->methodCallCheck->check($scope, $methodName, $node->var, $node->name);
6262
if ($methodReflection === null) {
6363
return $errors;
6464
}

src/Rules/Methods/MethodCallCheck.php

+15
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,10 @@
22

33
namespace PHPStan\Rules\Methods;
44

5+
use PhpParser\Node\Arg;
56
use PhpParser\Node\Expr;
7+
use PhpParser\Node\Identifier;
8+
use PhpParser\Node\Name\FullyQualified;
69
use PHPStan\Analyser\NullsafeOperatorHelper;
710
use PHPStan\Analyser\Scope;
811
use PHPStan\Internal\SprintfHelper;
@@ -38,6 +41,7 @@ public function check(
3841
Scope $scope,
3942
string $methodName,
4043
Expr $var,
44+
Identifier|Expr $astName,
4145
): array
4246
{
4347
$typeResult = $this->ruleLevelHelper->findTypeToCheck(
@@ -107,6 +111,17 @@ public function check(
107111
}
108112
}
109113

114+
if ($astName instanceof Expr) {
115+
$methodExistsExpr = new Expr\FuncCall(new FullyQualified('method_exists'), [
116+
new Arg($var),
117+
new Arg($astName),
118+
]);
119+
120+
if ($scope->getType($methodExistsExpr)->isTrue()->yes()) {
121+
return [[], null];
122+
}
123+
}
124+
110125
return [
111126
[
112127
RuleErrorBuilder::message(sprintf(

src/Rules/Methods/MethodCallableRule.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ public function processNode(Node $node, Scope $scope): array
4444

4545
$methodNameName = $methodName->toString();
4646

47-
[$errors, $methodReflection] = $this->methodCallCheck->check($scope, $methodNameName, $node->getVar());
47+
[$errors, $methodReflection] = $this->methodCallCheck->check($scope, $methodNameName, $node->getVar(), $node->getName());
4848
if ($methodReflection === null) {
4949
return $errors;
5050
}

src/Type/Php/MethodExistsTypeSpecifyingExtension.php

+8-1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
namespace PHPStan\Type\Php;
44

55
use PhpParser\Node\Expr\FuncCall;
6+
use PhpParser\Node\Name\FullyQualified;
67
use PHPStan\Analyser\Scope;
78
use PHPStan\Analyser\SpecifiedTypes;
89
use PHPStan\Analyser\TypeSpecifier;
@@ -11,6 +12,7 @@
1112
use PHPStan\Reflection\FunctionReflection;
1213
use PHPStan\Type\Accessory\HasMethodType;
1314
use PHPStan\Type\ClassStringType;
15+
use PHPStan\Type\Constant\ConstantBooleanType;
1416
use PHPStan\Type\Constant\ConstantStringType;
1517
use PHPStan\Type\FunctionTypeSpecifyingExtension;
1618
use PHPStan\Type\IntersectionType;
@@ -48,7 +50,12 @@ public function specifyTypes(
4850
{
4951
$methodNameType = $scope->getType($node->getArgs()[1]->value);
5052
if (!$methodNameType instanceof ConstantStringType) {
51-
return new SpecifiedTypes([], []);
53+
return $this->typeSpecifier->create(
54+
new FuncCall(new FullyQualified('method_exists'), $node->getRawArgs()),
55+
new ConstantBooleanType(true),
56+
$context,
57+
$scope,
58+
);
5259
}
5360

5461
$objectType = $scope->getType($node->getArgs()[0]->value);

tests/PHPStan/Rules/Methods/CallMethodsRuleTest.php

+10
Original file line numberDiff line numberDiff line change
@@ -3587,4 +3587,14 @@ public function testDynamicCall(): void
35873587
]);
35883588
}
35893589

3590+
public function testBu12793(): void
3591+
{
3592+
$this->checkThisOnly = false;
3593+
$this->checkNullables = true;
3594+
$this->checkUnionTypes = true;
3595+
$this->checkExplicitMixed = true;
3596+
3597+
$this->analyse([__DIR__ . '/data/bug-12793.php'], []);
3598+
}
3599+
35903600
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
<?php
2+
3+
namespace Bug12793;
4+
5+
class Service
6+
{
7+
protected function defaults(Model $model): void
8+
{
9+
$columns = [
10+
'getCreatedByColumn',
11+
'getUpdatedByColumn',
12+
'getDeletedByColumn',
13+
'getCreatedAtColumn',
14+
'getUpdatedAtColumn',
15+
'getDeletedAtColumn',
16+
];
17+
18+
foreach ($columns as $column) {
19+
if (method_exists($model, $column)) {
20+
$model->{$column}();
21+
}
22+
}
23+
}
24+
}
25+
26+
class Model {}

0 commit comments

Comments
 (0)