mirror of
https://github.com/rectorphp/rector.git
synced 2024-06-03 18:00:50 +00:00
96112cb1f0
2da49992cc
[Downgrade] [PHP 7.2] Make DowngradeParameterTypeWideningRector always downgrade to phpdoc type (#390)
139 lines
5.4 KiB
PHP
139 lines
5.4 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\EarlyReturn\Rector\Foreach_;
|
|
|
|
use PhpParser\Node;
|
|
use PhpParser\Node\Expr;
|
|
use PhpParser\Node\Expr\Assign;
|
|
use PhpParser\Node\Stmt\Break_;
|
|
use PhpParser\Node\Stmt\Expression;
|
|
use PhpParser\Node\Stmt\Foreach_;
|
|
use PhpParser\Node\Stmt\Return_;
|
|
use Rector\Core\Rector\AbstractRector;
|
|
use Rector\NodeTypeResolver\Node\AttributeKey;
|
|
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
|
|
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
|
|
/**
|
|
* @see \Rector\Tests\EarlyReturn\Rector\Foreach_\ReturnAfterToEarlyOnBreakRector\ReturnAfterToEarlyOnBreakRectorTest
|
|
*/
|
|
final class ReturnAfterToEarlyOnBreakRector extends \Rector\Core\Rector\AbstractRector
|
|
{
|
|
public function getRuleDefinition() : \Symplify\RuleDocGenerator\ValueObject\RuleDefinition
|
|
{
|
|
return new \Symplify\RuleDocGenerator\ValueObject\RuleDefinition('Change return after foreach to early return in foreach on break', [new \Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample(<<<'CODE_SAMPLE'
|
|
class SomeClass
|
|
{
|
|
public function run(array $pathConstants, string $allowedPath)
|
|
{
|
|
$pathOK = false;
|
|
|
|
foreach ($pathConstants as $allowedPath) {
|
|
if ($dirPath == $allowedPath) {
|
|
$pathOK = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return $pathOK;
|
|
}
|
|
}
|
|
CODE_SAMPLE
|
|
, <<<'CODE_SAMPLE'
|
|
class SomeClass
|
|
{
|
|
public function run(array $pathConstants, string $allowedPath)
|
|
{
|
|
foreach ($pathConstants as $allowedPath) {
|
|
if ($dirPath == $allowedPath) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
CODE_SAMPLE
|
|
)]);
|
|
}
|
|
/**
|
|
* @return array<class-string<Node>>
|
|
*/
|
|
public function getNodeTypes() : array
|
|
{
|
|
return [\PhpParser\Node\Stmt\Foreach_::class];
|
|
}
|
|
/**
|
|
* @param Foreach_ $node
|
|
*/
|
|
public function refactor(\PhpParser\Node $node) : ?\PhpParser\Node
|
|
{
|
|
/** @var Break_[] $breaks */
|
|
$breaks = $this->betterNodeFinder->findInstanceOf($node->stmts, \PhpParser\Node\Stmt\Break_::class);
|
|
if (\count($breaks) !== 1) {
|
|
return null;
|
|
}
|
|
$beforeBreak = $breaks[0]->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::PREVIOUS_NODE);
|
|
if (!$beforeBreak instanceof \PhpParser\Node\Stmt\Expression) {
|
|
return null;
|
|
}
|
|
$assign = $beforeBreak->expr;
|
|
if (!$assign instanceof \PhpParser\Node\Expr\Assign) {
|
|
return null;
|
|
}
|
|
$nextForeach = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::NEXT_NODE);
|
|
if (!$nextForeach instanceof \PhpParser\Node\Stmt\Return_) {
|
|
return null;
|
|
}
|
|
$assignVariable = $assign->var;
|
|
/** @var Expr $variablePrevious */
|
|
$variablePrevious = $this->betterNodeFinder->findFirstPreviousOfNode($node, function (\PhpParser\Node $node) use($assignVariable) : bool {
|
|
$parent = $node->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::PARENT_NODE);
|
|
if (!$parent instanceof \PhpParser\Node\Expr\Assign) {
|
|
return \false;
|
|
}
|
|
return $this->nodeComparator->areNodesEqual($node, $assignVariable);
|
|
});
|
|
if ($this->shouldSkip($nextForeach, $node, $assignVariable, $variablePrevious)) {
|
|
return null;
|
|
}
|
|
/** @var Assign $assignPreviousVariable */
|
|
$assignPreviousVariable = $variablePrevious->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::PARENT_NODE);
|
|
$parent = $assignPreviousVariable->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::PARENT_NODE);
|
|
if (!$parent instanceof \PhpParser\Node\Stmt\Expression) {
|
|
return null;
|
|
}
|
|
$nextParent = $parent->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::NEXT_NODE);
|
|
if ($nextParent !== $node) {
|
|
return null;
|
|
}
|
|
return $this->processEarlyReturn($beforeBreak, $assign, $breaks, $nextForeach, $assignPreviousVariable, $node);
|
|
}
|
|
/**
|
|
* @param Break_[] $breaks
|
|
*/
|
|
private function processEarlyReturn(\PhpParser\Node\Stmt\Expression $expression, \PhpParser\Node\Expr\Assign $assign, array $breaks, \PhpParser\Node\Stmt\Return_ $return, \PhpParser\Node\Expr\Assign $assignPreviousVariable, \PhpParser\Node\Stmt\Foreach_ $foreach) : \PhpParser\Node\Stmt\Foreach_
|
|
{
|
|
$this->removeNode($expression);
|
|
$this->addNodeBeforeNode(new \PhpParser\Node\Stmt\Return_($assign->expr), $breaks[0]);
|
|
$this->removeNode($breaks[0]);
|
|
$return->expr = $assignPreviousVariable->expr;
|
|
$this->removeNode($assignPreviousVariable);
|
|
return $foreach;
|
|
}
|
|
private function shouldSkip(\PhpParser\Node\Stmt\Return_ $return, \PhpParser\Node\Stmt\Foreach_ $foreach, \PhpParser\Node\Expr $assignVariable, ?\PhpParser\Node\Expr $expr = null) : bool
|
|
{
|
|
if (!$expr instanceof \PhpParser\Node\Expr) {
|
|
return \true;
|
|
}
|
|
if (!$this->nodeComparator->areNodesEqual($return->expr, $expr)) {
|
|
return \true;
|
|
}
|
|
// ensure the variable only used once in foreach
|
|
$usedVariable = $this->betterNodeFinder->find($foreach->stmts, function (\PhpParser\Node $node) use($assignVariable) : bool {
|
|
return $this->nodeComparator->areNodesEqual($node, $assignVariable);
|
|
});
|
|
return \count($usedVariable) > 1;
|
|
}
|
|
}
|