-
Notifications
You must be signed in to change notification settings - Fork 473
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
f11480b
commit 217fac3
Showing
4 changed files
with
154 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
62 changes: 62 additions & 0 deletions
62
src/Rules/Comparison/WhileLoopAlwaysFalseConditionRule.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,62 @@ | ||
<?php declare(strict_types = 1); | ||
|
||
namespace PHPStan\Rules\Comparison; | ||
|
||
use PhpParser\Node\Stmt\While_; | ||
use PHPStan\Rules\RuleErrorBuilder; | ||
use PHPStan\Type\Constant\ConstantBooleanType; | ||
|
||
/** | ||
* @implements \PHPStan\Rules\Rule<While_> | ||
*/ | ||
class WhileLoopAlwaysFalseConditionRule implements \PHPStan\Rules\Rule | ||
{ | ||
|
||
private ConstantConditionRuleHelper $helper; | ||
|
||
private bool $treatPhpDocTypesAsCertain; | ||
|
||
public function __construct( | ||
ConstantConditionRuleHelper $helper, | ||
bool $treatPhpDocTypesAsCertain | ||
) | ||
{ | ||
$this->helper = $helper; | ||
$this->treatPhpDocTypesAsCertain = $treatPhpDocTypesAsCertain; | ||
} | ||
|
||
public function getNodeType(): string | ||
{ | ||
return While_::class; | ||
} | ||
|
||
public function processNode( | ||
\PhpParser\Node $node, | ||
\PHPStan\Analyser\Scope $scope | ||
): array | ||
{ | ||
$exprType = $this->helper->getBooleanType($scope, $node->cond); | ||
if ($exprType instanceof ConstantBooleanType && !$exprType->getValue()) { | ||
$addTip = function (RuleErrorBuilder $ruleErrorBuilder) use ($scope, $node): RuleErrorBuilder { | ||
if (!$this->treatPhpDocTypesAsCertain) { | ||
return $ruleErrorBuilder; | ||
} | ||
|
||
$booleanNativeType = $this->helper->getNativeBooleanType($scope, $node->cond); | ||
if ($booleanNativeType instanceof ConstantBooleanType) { | ||
return $ruleErrorBuilder; | ||
} | ||
|
||
return $ruleErrorBuilder->tip('Because the type is coming from a PHPDoc, you can turn off this check by setting <fg=cyan>treatPhpDocTypesAsCertain: false</> in your <fg=cyan>%configurationFile%</>.'); | ||
}; | ||
|
||
return [ | ||
$addTip(RuleErrorBuilder::message('While loop condition is always false.'))->line($node->cond->getLine()) | ||
->build(), | ||
]; | ||
} | ||
|
||
return []; | ||
} | ||
|
||
} |
50 changes: 50 additions & 0 deletions
50
tests/PHPStan/Rules/Comparison/WhileLoopAlwaysFalseConditionRuleTest.php
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
<?php declare(strict_types = 1); | ||
|
||
namespace PHPStan\Rules\Comparison; | ||
|
||
/** | ||
* @extends \PHPStan\Testing\RuleTestCase<WhileLoopAlwaysFalseConditionRule> | ||
*/ | ||
class WhileLoopAlwaysFalseConditionRuleTest extends \PHPStan\Testing\RuleTestCase | ||
{ | ||
|
||
/** @var bool */ | ||
private $treatPhpDocTypesAsCertain = true; | ||
|
||
protected function getRule(): \PHPStan\Rules\Rule | ||
{ | ||
return new WhileLoopAlwaysFalseConditionRule( | ||
new ConstantConditionRuleHelper( | ||
new ImpossibleCheckTypeHelper( | ||
$this->createReflectionProvider(), | ||
$this->getTypeSpecifier(), | ||
[], | ||
$this->treatPhpDocTypesAsCertain | ||
), | ||
$this->treatPhpDocTypesAsCertain | ||
), | ||
$this->treatPhpDocTypesAsCertain | ||
); | ||
} | ||
|
||
protected function shouldTreatPhpDocTypesAsCertain(): bool | ||
{ | ||
return $this->treatPhpDocTypesAsCertain; | ||
} | ||
|
||
public function testRule(): void | ||
{ | ||
$this->analyse([__DIR__ . '/data/while-loop-false.php'], [ | ||
[ | ||
'While loop condition is always false.', | ||
10, | ||
], | ||
[ | ||
'While loop condition is always false.', | ||
20, | ||
'Because the type is coming from a PHPDoc, you can turn off this check by setting <fg=cyan>treatPhpDocTypesAsCertain: false</> in your <fg=cyan>%configurationFile%</>.', | ||
], | ||
]); | ||
} | ||
|
||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
<?php | ||
|
||
namespace WhileLoopFalse; | ||
|
||
class Foo | ||
{ | ||
|
||
public function doFoo(): void | ||
{ | ||
while (false) { | ||
|
||
} | ||
} | ||
|
||
/** | ||
* @param 0 $s | ||
*/ | ||
public function doBar($s): void | ||
{ | ||
while ($s) { | ||
|
||
} | ||
} | ||
|
||
/** | ||
* @param string $s | ||
*/ | ||
public function doBar2($s): void | ||
{ | ||
while ($s === null) { // reported by StrictComparisonOfDifferentTypesRule | ||
|
||
} | ||
} | ||
|
||
} |