mirror of
https://github.com/rectorphp/rector.git
synced 2025-02-24 03:35:01 +01:00
160 lines
4.7 KiB
PHP
160 lines
4.7 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace Rector\Naming;
|
|
|
|
use PhpParser\Node;
|
|
use PhpParser\Node\Expr\Assign;
|
|
use PhpParser\Node\Expr\Closure;
|
|
use PhpParser\Node\Expr\Variable;
|
|
use PhpParser\Node\FunctionLike;
|
|
use PhpParser\Node\Param;
|
|
use PhpParser\Node\Stmt\ClassMethod;
|
|
use PhpParser\Node\Stmt\Function_;
|
|
use Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfo;
|
|
use Rector\BetterPhpDocParser\PhpDocInfo\PhpDocInfoFactory;
|
|
use Rector\Core\PhpParser\Node\BetterNodeFinder;
|
|
use Rector\Naming\PhpDoc\VarTagValueNodeRenamer;
|
|
use Rector\NodeNameResolver\NodeNameResolver;
|
|
use Rector\NodeTypeResolver\Node\AttributeKey;
|
|
use Symplify\Astral\NodeTraverser\SimpleCallableNodeTraverser;
|
|
|
|
final class VariableRenamer
|
|
{
|
|
/**
|
|
* @var SimpleCallableNodeTraverser
|
|
*/
|
|
private $simpleCallableNodeTraverser;
|
|
|
|
/**
|
|
* @var NodeNameResolver
|
|
*/
|
|
private $nodeNameResolver;
|
|
|
|
/**
|
|
* @var VarTagValueNodeRenamer
|
|
*/
|
|
private $varTagValueNodeRenamer;
|
|
|
|
/**
|
|
* @var PhpDocInfoFactory
|
|
*/
|
|
private $phpDocInfoFactory;
|
|
|
|
/**
|
|
* @var BetterNodeFinder
|
|
*/
|
|
private $betterNodeFinder;
|
|
|
|
public function __construct(
|
|
SimpleCallableNodeTraverser $simpleCallableNodeTraverser,
|
|
NodeNameResolver $nodeNameResolver,
|
|
VarTagValueNodeRenamer $varTagValueNodeRenamer,
|
|
PhpDocInfoFactory $phpDocInfoFactory,
|
|
BetterNodeFinder $betterNodeFinder
|
|
) {
|
|
$this->simpleCallableNodeTraverser = $simpleCallableNodeTraverser;
|
|
$this->nodeNameResolver = $nodeNameResolver;
|
|
$this->varTagValueNodeRenamer = $varTagValueNodeRenamer;
|
|
$this->phpDocInfoFactory = $phpDocInfoFactory;
|
|
$this->betterNodeFinder = $betterNodeFinder;
|
|
}
|
|
|
|
/**
|
|
* @param ClassMethod|Function_|Closure $functionLike
|
|
*/
|
|
public function renameVariableInFunctionLike(
|
|
FunctionLike $functionLike,
|
|
?Assign $assign = null,
|
|
string $oldName,
|
|
string $expectedName
|
|
): void {
|
|
$isRenamingActive = false;
|
|
|
|
if ($assign === null) {
|
|
$isRenamingActive = true;
|
|
}
|
|
|
|
$this->simpleCallableNodeTraverser->traverseNodesWithCallable(
|
|
(array) $functionLike->stmts,
|
|
function (Node $node) use ($oldName, $expectedName, $assign, &$isRenamingActive): ?Variable {
|
|
if ($assign !== null && $node === $assign) {
|
|
$isRenamingActive = true;
|
|
return null;
|
|
}
|
|
|
|
if (! $node instanceof Variable) {
|
|
return null;
|
|
}
|
|
|
|
// skip param names
|
|
$parent = $node->getAttribute(AttributeKey::PARENT_NODE);
|
|
if ($parent instanceof Param) {
|
|
return null;
|
|
}
|
|
|
|
// TODO: Remove in next PR (with above param check?),
|
|
// TODO: Should be implemented in BreakingVariableRenameGuard::shouldSkipParam()
|
|
if ($this->isParamInParentFunction($node)) {
|
|
return null;
|
|
}
|
|
|
|
if (! $isRenamingActive) {
|
|
return null;
|
|
}
|
|
|
|
return $this->renameVariableIfMatchesName($node, $oldName, $expectedName);
|
|
}
|
|
);
|
|
}
|
|
|
|
private function isParamInParentFunction(Variable $variable): bool
|
|
{
|
|
$closure = $this->betterNodeFinder->findParentType($variable, Closure::class);
|
|
if (! $closure instanceof Closure) {
|
|
return false;
|
|
}
|
|
|
|
$variableName = $this->nodeNameResolver->getName($variable);
|
|
if ($variableName === null) {
|
|
return false;
|
|
}
|
|
|
|
foreach ($closure->params as $param) {
|
|
if ($this->nodeNameResolver->isName($param, $variableName)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function renameVariableIfMatchesName(Variable $variable, string $oldName, string $expectedName): ?Variable
|
|
{
|
|
if (! $this->nodeNameResolver->isName($variable, $oldName)) {
|
|
return null;
|
|
}
|
|
|
|
$variable->name = $expectedName;
|
|
|
|
$variablePhpDocInfo = $this->resolvePhpDocInfo($variable);
|
|
$this->varTagValueNodeRenamer->renameAssignVarTagVariableName($variablePhpDocInfo, $oldName, $expectedName);
|
|
|
|
return $variable;
|
|
}
|
|
|
|
/**
|
|
* Expression doc block has higher priority
|
|
*/
|
|
private function resolvePhpDocInfo(Variable $variable): PhpDocInfo
|
|
{
|
|
$expression = $variable->getAttribute(AttributeKey::CURRENT_STATEMENT);
|
|
if ($expression instanceof Node) {
|
|
return $this->phpDocInfoFactory->createFromNodeOrEmpty($expression);
|
|
}
|
|
|
|
return $this->phpDocInfoFactory->createFromNodeOrEmpty($variable);
|
|
}
|
|
}
|