rector/rules/Php80/Rector/FuncCall/TokenGetAllToObjectRector.php
Tomas Votruba 816016aac7 Updated Rector to commit 294bea2d18
294bea2d18 [FileProcessor] Run untill the file is fixed completelly (#432)
2021-07-15 03:40:51 +00:00

153 lines
5.6 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\Php80\Rector\FuncCall;
use PhpParser\Node;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\ClassMethod;
use PhpParser\Node\Stmt\Foreach_;
use PhpParser\Node\Stmt\Function_;
use PhpParser\Node\Stmt\If_;
use Rector\Core\Rector\AbstractRector;
use Rector\NodeNestingScope\ParentFinder;
use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\Php80\NodeManipulator\TokenManipulator;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @changelog https://wiki.php.net/rfc/token_as_object
*
* @see \Rector\Tests\Php80\Rector\FuncCall\TokenGetAllToObjectRector\TokenGetAllToObjectRectorTest
*/
final class TokenGetAllToObjectRector extends \Rector\Core\Rector\AbstractRector
{
/**
* @var \Rector\Php80\NodeManipulator\TokenManipulator
*/
private $tokenManipulator;
/**
* @var \Rector\NodeNestingScope\ParentFinder
*/
private $parentFinder;
public function __construct(\Rector\Php80\NodeManipulator\TokenManipulator $tokenManipulator, \Rector\NodeNestingScope\ParentFinder $parentFinder)
{
$this->tokenManipulator = $tokenManipulator;
$this->parentFinder = $parentFinder;
}
public function getRuleDefinition() : \Symplify\RuleDocGenerator\ValueObject\RuleDefinition
{
return new \Symplify\RuleDocGenerator\ValueObject\RuleDefinition('Convert `token_get_all` to `PhpToken::tokenize`', [new \Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample(<<<'CODE_SAMPLE'
final class SomeClass
{
public function run()
{
$tokens = token_get_all($code);
foreach ($tokens as $token) {
if (is_array($token)) {
$name = token_name($token[0]);
$text = $token[1];
} else {
$name = null;
$text = $token;
}
}
}
}
CODE_SAMPLE
, <<<'CODE_SAMPLE'
final class SomeClass
{
public function run()
{
$tokens = \PhpToken::tokenize($code);
foreach ($tokens as $phpToken) {
$name = $phpToken->getTokenName();
$text = $phpToken->text;
}
}
}
CODE_SAMPLE
)]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes() : array
{
return [\PhpParser\Node\Expr\FuncCall::class];
}
/**
* @param FuncCall $node
*/
public function refactor(\PhpParser\Node $node) : ?\PhpParser\Node
{
if (!$this->nodeNameResolver->isName($node, 'token_get_all')) {
return null;
}
$this->refactorTokensVariable($node);
return $this->nodeFactory->createStaticCall('PhpToken', 'tokenize', $node->args);
}
private function refactorTokensVariable(\PhpParser\Node\Expr\FuncCall $funcCall) : void
{
$assign = $funcCall->getAttribute(\Rector\NodeTypeResolver\Node\AttributeKey::PARENT_NODE);
if (!$assign instanceof \PhpParser\Node\Expr\Assign) {
return;
}
/** @var ClassMethod|Function_|null $classMethodOrFunction */
$classMethodOrFunction = $this->parentFinder->findByTypes($funcCall, [\PhpParser\Node\Stmt\ClassMethod::class, \PhpParser\Node\Stmt\Function_::class]);
if ($classMethodOrFunction === null) {
return;
}
// dummy approach, improve when needed
$this->replaceGetNameOrGetValue($classMethodOrFunction, $assign->var);
}
/**
* @param \PhpParser\Node\Stmt\ClassMethod|\PhpParser\Node\Stmt\Function_ $functionLike
*/
private function replaceGetNameOrGetValue($functionLike, \PhpParser\Node\Expr $assignedExpr) : void
{
$tokensForeaches = $this->findForeachesOverTokenVariable($functionLike, $assignedExpr);
foreach ($tokensForeaches as $tokenForeach) {
$this->refactorTokenInForeach($tokenForeach);
}
}
/**
* @return Foreach_[]
* @param \PhpParser\Node\Stmt\ClassMethod|\PhpParser\Node\Stmt\Function_ $functionLike
*/
private function findForeachesOverTokenVariable($functionLike, \PhpParser\Node\Expr $assignedExpr) : array
{
return $this->betterNodeFinder->find((array) $functionLike->stmts, function (\PhpParser\Node $node) use($assignedExpr) : bool {
if (!$node instanceof \PhpParser\Node\Stmt\Foreach_) {
return \false;
}
return $this->nodeComparator->areNodesEqual($node->expr, $assignedExpr);
});
}
private function refactorTokenInForeach(\PhpParser\Node\Stmt\Foreach_ $tokensForeach) : void
{
$singleToken = $tokensForeach->valueVar;
if (!$singleToken instanceof \PhpParser\Node\Expr\Variable) {
return;
}
$this->traverseNodesWithCallable($tokensForeach, function (\PhpParser\Node $node) use($singleToken) {
$this->tokenManipulator->refactorArrayToken([$node], $singleToken);
$this->tokenManipulator->refactorNonArrayToken([$node], $singleToken);
$this->tokenManipulator->refactorTokenIsKind([$node], $singleToken);
$this->tokenManipulator->removeIsArray([$node], $singleToken);
// drop if "If_" node not needed
if ($node instanceof \PhpParser\Node\Stmt\If_ && $node->else !== null) {
if (!$this->nodeComparator->areNodesEqual($node->stmts, $node->else->stmts)) {
return null;
}
$this->unwrapStmts($node->stmts, $node);
$this->removeNode($node);
}
});
}
}