rector/rules/Defluent/NodeAnalyzer/FluentChainMethodCallRootExtractor.php

197 lines
6.4 KiB
PHP
Raw Normal View History

2020-06-30 11:01:20 +02:00
<?php
declare(strict_types=1);
namespace Rector\Defluent\NodeAnalyzer;
2020-06-30 11:01:20 +02:00
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\PropertyFetch;
use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\Expression;
2020-06-30 11:01:20 +02:00
use PhpParser\Node\Stmt\Return_;
use Rector\Core\Exception\ShouldNotHappenException;
2020-06-30 11:01:20 +02:00
use Rector\Core\PhpParser\Node\BetterNodeFinder;
use Rector\Defluent\ValueObject\AssignAndRootExpr;
use Rector\Defluent\ValueObject\FluentCallsKind;
2020-06-30 11:01:20 +02:00
use Rector\Naming\Naming\PropertyNaming;
use Rector\Naming\Naming\VariableNaming;
use Rector\Naming\ValueObject\ExpectedName;
2020-06-30 11:01:20 +02:00
use Rector\NodeNameResolver\NodeNameResolver;
2020-08-12 20:24:52 +02:00
use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\NodeTypeResolver\NodeTypeResolver;
use Rector\StaticTypeMapper\ValueObject\Type\FullyQualifiedObjectType;
2020-06-30 11:01:20 +02:00
/**
* @see \Rector\Tests\Defluent\NodeFactory\FluentChainMethodCallRootExtractor\FluentChainMethodCallRootExtractorTest
*/
final class FluentChainMethodCallRootExtractor
2020-06-30 11:01:20 +02:00
{
/**
* @var PropertyNaming
*/
private $propertyNaming;
/**
* @var BetterNodeFinder
*/
private $betterNodeFinder;
/**
* @var NodeNameResolver
*/
private $nodeNameResolver;
2020-08-01 18:59:36 +02:00
/**
* @var VariableNaming
*/
private $variableNaming;
2020-08-12 20:24:52 +02:00
/**
* @var ExprStringTypeResolver
*/
private $exprStringTypeResolver;
/**
* @var NodeTypeResolver
*/
private $nodeTypeResolver;
2020-06-30 11:01:20 +02:00
public function __construct(
BetterNodeFinder $betterNodeFinder,
2020-07-26 09:49:22 +02:00
NodeNameResolver $nodeNameResolver,
2020-08-01 18:59:36 +02:00
PropertyNaming $propertyNaming,
2020-08-12 20:24:52 +02:00
VariableNaming $variableNaming,
ExprStringTypeResolver $exprStringTypeResolver,
NodeTypeResolver $nodeTypeResolver
2020-06-30 11:01:20 +02:00
) {
$this->propertyNaming = $propertyNaming;
$this->betterNodeFinder = $betterNodeFinder;
$this->nodeNameResolver = $nodeNameResolver;
2020-08-01 18:59:36 +02:00
$this->variableNaming = $variableNaming;
2020-08-12 20:24:52 +02:00
$this->exprStringTypeResolver = $exprStringTypeResolver;
$this->nodeTypeResolver = $nodeTypeResolver;
2020-06-30 11:01:20 +02:00
}
/**
* @param MethodCall[] $methodCalls
*/
2020-08-01 18:59:36 +02:00
public function extractFromMethodCalls(array $methodCalls, string $kind): ?AssignAndRootExpr
2020-06-30 11:01:20 +02:00
{
// we need at least 2 method call for fluent
if (count($methodCalls) < 2) {
return null;
}
2020-06-30 11:01:20 +02:00
foreach ($methodCalls as $methodCall) {
if ($methodCall->var instanceof Variable || $methodCall->var instanceof PropertyFetch) {
return $this->createAssignAndRootExprForVariableOrPropertyFetch($methodCall);
2020-06-30 11:01:20 +02:00
}
2020-06-30 11:01:20 +02:00
if ($methodCall->var instanceof New_) {
2020-08-01 18:59:36 +02:00
// direct = no parent
if ($kind === FluentCallsKind::IN_ARGS) {
return $this->resolveKindInArgs($methodCall);
2020-08-01 18:59:36 +02:00
}
return $this->matchMethodCallOnNew($methodCall->var);
2020-06-30 11:01:20 +02:00
}
}
return null;
}
2020-08-12 20:24:52 +02:00
/**
* beware: fluent vs. factory
* A. FLUENT: $cook->bake()->serve() // only "Cook"
* B. FACTORY: $food = $cook->bake()->warmUp(); // only "Food"
*/
public function resolveIsFirstMethodCallFactory(MethodCall $methodCall): bool
2020-08-12 20:24:52 +02:00
{
$variableStaticType = $this->exprStringTypeResolver->resolve($methodCall->var);
$calledMethodStaticType = $this->exprStringTypeResolver->resolve($methodCall);
// get next method call
$nextMethodCall = $methodCall->getAttribute(AttributeKey::PARENT_NODE);
if (! $nextMethodCall instanceof MethodCall) {
return false;
}
$nestedCallStaticType = $this->exprStringTypeResolver->resolve($nextMethodCall);
if ($nestedCallStaticType === null) {
return false;
}
if ($nestedCallStaticType !== $calledMethodStaticType) {
return false;
}
return $variableStaticType !== $calledMethodStaticType;
}
private function createAssignAndRootExprForVariableOrPropertyFetch(MethodCall $methodCall): AssignAndRootExpr
{
$isFirstCallFactory = $this->resolveIsFirstMethodCallFactory($methodCall);
// the method call, does not belong to the
$staticType = $this->nodeTypeResolver->getStaticType($methodCall);
$parentNode = $methodCall->getAttribute(AttributeKey::PARENT_NODE);
// no assign
if ($parentNode instanceof Expression) {
$variableName = $this->propertyNaming->fqnToVariableName($staticType);
// the assign expresison must be break
// pesuero code bsaed on type
$variable = new Variable($variableName);
return new AssignAndRootExpr($methodCall->var, $methodCall->var, $variable, $isFirstCallFactory);
}
return new AssignAndRootExpr($methodCall->var, $methodCall->var, null, $isFirstCallFactory);
}
private function resolveKindInArgs(MethodCall $methodCall): AssignAndRootExpr
{
$variableName = $this->variableNaming->resolveFromNode($methodCall->var);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
$silentVariable = new Variable($variableName);
return new AssignAndRootExpr($methodCall->var, $methodCall->var, $silentVariable);
}
private function matchMethodCallOnNew(New_ $new): ?AssignAndRootExpr
2020-06-30 11:01:20 +02:00
{
// we need assigned left variable here
$previousAssignOrReturn = $this->betterNodeFinder->findFirstPreviousOfTypes(
$new,
2020-06-30 11:01:20 +02:00
[Assign::class, Return_::class]
);
if ($previousAssignOrReturn instanceof Assign) {
return new AssignAndRootExpr($previousAssignOrReturn->var, $new);
2020-06-30 11:01:20 +02:00
}
if ($previousAssignOrReturn instanceof Return_) {
$className = $this->nodeNameResolver->getName($new->class);
2020-06-30 11:01:20 +02:00
if ($className === null) {
return null;
}
$fullyQualifiedObjectType = new FullyQualifiedObjectType($className);
$expectedName = $this->propertyNaming->getExpectedNameFromType($fullyQualifiedObjectType);
if (! $expectedName instanceof ExpectedName) {
2020-06-30 11:01:20 +02:00
return null;
}
$variable = new Variable($expectedName->getName());
return new AssignAndRootExpr($new, $new, $variable);
2020-06-30 11:01:20 +02:00
}
// no assign, just standalone call
return null;
}
}