rector/rules/CodeQuality/NodeAnalyzer/ForeachAnalyzer.php

131 lines
4.7 KiB
PHP
Raw Normal View History

<?php
declare (strict_types=1);
namespace Rector\CodeQuality\NodeAnalyzer;
use PhpParser\Node;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\ArrayDimFetch;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt;
use PhpParser\Node\Stmt\Expression;
use PhpParser\Node\Stmt\Foreach_;
use Rector\Core\PhpParser\Comparing\NodeComparator;
use Rector\Core\PhpParser\Node\BetterNodeFinder;
use Rector\NodeNameResolver\NodeNameResolver;
use Rector\PhpDocParser\NodeTraverser\SimpleCallableNodeTraverser;
final class ForeachAnalyzer
{
/**
* @readonly
* @var \Rector\Core\PhpParser\Comparing\NodeComparator
*/
private $nodeComparator;
/**
* @readonly
* @var \Rector\CodeQuality\NodeAnalyzer\ForAnalyzer
*/
private $forAnalyzer;
/**
* @readonly
* @var \Rector\NodeNameResolver\NodeNameResolver
*/
private $nodeNameResolver;
/**
* @readonly
* @var \Rector\PhpDocParser\NodeTraverser\SimpleCallableNodeTraverser
*/
private $simpleCallableNodeTraverser;
/**
* @readonly
* @var \Rector\Core\PhpParser\Node\BetterNodeFinder
*/
private $betterNodeFinder;
/**
* @readonly
* @var \Rector\CodeQuality\NodeAnalyzer\VariableNameUsedNextAnalyzer
*/
private $variableNameUsedNextAnalyzer;
public function __construct(NodeComparator $nodeComparator, \Rector\CodeQuality\NodeAnalyzer\ForAnalyzer $forAnalyzer, NodeNameResolver $nodeNameResolver, SimpleCallableNodeTraverser $simpleCallableNodeTraverser, BetterNodeFinder $betterNodeFinder, \Rector\CodeQuality\NodeAnalyzer\VariableNameUsedNextAnalyzer $variableNameUsedNextAnalyzer)
{
$this->nodeComparator = $nodeComparator;
$this->forAnalyzer = $forAnalyzer;
$this->nodeNameResolver = $nodeNameResolver;
$this->simpleCallableNodeTraverser = $simpleCallableNodeTraverser;
$this->betterNodeFinder = $betterNodeFinder;
$this->variableNameUsedNextAnalyzer = $variableNameUsedNextAnalyzer;
}
/**
* Matches$
* foreach ($values as $value) {
* <$assigns[]> = $value;
* }
*/
public function matchAssignItemsOnlyForeachArrayVariable(Foreach_ $foreach) : ?Expr
{
if (\count($foreach->stmts) !== 1) {
return null;
}
$onlyStatement = $foreach->stmts[0];
if ($onlyStatement instanceof Expression) {
$onlyStatement = $onlyStatement->expr;
}
if (!$onlyStatement instanceof Assign) {
return null;
}
if (!$onlyStatement->var instanceof ArrayDimFetch) {
return null;
}
if ($onlyStatement->var->dim instanceof Expr) {
return null;
}
if (!$this->nodeComparator->areNodesEqual($foreach->valueVar, $onlyStatement->expr)) {
return null;
}
return $onlyStatement->var->var;
}
/**
* @param Stmt[] $stmts
*/
public function useForeachVariableInStmts(Expr $foreachedValue, Expr $singleValue, array $stmts, string $keyValueName) : void
{
$this->simpleCallableNodeTraverser->traverseNodesWithCallable($stmts, function (Node $node) use($foreachedValue, $singleValue, $keyValueName) : ?Expr {
if (!$node instanceof ArrayDimFetch) {
return null;
}
// must be the same as foreach value
if (!$this->nodeComparator->areNodesEqual($node->var, $foreachedValue)) {
return null;
}
if ($this->forAnalyzer->isArrayDimFetchPartOfAssignOrArgParentCount($node)) {
return null;
}
// is dim same as key value name, ...[$i]
if (!$node->dim instanceof Variable) {
return null;
}
if (!$this->nodeNameResolver->isName($node->dim, $keyValueName)) {
return null;
}
return $singleValue;
});
}
public function isValueVarUsed(Foreach_ $foreach, string $singularValueVarName) : bool
{
$isUsedInStmts = (bool) $this->betterNodeFinder->findFirst($foreach->stmts, function (Node $node) use($singularValueVarName) : bool {
if (!$node instanceof Variable) {
return \false;
}
return $this->nodeNameResolver->isName($node, $singularValueVarName);
});
if ($isUsedInStmts) {
return \true;
}
if ($this->variableNameUsedNextAnalyzer->isValueVarUsedNext($foreach, $singularValueVarName)) {
return \true;
}
return $this->variableNameUsedNextAnalyzer->isValueVarUsedNext($foreach, (string) $this->nodeNameResolver->getName($foreach->valueVar));
}
}