refactoring of variable name counted (#3999)

* fix variable rename scoping

* [rector] fix variable rename scoping

* [cs] fix variable rename scoping

Co-authored-by: rector-bot <tomas@getrector.org>
This commit is contained in:
Tomas Votruba 2020-08-22 15:02:55 +02:00 committed by GitHub
parent 04597e9685
commit 786095f328
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
19 changed files with 213 additions and 131 deletions

View File

@ -20,7 +20,7 @@ return static function (ContainerConfigurator $containerConfigurator): void {
->exclude([ ->exclude([
__DIR__ . '/../src/Exception/*', __DIR__ . '/../src/Exception/*',
__DIR__ . '/../src/ValueObject/*', __DIR__ . '/../src/ValueObject/*',
__DIR__ . '/../src/Rector/*', __DIR__ . '/../src/Rector',
]); ]);
$services->set(AddNewServiceToSymfonyPhpConfigRector::class) $services->set(AddNewServiceToSymfonyPhpConfigRector::class)

View File

@ -3,6 +3,7 @@ includes:
- vendor/symplify/phpstan-extensions/config/config.neon - vendor/symplify/phpstan-extensions/config/config.neon
- vendor/thecodingmachine/phpstan-strict-rules/phpstan-strict-rules.neon - vendor/thecodingmachine/phpstan-strict-rules/phpstan-strict-rules.neon
- vendor/slam/phpstan-extensions/conf/slam-rules.neon - vendor/slam/phpstan-extensions/conf/slam-rules.neon
- vendor/phpstan/phpstan/conf/bleedingEdge.neon
# see https://github.com/symplify/coding-standard # see https://github.com/symplify/coding-standard
- vendor/symplify/coding-standard/config/symplify-rules.neon - vendor/symplify/coding-standard/config/symplify-rules.neon
@ -375,7 +376,6 @@ parameters:
# is nested expr # is nested expr
- '#Access to an undefined property PhpParser\\Node\\Expr\:\:\$expr#' - '#Access to an undefined property PhpParser\\Node\\Expr\:\:\$expr#'
- '#Cognitive complexity for "Rector\\DeadCode\\NodeManipulator\\LivingCodeManipulator\:\:keepLivingCodeFromExpr\(\)" is \d+, keep it under 9#' - '#Cognitive complexity for "Rector\\DeadCode\\NodeManipulator\\LivingCodeManipulator\:\:keepLivingCodeFromExpr\(\)" is \d+, keep it under 9#'
- '#Class with base "LexerFactory" name is already used in "PHPStan\\Parser\\LexerFactory", "Rector\\Core\\PhpParser\\Parser\\LexerFactory"\. Use unique name to make classes easy to recognize#'
- '#Parameter \#1 \$files of method Symplify\\SmartFileSystem\\Finder\\FinderSanitizer\:\:sanitize\(\) expects \(iterable<SplFileInfo\|string\>&Nette\\Utils\\Finder\)\|Symfony\\Component\\Finder\\Finder, array<string\> given#' - '#Parameter \#1 \$files of method Symplify\\SmartFileSystem\\Finder\\FinderSanitizer\:\:sanitize\(\) expects \(iterable<SplFileInfo\|string\>&Nette\\Utils\\Finder\)\|Symfony\\Component\\Finder\\Finder, array<string\> given#'
- '#Static property Rector\\Core\\Testing\\PHPUnit\\AbstractGenericRectorTestCase\:\:\$allRectorContainer \(Rector\\Naming\\Tests\\Rector\\Class_\\RenamePropertyToMatchTypeRector\\Source\\ContainerInterface\|Symfony\\Component\\DependencyInjection\\Container\|null\) does not accept Psr\\Container\\ContainerInterface#' - '#Static property Rector\\Core\\Testing\\PHPUnit\\AbstractGenericRectorTestCase\:\:\$allRectorContainer \(Rector\\Naming\\Tests\\Rector\\Class_\\RenamePropertyToMatchTypeRector\\Source\\ContainerInterface\|Symfony\\Component\\DependencyInjection\\Container\|null\) does not accept Psr\\Container\\ContainerInterface#'
# stubs # stubs

View File

@ -7,14 +7,12 @@ namespace Rector\CodeQuality\Rector\For_;
use PhpParser\Node; use PhpParser\Node;
use PhpParser\Node\Expr\Assign; use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Expr\FuncCall; use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\StaticCall;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\For_; use PhpParser\Node\Stmt\For_;
use PHPStan\Analyser\Scope;
use Rector\Core\Rector\AbstractRector; use Rector\Core\Rector\AbstractRector;
use Rector\Core\RectorDefinition\CodeSample; use Rector\Core\RectorDefinition\CodeSample;
use Rector\Core\RectorDefinition\RectorDefinition; use Rector\Core\RectorDefinition\RectorDefinition;
use Rector\NetteKdyby\Naming\VariableNaming;
use Rector\NodeTypeResolver\Node\AttributeKey; use Rector\NodeTypeResolver\Node\AttributeKey;
/** /**
@ -23,9 +21,14 @@ use Rector\NodeTypeResolver\Node\AttributeKey;
final class ForRepeatedCountToOwnVariableRector extends AbstractRector final class ForRepeatedCountToOwnVariableRector extends AbstractRector
{ {
/** /**
* @var string * @var VariableNaming
*/ */
private const DEFAULT_VARIABLE_COUNT_NAME = 'itemsCount'; private $variableNaming;
public function __construct(VariableNaming $variableNaming)
{
$this->variableNaming = $variableNaming;
}
public function getDefinition(): RectorDefinition public function getDefinition(): RectorDefinition
{ {
@ -73,13 +76,14 @@ PHP
public function refactor(Node $node): ?Node public function refactor(Node $node): ?Node
{ {
$countInCond = null; $countInCond = null;
$countedValueName = null; $variableName = null;
$for = $node;
$forScope = $node->getAttribute(AttributeKey::SCOPE);
$this->traverseNodesWithCallable($node->cond, function (Node $node) use ( $this->traverseNodesWithCallable($node->cond, function (Node $node) use (
&$countInCond, &$countInCond,
&$countedValueName, &$variableName,
$for $forScope
) { ) {
if (! $node instanceof FuncCall) { if (! $node instanceof FuncCall) {
return null; return null;
@ -91,36 +95,23 @@ PHP
$countInCond = $node; $countInCond = $node;
$valueName = $this->resolveValueName($node); $variableName = $this->variableNaming->resolveFromFuncCallFirstArgumentWithSuffix(
$countedValueName = $this->createCountedValueName($valueName, $for->getAttribute(AttributeKey::SCOPE)); $node,
'Count',
'itemsCount',
$forScope
);
return new Variable($countedValueName); return new Variable($variableName);
}); });
if ($countInCond === null || $countedValueName === null) { if ($countInCond === null || $variableName === null) {
return null; return null;
} }
$countAssign = new Assign(new Variable($countedValueName), $countInCond); $countAssign = new Assign(new Variable($variableName), $countInCond);
$this->addNodeBeforeNode($countAssign, $node); $this->addNodeBeforeNode($countAssign, $node);
return $node; return $node;
} }
protected function createCountedValueName(?string $valueName, ?Scope $scope): string
{
$countedValueName = $valueName === null ? self::DEFAULT_VARIABLE_COUNT_NAME : $valueName . 'Count';
return parent::createCountedValueName($countedValueName, $scope);
}
private function resolveValueName(FuncCall $funcCall): ?string
{
$argumentValue = $funcCall->args[0]->value;
if ($argumentValue instanceof MethodCall || $argumentValue instanceof StaticCall) {
return $this->getName($argumentValue->name);
}
return $this->getName($argumentValue);
}
} }

View File

@ -11,6 +11,7 @@ use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\Expression; use PhpParser\Node\Stmt\Expression;
use PhpParser\Node\Stmt\Return_; use PhpParser\Node\Stmt\Return_;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\MagicDisclosure\NodeAnalyzer\FluentChainMethodCallNodeAnalyzer; use Rector\MagicDisclosure\NodeAnalyzer\FluentChainMethodCallNodeAnalyzer;
use Rector\MagicDisclosure\NodeManipulator\FluentChainMethodCallRootExtractor; use Rector\MagicDisclosure\NodeManipulator\FluentChainMethodCallRootExtractor;
use Rector\MagicDisclosure\ValueObject\AssignAndRootExpr; use Rector\MagicDisclosure\ValueObject\AssignAndRootExpr;
@ -42,6 +43,10 @@ final class NonFluentChainMethodCallFactory
public function createFromNewAndRootMethodCall(New_ $new, MethodCall $rootMethodCall): array public function createFromNewAndRootMethodCall(New_ $new, MethodCall $rootMethodCall): array
{ {
$variableName = $this->variableNaming->resolveFromNode($new); $variableName = $this->variableNaming->resolveFromNode($new);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
$newVariable = new Variable($variableName); $newVariable = new Variable($variableName);
$newStmts = []; $newStmts = [];

View File

@ -10,6 +10,7 @@ use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\PropertyFetch; use PhpParser\Node\Expr\PropertyFetch;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Stmt\Return_; use PhpParser\Node\Stmt\Return_;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\Core\PhpParser\Node\BetterNodeFinder; use Rector\Core\PhpParser\Node\BetterNodeFinder;
use Rector\MagicDisclosure\NodeAnalyzer\ExprStringTypeResolver; use Rector\MagicDisclosure\NodeAnalyzer\ExprStringTypeResolver;
use Rector\MagicDisclosure\ValueObject\AssignAndRootExpr; use Rector\MagicDisclosure\ValueObject\AssignAndRootExpr;
@ -79,9 +80,7 @@ final class FluentChainMethodCallRootExtractor
if ($methodCall->var instanceof New_) { if ($methodCall->var instanceof New_) {
// direct = no parent // direct = no parent
if ($kind === self::KIND_IN_ARGS) { if ($kind === self::KIND_IN_ARGS) {
$variableName = $this->variableNaming->resolveFromNode($methodCall->var); return $this->resolveKindInArgs($methodCall);
$silentVariable = new Variable($variableName);
return new AssignAndRootExpr($methodCall->var, $methodCall->var, $silentVariable);
} }
return $this->matchMethodCallOnNew($methodCall); return $this->matchMethodCallOnNew($methodCall);
@ -119,6 +118,17 @@ final class FluentChainMethodCallRootExtractor
return $variableStaticType !== $calledMethodStaticType; return $variableStaticType !== $calledMethodStaticType;
} }
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(MethodCall $methodCall): ?AssignAndRootExpr private function matchMethodCallOnNew(MethodCall $methodCall): ?AssignAndRootExpr
{ {
// we need assigned left variable here // we need assigned left variable here

View File

@ -9,6 +9,7 @@ use PhpParser\Node\Arg;
use PhpParser\Node\Expr\MethodCall; use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\New_; use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\Core\RectorDefinition\CodeSample; use Rector\Core\RectorDefinition\CodeSample;
use Rector\Core\RectorDefinition\RectorDefinition; use Rector\Core\RectorDefinition\RectorDefinition;
use Rector\MagicDisclosure\NodeAnalyzer\NewFluentChainMethodCallNodeAnalyzer; use Rector\MagicDisclosure\NodeAnalyzer\NewFluentChainMethodCallNodeAnalyzer;
@ -140,6 +141,10 @@ PHP
private function crateVariableFromNew(New_ $new): Variable private function crateVariableFromNew(New_ $new): Variable
{ {
$variableName = $this->variableNaming->resolveFromNode($new); $variableName = $this->variableNaming->resolveFromNode($new);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
return new Variable($variableName); return new Variable($variableName);
} }

View File

@ -9,6 +9,7 @@ use PhpParser\Node\Arg;
use PhpParser\Node\Expr\MethodCall; use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\New_; use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\Core\RectorDefinition\CodeSample; use Rector\Core\RectorDefinition\CodeSample;
use Rector\Core\RectorDefinition\RectorDefinition; use Rector\Core\RectorDefinition\RectorDefinition;
use Rector\MagicDisclosure\NodeAnalyzer\NewFluentChainMethodCallNodeAnalyzer; use Rector\MagicDisclosure\NodeAnalyzer\NewFluentChainMethodCallNodeAnalyzer;
@ -126,6 +127,10 @@ PHP
private function crateVariableFromNew(New_ $new): Variable private function crateVariableFromNew(New_ $new): Variable
{ {
$variableName = $this->variableNaming->resolveFromNode($new); $variableName = $this->variableNaming->resolveFromNode($new);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
return new Variable($variableName); return new Variable($variableName);
} }
} }

View File

@ -7,6 +7,7 @@ namespace Rector\NetteKdyby\BlueprintFactory;
use PhpParser\Node\Arg; use PhpParser\Node\Arg;
use PHPStan\Type\ObjectType; use PHPStan\Type\ObjectType;
use PHPStan\Type\StaticType; use PHPStan\Type\StaticType;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\NetteKdyby\Naming\VariableNaming; use Rector\NetteKdyby\Naming\VariableNaming;
use Rector\NetteKdyby\ValueObject\VariableWithType; use Rector\NetteKdyby\ValueObject\VariableWithType;
use Rector\NodeTypeResolver\NodeTypeResolver; use Rector\NodeTypeResolver\NodeTypeResolver;
@ -50,6 +51,9 @@ final class VariableWithTypesFactory
foreach ($args as $arg) { foreach ($args as $arg) {
$staticType = $this->nodeTypeResolver->getStaticType($arg->value); $staticType = $this->nodeTypeResolver->getStaticType($arg->value);
$variableName = $this->variableNaming->resolveFromNodeAndType($arg, $staticType); $variableName = $this->variableNaming->resolveFromNodeAndType($arg, $staticType);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
// compensate for static // compensate for static
if ($staticType instanceof StaticType) { if ($staticType instanceof StaticType) {

View File

@ -173,6 +173,10 @@ final class ContributeEventClassResolver
private function createGetterFromParamAndStaticType(Param $param, Type $type): string private function createGetterFromParamAndStaticType(Param $param, Type $type): string
{ {
$variableName = $this->variableNaming->resolveFromNodeAndType($param, $type); $variableName = $this->variableNaming->resolveFromNodeAndType($param, $type);
if ($variableName === null) {
throw new ShouldNotHappenException();
}
return 'get' . ucfirst($variableName); return 'get' . ucfirst($variableName);
} }
} }

View File

@ -8,13 +8,16 @@ use PhpParser\Node;
use PhpParser\Node\Arg; use PhpParser\Node\Arg;
use PhpParser\Node\Expr\ArrayDimFetch; use PhpParser\Node\Expr\ArrayDimFetch;
use PhpParser\Node\Expr\Cast; use PhpParser\Node\Expr\Cast;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\MethodCall; use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\New_; use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\PropertyFetch; use PhpParser\Node\Expr\PropertyFetch;
use PhpParser\Node\Expr\StaticCall;
use PhpParser\Node\Expr\Ternary; use PhpParser\Node\Expr\Ternary;
use PhpParser\Node\Name; use PhpParser\Node\Name;
use PhpParser\Node\Scalar; use PhpParser\Node\Scalar;
use PhpParser\Node\Scalar\String_; use PhpParser\Node\Scalar\String_;
use PHPStan\Analyser\Scope;
use PHPStan\Type\ThisType; use PHPStan\Type\ThisType;
use PHPStan\Type\Type; use PHPStan\Type\Type;
use Rector\CodingStyle\Naming\ClassNaming; use Rector\CodingStyle\Naming\ClassNaming;
@ -25,6 +28,9 @@ use Rector\Core\Util\StaticRectorStrings;
use Rector\NodeNameResolver\NodeNameResolver; use Rector\NodeNameResolver\NodeNameResolver;
use Rector\NodeTypeResolver\NodeTypeResolver; use Rector\NodeTypeResolver\NodeTypeResolver;
/**
* @todo decouple to collector?
*/
final class VariableNaming final class VariableNaming
{ {
/** /**
@ -59,16 +65,18 @@ final class VariableNaming
$this->nodeTypeResolver = $nodeTypeResolver; $this->nodeTypeResolver = $nodeTypeResolver;
} }
public function resolveFromNode(Node $node): string public function resolveFromNode(Node $node): ?string
{ {
$nodeType = $this->nodeTypeResolver->getStaticType($node); $nodeType = $this->nodeTypeResolver->getStaticType($node);
return $this->resolveFromNodeAndType($node, $nodeType); return $this->resolveFromNodeAndType($node, $nodeType);
} }
public function resolveFromNodeAndType(Node $node, Type $type): string public function resolveFromNodeAndType(Node $node, Type $type): ?string
{ {
$variableName = $this->resolveBareFromNode($node); $variableName = $this->resolveBareFromNode($node);
if ($variableName === null) {
return null;
}
// adjust static to specific class // adjust static to specific class
if ($variableName === 'this' && $type instanceof ThisType) { if ($variableName === 'this' && $type instanceof ThisType) {
@ -79,7 +87,52 @@ final class VariableNaming
return StaticRectorStrings::underscoreToPascalCase($variableName); return StaticRectorStrings::underscoreToPascalCase($variableName);
} }
private function resolveBareFromNode(Node $node): string public function resolveFromNodeWithScopeCountAndFallbackName(
Node\Expr $expr,
Scope $scope,
string $fallbackName
): string {
$name = $this->resolveFromNode($expr);
if ($name === null) {
$name = $fallbackName;
}
return lcfirst($this->createCountedValueName($name, $scope));
}
public function createCountedValueName(string $valueName, ?Scope $scope): string
{
if ($scope === null) {
return $valueName;
}
// make sure variable name is unique
if (! $scope->hasVariableType($valueName)->yes()) {
return $valueName;
}
// we need to add number suffix until the variable is unique
$i = 2;
$countedValueNamePart = $valueName;
while ($scope->hasVariableType($valueName)->yes()) {
$valueName = $countedValueNamePart . $i;
++$i;
}
return $valueName;
}
public function resolveFromFuncCallFirstArgumentWithSuffix(
FuncCall $funcCall,
string $suffix,
string $fallbackName,
?Scope $scope
): string {
$bareName = $this->resolveBareFuncCallArgumentName($funcCall, $fallbackName, $suffix);
return $this->createCountedValueName($bareName, $scope);
}
private function resolveBareFromNode(Node $node): ?string
{ {
$node = $this->unwrapNode($node); $node = $this->unwrapNode($node);
@ -99,6 +152,10 @@ final class VariableNaming
return $this->resolveFromNew($node); return $this->resolveFromNew($node);
} }
if ($node instanceof FuncCall) {
return $this->resolveFromNode($node->name);
}
if ($node === null) { if ($node === null) {
throw new NotImplementedException(); throw new NotImplementedException();
} }
@ -112,7 +169,23 @@ final class VariableNaming
return $node->value; return $node->value;
} }
throw new NotImplementedException(); return null;
}
private function resolveBareFuncCallArgumentName(FuncCall $funcCall, string $fallbackName, string $suffix): string
{
$argumentValue = $funcCall->args[0]->value;
if ($argumentValue instanceof MethodCall || $argumentValue instanceof StaticCall) {
$name = $this->nodeNameResolver->getName($argumentValue->name);
} else {
$name = $this->nodeNameResolver->getName($argumentValue);
}
if ($name === null) {
return $fallbackName;
}
return $name . $suffix;
} }
private function unwrapNode(Node $node): ?Node private function unwrapNode(Node $node): ?Node
@ -132,7 +205,7 @@ final class VariableNaming
return $node; return $node;
} }
private function resolveParamNameFromArrayDimFetch(ArrayDimFetch $arrayDimFetch): string private function resolveParamNameFromArrayDimFetch(ArrayDimFetch $arrayDimFetch): ?string
{ {
while ($arrayDimFetch instanceof ArrayDimFetch) { while ($arrayDimFetch instanceof ArrayDimFetch) {
if ($arrayDimFetch->dim instanceof Scalar) { if ($arrayDimFetch->dim instanceof Scalar) {
@ -165,19 +238,18 @@ final class VariableNaming
return $varName . ucfirst($propertyName); return $varName . ucfirst($propertyName);
} }
private function resolveFromMethodCall(MethodCall $methodCall): string private function resolveFromMethodCall(MethodCall $methodCall): ?string
{ {
$varName = $this->nodeNameResolver->getName($methodCall->var); if ($methodCall->name instanceof MethodCall) {
if (! is_string($varName)) { return $this->resolveFromMethodCall($methodCall->name);
throw new NotImplementedException();
} }
$methodName = $this->nodeNameResolver->getName($methodCall->name); $methodName = $this->nodeNameResolver->getName($methodCall->name);
if (! is_string($methodName)) { if (! is_string($methodName)) {
throw new NotImplementedException(); return null;
} }
return $varName . ucfirst($methodName); return $methodName;
} }
private function resolveFromNew(New_ $new): string private function resolveFromNew(New_ $new): string

View File

@ -4,7 +4,6 @@ declare(strict_types=1);
namespace Rector\Php70\Rector\FuncCall; namespace Rector\Php70\Rector\FuncCall;
use function lcfirst;
use PhpParser\Node; use PhpParser\Node;
use PhpParser\Node\Expr; use PhpParser\Node\Expr;
use PhpParser\Node\Expr\ArrayDimFetch; use PhpParser\Node\Expr\ArrayDimFetch;
@ -13,20 +12,20 @@ use PhpParser\Node\Expr\AssignOp;
use PhpParser\Node\Expr\AssignRef; use PhpParser\Node\Expr\AssignRef;
use PhpParser\Node\Expr\FuncCall; use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\MethodCall; use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Expr\New_;
use PhpParser\Node\Expr\PropertyFetch; use PhpParser\Node\Expr\PropertyFetch;
use PhpParser\Node\Expr\StaticCall; use PhpParser\Node\Expr\StaticCall;
use PhpParser\Node\Expr\StaticPropertyFetch; use PhpParser\Node\Expr\StaticPropertyFetch;
use PhpParser\Node\Expr\Variable; use PhpParser\Node\Expr\Variable;
use PhpParser\Node\Name;
use PhpParser\Node\Stmt\Return_; use PhpParser\Node\Stmt\Return_;
use PHPStan\Analyser\MutatingScope; use PHPStan\Analyser\MutatingScope;
use PHPStan\Analyser\Scope; use PHPStan\Analyser\Scope;
use PHPStan\Reflection\ParameterReflection; use PHPStan\Reflection\ParameterReflection;
use PHPStan\Type\MixedType;
use Rector\Core\PHPStan\Reflection\CallReflectionResolver; use Rector\Core\PHPStan\Reflection\CallReflectionResolver;
use Rector\Core\Rector\AbstractRector; use Rector\Core\Rector\AbstractRector;
use Rector\Core\RectorDefinition\CodeSample; use Rector\Core\RectorDefinition\CodeSample;
use Rector\Core\RectorDefinition\RectorDefinition; use Rector\Core\RectorDefinition\RectorDefinition;
use Rector\NetteKdyby\Naming\VariableNaming;
use Rector\NodeTypeResolver\Node\AttributeKey; use Rector\NodeTypeResolver\Node\AttributeKey;
use Rector\Php70\ValueObject\VariableAssignPair; use Rector\Php70\ValueObject\VariableAssignPair;
@ -37,19 +36,20 @@ use Rector\Php70\ValueObject\VariableAssignPair;
*/ */
final class NonVariableToVariableOnFunctionCallRector extends AbstractRector final class NonVariableToVariableOnFunctionCallRector extends AbstractRector
{ {
/**
* @var string
*/
private const DEFAULT_VARIABLE_NAME = 'tmp';
/** /**
* @var CallReflectionResolver * @var CallReflectionResolver
*/ */
private $callReflectionResolver; private $callReflectionResolver;
public function __construct(CallReflectionResolver $callReflectionResolver) /**
* @var VariableNaming
*/
private $variableNaming;
public function __construct(CallReflectionResolver $callReflectionResolver, VariableNaming $variableNaming)
{ {
$this->callReflectionResolver = $callReflectionResolver; $this->callReflectionResolver = $callReflectionResolver;
$this->variableNaming = $variableNaming;
} }
public function getDefinition(): RectorDefinition public function getDefinition(): RectorDefinition
@ -73,27 +73,32 @@ final class NonVariableToVariableOnFunctionCallRector extends AbstractRector
*/ */
public function refactor(Node $node): ?Node public function refactor(Node $node): ?Node
{ {
$scope = $node->getAttribute(AttributeKey::SCOPE);
if (! $scope instanceof MutatingScope) {
return null;
}
$arguments = $this->getNonVariableArguments($node); $arguments = $this->getNonVariableArguments($node);
if ($arguments === []) { if ($arguments === []) {
return null; return null;
} }
$scopeNode = $node->getAttribute(AttributeKey::METHOD_NODE) ?? $node->getAttribute(
AttributeKey::FUNCTION_NODE
) ?? $node->getAttribute(AttributeKey::CLOSURE_NODE);
/** @var Scope $currentScope */
$currentScope = $scopeNode->getAttribute(AttributeKey::SCOPE);
foreach ($arguments as $key => $argument) { foreach ($arguments as $key => $argument) {
$replacements = $this->getReplacementsFor($argument, $scope); $replacements = $this->getReplacementsFor($argument, $currentScope, $scopeNode);
$current = $node->getAttribute(AttributeKey::CURRENT_STATEMENT); $current = $node->getAttribute(AttributeKey::CURRENT_STATEMENT);
$this->addNodeBeforeNode($replacements->assign(), $current instanceof Return_ ? $current : $node); $this->addNodeBeforeNode($replacements->getAssign(), $current instanceof Return_ ? $current : $node);
$node->args[$key]->value = $replacements->variable(); $node->args[$key]->value = $replacements->getVariable();
$scope = $scope->assignExpression($replacements->variable(), $scope->getType($replacements->variable())); // add variable name to scope, so we prevent duplication of new variable of the same name
$currentScope = $currentScope->assignExpression(
$replacements->getVariable(),
$currentScope->getType($replacements->getVariable())
);
} }
$node->setAttribute(AttributeKey::SCOPE, $scope); $scopeNode->setAttribute(AttributeKey::SCOPE, $currentScope);
return $node; return $node;
} }
@ -139,20 +144,24 @@ final class NonVariableToVariableOnFunctionCallRector extends AbstractRector
return $arguments; return $arguments;
} }
private function getReplacementsFor(Expr $expr, Scope $scope): VariableAssignPair private function getReplacementsFor(Expr $expr, MutatingScope $mutatingScope, Node $scopeNode): VariableAssignPair
{ {
if ( /** @var Assign|AssignOp|AssignRef $expr */
( if ($this->isAssign($expr) && $this->isVariableLikeNode($expr->var)) {
$expr instanceof Assign
|| $expr instanceof AssignRef
|| $expr instanceof AssignOp
)
&& $this->isVariableLikeNode($expr->var)
) {
return new VariableAssignPair($expr->var, $expr); return new VariableAssignPair($expr->var, $expr);
} }
$variable = new Variable($this->getVariableNameFor($expr, $scope)); $variableName = $this->variableNaming->resolveFromNodeWithScopeCountAndFallbackName(
$expr,
$mutatingScope,
'tmp'
);
$variable = new Variable($variableName);
// add a new scope with this variable
$newVariableAwareScope = $mutatingScope->assignExpression($variable, new MixedType());
$scopeNode->setAttribute(AttributeKey::SCOPE, $newVariableAwareScope);
return new VariableAssignPair($variable, new Assign($variable, $expr)); return new VariableAssignPair($variable, new Assign($variable, $expr));
} }
@ -165,16 +174,16 @@ final class NonVariableToVariableOnFunctionCallRector extends AbstractRector
|| $node instanceof StaticPropertyFetch; || $node instanceof StaticPropertyFetch;
} }
private function getVariableNameFor(Expr $expr, Scope $scope): string private function isAssign(Expr $expr): bool
{ {
if ($expr instanceof New_ && $expr->class instanceof Name) { if ($expr instanceof Assign) {
$name = $this->getShortName($expr->class); return true;
} elseif ($expr instanceof MethodCall || $expr instanceof StaticCall) {
$name = $this->getName($expr->name);
} else {
$name = $this->getName($expr);
} }
return lcfirst($this->createCountedValueName($name ?? self::DEFAULT_VARIABLE_NAME, $scope)); if ($expr instanceof AssignRef) {
return true;
}
return $expr instanceof AssignOp;
} }
} }

View File

@ -38,7 +38,7 @@ final class VariableAssignPair
/** /**
* @return Variable|ArrayDimFetch|PropertyFetch|StaticPropertyFetch * @return Variable|ArrayDimFetch|PropertyFetch|StaticPropertyFetch
*/ */
public function variable(): Node public function getVariable(): Node
{ {
return $this->variable; return $this->variable;
} }
@ -46,7 +46,7 @@ final class VariableAssignPair
/** /**
* @return Assign|AssignOp|AssignRef * @return Assign|AssignOp|AssignRef
*/ */
public function assign(): Node public function getAssign(): Node
{ {
return $this->assign; return $this->assign;
} }

View File

@ -22,8 +22,8 @@ function anonymousFunction()
$bar = bar(); $bar = bar();
$anonymousFunction($bar); $anonymousFunction($bar);
$staticAnonymousFunction = static function (&$bar) {}; $staticAnonymousFunction = static function (&$bar) {};
$bar = bar(); $bar2 = bar();
$staticAnonymousFunction($bar); $staticAnonymousFunction($bar2);
} }
?> ?>

View File

@ -37,15 +37,15 @@ function arrayCallable()
$bar = bar(); $bar = bar();
[ArrayCallable::class, 'someStaticMethod']($bar); [ArrayCallable::class, 'someStaticMethod']($bar);
$callable = [ArrayCallable::class, 'someStaticMethod']; $callable = [ArrayCallable::class, 'someStaticMethod'];
$bar = bar(); $bar2 = bar();
$callable($bar); $callable($bar2);
$arrayCallable = new ArrayCallable(); $arrayCallable = new ArrayCallable();
$bar = bar(); $bar3 = bar();
[$arrayCallable, 'someMethod']($bar); [$arrayCallable, 'someMethod']($bar3);
$callable = [$arrayCallable, 'someMethod']; $callable = [$arrayCallable, 'someMethod'];
$bar = bar(); $bar4 = bar();
$callable($bar); $callable($bar4);
} }
?> ?>

View File

@ -18,8 +18,8 @@ function binaryOp()
{ {
$tmp = 1 + 1; $tmp = 1 + 1;
reset($tmp); reset($tmp);
$tmp = 2 + $var; $tmp2 = 2 + $var;
reset($tmp); reset($tmp2);
} }
?> ?>

View File

@ -32,15 +32,15 @@ function funcCalls()
reset($bar); reset($bar);
$baz = baz(); $baz = baz();
byRef(bar(), $baz); byRef(bar(), $baz);
$bar = bar(); $bar2 = bar();
$baz = baz(); $baz2 = baz();
allByRef($bar, $baz); allByRef($bar2, $baz2);
$tmp = 1; $tmp = 1;
$tmp2 = 2; $tmp2 = 2;
allByRef($tmp, $tmp2); allByRef($tmp, $tmp2);
$bar = bar(); $bar3 = bar();
return byRef(1, $bar); return byRef(1, $bar3);
} }
?> ?>

