rector/rules/Naming/Naming/OverridenExistingNamesResol...

98 lines
3.7 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\Naming\Naming;
use PhpParser\Node\Expr\ArrowFunction;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\Closure;
use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\ClassMethod;
use PhpParser\Node\Stmt\Function_;
use Rector\Naming\PhpArray\ArrayFilter;
use Rector\NodeNameResolver\NodeNameResolver;
use Rector\PhpParser\Node\BetterNodeFinder;
final class OverridenExistingNamesResolver
{
/**
* @readonly
* @var \Rector\Naming\PhpArray\ArrayFilter
*/
private $arrayFilter;
/**
* @readonly
* @var \Rector\PhpParser\Node\BetterNodeFinder
*/
private $betterNodeFinder;
/**
* @readonly
* @var \Rector\NodeNameResolver\NodeNameResolver
*/
private $nodeNameResolver;
/**
* @var array<int, array<int, string>>
*/
private $overridenExistingVariableNamesByClassMethod = [];
public function __construct(ArrayFilter $arrayFilter, BetterNodeFinder $betterNodeFinder, NodeNameResolver $nodeNameResolver)
{
$this->arrayFilter = $arrayFilter;
$this->betterNodeFinder = $betterNodeFinder;
$this->nodeNameResolver = $nodeNameResolver;
}
/**
* @param \PhpParser\Node\Stmt\ClassMethod|\PhpParser\Node\Stmt\Function_|\PhpParser\Node\Expr\Closure $functionLike
*/
public function hasNameInClassMethodForNew(string $variableName, $functionLike) : bool
{
$overridenVariableNames = $this->resolveOveriddenNamesForNew($functionLike);
return \in_array($variableName, $overridenVariableNames, \true);
}
/**
* @param \PhpParser\Node\Stmt\ClassMethod|\PhpParser\Node\Stmt\Function_|\PhpParser\Node\Expr\Closure|\PhpParser\Node\Expr\ArrowFunction $classMethod
*/
public function hasNameInFunctionLikeForParam(string $expectedName, $classMethod) : bool
{
/** @var Assign[] $assigns */
$assigns = $this->betterNodeFinder->findInstanceOf((array) $classMethod->getStmts(), Assign::class);
$usedVariableNames = [];
foreach ($assigns as $assign) {
if (!$assign->var instanceof Variable) {
continue;
}
$variableName = $this->nodeNameResolver->getName($assign->var);
if ($variableName === null) {
continue;
}
$usedVariableNames[] = $variableName;
}
return \in_array($expectedName, $usedVariableNames, \true);
}
/**
* @return string[]
* @param \PhpParser\Node\Stmt\ClassMethod|\PhpParser\Node\Stmt\Function_|\PhpParser\Node\Expr\Closure $functionLike
*/
private function resolveOveriddenNamesForNew($functionLike) : array
{
$classMethodId = \spl_object_id($functionLike);
if (isset($this->overridenExistingVariableNamesByClassMethod[$classMethodId])) {
return $this->overridenExistingVariableNamesByClassMethod[$classMethodId];
}
$currentlyUsedNames = [];
/** @var Assign[] $assigns */
$assigns = $this->betterNodeFinder->findInstanceOf((array) $functionLike->stmts, Assign::class);
foreach ($assigns as $assign) {
/** @var Variable $assignVariable */
$assignVariable = $assign->var;
$currentVariableName = $this->nodeNameResolver->getName($assignVariable);
if ($currentVariableName === null) {
continue;
}
$currentlyUsedNames[] = $currentVariableName;
}
$currentlyUsedNames = \array_values($currentlyUsedNames);
$currentlyUsedNames = $this->arrayFilter->filterWithAtLeastTwoOccurences($currentlyUsedNames);
$this->overridenExistingVariableNamesByClassMethod[$classMethodId] = $currentlyUsedNames;
return $currentlyUsedNames;
}
}