diff --git a/conf/config.level4.neon b/conf/config.level4.neon index 31b3cf0d0d..43fe1bddf6 100644 --- a/conf/config.level4.neon +++ b/conf/config.level4.neon @@ -144,6 +144,13 @@ services: tags: - phpstan.rules.rule + - + class: PHPStan\Rules\Comparison\WhileLoopAlwaysTrueConditionRule + arguments: + treatPhpDocTypesAsCertain: %treatPhpDocTypesAsCertain% + tags: + - phpstan.rules.rule + - class: PHPStan\Rules\TooWideTypehints\TooWideMethodReturnTypehintRule arguments: diff --git a/src/Analyser/NodeScopeResolver.php b/src/Analyser/NodeScopeResolver.php index ad1ca635e3..8c6dc0db93 100644 --- a/src/Analyser/NodeScopeResolver.php +++ b/src/Analyser/NodeScopeResolver.php @@ -56,6 +56,7 @@ use PHPStan\File\FileReader; use PHPStan\Node\BooleanAndNode; use PHPStan\Node\BooleanOrNode; +use PHPStan\Node\BreaklessWhileLoopNode; use PHPStan\Node\CatchWithUnthrownExceptionNode; use PHPStan\Node\ClassConstantsNode; use PHPStan\Node\ClassMethodsNode; @@ -899,9 +900,13 @@ private function processStmtNode( $isIterableAtLeastOnce = $beforeCondBooleanType instanceof ConstantBooleanType && $beforeCondBooleanType->getValue(); $alwaysIterates = $condBooleanType instanceof ConstantBooleanType && $condBooleanType->getValue(); $neverIterates = $condBooleanType instanceof ConstantBooleanType && !$condBooleanType->getValue(); + $breakCount = count($finalScopeResult->getExitPointsByType(Break_::class)); + if ($breakCount === 0) { + $nodeCallback(new BreaklessWhileLoopNode($stmt), $bodyScopeMaybeRan); + } if ($alwaysIterates) { - $isAlwaysTerminating = count($finalScopeResult->getExitPointsByType(Break_::class)) === 0; + $isAlwaysTerminating = $breakCount === 0; } elseif ($isIterableAtLeastOnce) { $isAlwaysTerminating = $finalScopeResult->isAlwaysTerminating(); } else { diff --git a/src/Node/BreaklessWhileLoopNode.php b/src/Node/BreaklessWhileLoopNode.php new file mode 100644 index 0000000000..f662a139fe --- /dev/null +++ b/src/Node/BreaklessWhileLoopNode.php @@ -0,0 +1,38 @@ +getAttributes()); + $this->originalNode = $originalNode; + } + + public function getOriginalNode(): While_ + { + return $this->originalNode; + } + + public function getType(): string + { + return 'PHPStan_Node_BreaklessWhileLoop'; + } + + /** + * @return string[] + */ + public function getSubNodeNames(): array + { + return []; + } + +} diff --git a/src/Rules/Comparison/WhileLoopAlwaysTrueConditionRule.php b/src/Rules/Comparison/WhileLoopAlwaysTrueConditionRule.php new file mode 100644 index 0000000000..15d696408b --- /dev/null +++ b/src/Rules/Comparison/WhileLoopAlwaysTrueConditionRule.php @@ -0,0 +1,63 @@ + + */ +class WhileLoopAlwaysTrueConditionRule 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 BreaklessWhileLoopNode::class; + } + + public function processNode( + \PhpParser\Node $node, + \PHPStan\Analyser\Scope $scope + ): array + { + $originalNode = $node->getOriginalNode(); + $exprType = $this->helper->getBooleanType($scope, $originalNode->cond); + if ($exprType instanceof ConstantBooleanType && $exprType->getValue()) { + $addTip = function (RuleErrorBuilder $ruleErrorBuilder) use ($scope, $originalNode): RuleErrorBuilder { + if (!$this->treatPhpDocTypesAsCertain) { + return $ruleErrorBuilder; + } + + $booleanNativeType = $this->helper->getNativeBooleanType($scope, $originalNode->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 treatPhpDocTypesAsCertain: false in your %configurationFile%.'); + }; + + return [ + $addTip(RuleErrorBuilder::message('While loop condition is always true.'))->line($originalNode->cond->getLine()) + ->build(), + ]; + } + + return []; + } + +} diff --git a/tests/PHPStan/Rules/Comparison/WhileLoopAlwaysTrueConditionRuleTest.php b/tests/PHPStan/Rules/Comparison/WhileLoopAlwaysTrueConditionRuleTest.php new file mode 100644 index 0000000000..f827c11f93 --- /dev/null +++ b/tests/PHPStan/Rules/Comparison/WhileLoopAlwaysTrueConditionRuleTest.php @@ -0,0 +1,50 @@ + + */ +class WhileLoopAlwaysTrueConditionRuleTest extends \PHPStan\Testing\RuleTestCase +{ + + /** @var bool */ + private $treatPhpDocTypesAsCertain = true; + + protected function getRule(): \PHPStan\Rules\Rule + { + return new WhileLoopAlwaysTrueConditionRule( + 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-true.php'], [ + [ + 'While loop condition is always true.', + 10, + ], + [ + 'While loop condition is always true.', + 20, + 'Because the type is coming from a PHPDoc, you can turn off this check by setting treatPhpDocTypesAsCertain: false in your %configurationFile%.', + ], + ]); + } + +} diff --git a/tests/PHPStan/Rules/Comparison/data/while-loop-true.php b/tests/PHPStan/Rules/Comparison/data/while-loop-true.php new file mode 100644 index 0000000000..717e753e9d --- /dev/null +++ b/tests/PHPStan/Rules/Comparison/data/while-loop-true.php @@ -0,0 +1,54 @@ +