View File

@ -57,8 +57,8 @@ function methodCalls()
$aClass = new AClass(); $aClass = new AClass();
$baz = baz(); $baz = baz();
$aClass->baz($baz); $aClass->baz($baz);
$bar = bar(); $bar2 = bar();
$aClass->child()->bar($bar); $aClass->child()->bar($bar2);
} }
?> ?>

View File

@ -35,12 +35,12 @@ function stringyCalls()
'reset'($bar); 'reset'($bar);
$functionName = 'reset'; $functionName = 'reset';
$bar = bar(); $bar2 = bar();
$functionName($bar); $functionName($bar2);
$methodName = MyClass::class.'::staticMethod'; $methodName = MyClass::class.'::staticMethod';
$bar = bar(); $bar3 = bar();
$methodName($bar); $methodName($bar3);
} }
?> ?>

View File

@ -15,7 +15,6 @@ use PhpParser\Node\Stmt;
use PhpParser\Node\Stmt\Expression; use PhpParser\Node\Stmt\Expression;
use PhpParser\Node\Stmt\Return_; use PhpParser\Node\Stmt\Return_;
use PhpParser\NodeVisitorAbstract; use PhpParser\NodeVisitorAbstract;
use PHPStan\Analyser\Scope;
use Rector\AnonymousClass\NodeAnalyzer\ClassNodeAnalyzer; use Rector\AnonymousClass\NodeAnalyzer\ClassNodeAnalyzer;
use Rector\Core\Configuration\Option; use Rector\Core\Configuration\Option;
use Rector\Core\Contract\Rector\PhpRectorInterface; use Rector\Core\Contract\Rector\PhpRectorInterface;
@ -251,28 +250,6 @@ abstract class AbstractRector extends NodeVisitorAbstract implements PhpRectorIn
return $this->classNodeAnalyzer->isAnonymousClass($node); return $this->classNodeAnalyzer->isAnonymousClass($node);
} }
protected function createCountedValueName(string $countedValueName, ?Scope $scope): string
{
if ($scope === null) {
return $countedValueName;
}
// make sure variable name is unique
if (! $scope->hasVariableType($countedValueName)->yes()) {
return $countedValueName;
}
// we need to add number suffix until the variable is unique
$i = 2;
$countedValueNamePart = $countedValueName;
while ($scope->hasVariableType($countedValueName)->yes()) {
$countedValueName = $countedValueNamePart . $i;
++$i;
}
return $countedValueName;
}
protected function mirrorComments(Node $newNode, Node $oldNode): void protected function mirrorComments(Node $newNode, Node $oldNode): void
{ {
$newNode->setAttribute(AttributeKey::PHP_DOC_INFO, $oldNode->getAttribute(AttributeKey::PHP_DOC_INFO)); $newNode->setAttribute(AttributeKey::PHP_DOC_INFO, $oldNode->getAttribute(AttributeKey::PHP_DOC_INFO));