rector/packages/NodeNestingScope/ContextAnalyzer.php
Tomas Votruba 679f8a6710 Updated Rector to commit 0389ec89d79713b46f280cc6f028783c1d4e48b8
0389ec89d7 Improve ArraySpreadInsteadOfArrayMergeRector (#3568) (#3589)
2023-04-08 16:24:17 +00:00

99 lines
3.2 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\NodeNestingScope;
use PhpParser\Node;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\FunctionLike;
use PhpParser\Node\Stmt\ClassMethod;
use PhpParser\Node\Stmt\If_;
use PhpParser\Node\Stmt\Return_;
use PhpParser\Node\Stmt\Switch_;
use PHPStan\Type\ObjectType;
use Rector\Core\PhpParser\Node\BetterNodeFinder;
use Rector\NodeNestingScope\ValueObject\ControlStructure;
use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\NodeTypeResolver\NodeTypeResolver;
final class ContextAnalyzer
{
/**
* Nodes that break the scope they way up, e.g. class method
* @var array<class-string<FunctionLike>>
*/
private const BREAK_NODES = [FunctionLike::class, ClassMethod::class];
/**
* @readonly
* @var \Rector\Core\PhpParser\Node\BetterNodeFinder
*/
private $betterNodeFinder;
/**
* @readonly
* @var \Rector\NodeTypeResolver\NodeTypeResolver
*/
private $nodeTypeResolver;
public function __construct(BetterNodeFinder $betterNodeFinder, NodeTypeResolver $nodeTypeResolver)
{
$this->betterNodeFinder = $betterNodeFinder;
$this->nodeTypeResolver = $nodeTypeResolver;
}
public function isInLoop(Node $node) : bool
{
$item0Unpacked = ControlStructure::LOOP_NODES;
$item1Unpacked = self::BREAK_NODES;
$firstParent = $this->betterNodeFinder->findParentByTypes($node, \array_merge($item0Unpacked, $item1Unpacked));
if (!$firstParent instanceof Node) {
return \false;
}
foreach (ControlStructure::LOOP_NODES as $type) {
if (\is_a($firstParent, $type, \true)) {
return \true;
}
}
return \false;
}
/**
* @api
*/
public function isInSwitch(Node $node) : bool
{
return (bool) $this->betterNodeFinder->findParentType($node, Switch_::class);
}
/**
* @api
*/
public function isInIf(Node $node) : bool
{
$item1Unpacked = self::BREAK_NODES;
$previousNode = $this->betterNodeFinder->findParentByTypes($node, \array_merge([If_::class], $item1Unpacked));
if (!$previousNode instanceof Node) {
return \false;
}
return $previousNode instanceof If_;
}
public function hasAssignWithIndirectReturn(Node $node, If_ $if) : bool
{
foreach (ControlStructure::LOOP_NODES as $loopNode) {
$loopObjectType = new ObjectType($loopNode);
$parentType = $this->nodeTypeResolver->getType($node);
$superType = $parentType->isSuperTypeOf($loopObjectType);
if (!$superType->yes()) {
continue;
}
$nextNode = $node->getAttribute(AttributeKey::NEXT_NODE);
if ($nextNode instanceof Node) {
if ($nextNode instanceof Return_ && !$nextNode->expr instanceof Expr) {
continue;
}
$hasAssign = (bool) $this->betterNodeFinder->findFirstInstanceOf($if->stmts, Assign::class);
if (!$hasAssign) {
continue;
}
return \true;
}
}
return \false;
}
}