Find this useful? Enter your email to receive occasional updates for securing PHP code.
Signing you up...
Thank you for signing up!
PHP Decode
<?php declare (strict_types=1); namespace Rector\Symfony\Configs\Rector\Closure; use Php..
Decoded Output download
<?php
declare (strict_types=1);
namespace Rector\Symfony\Configs\Rector\Closure;
use PhpParser\Node;
use PhpParser\Node\Arg;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Array_;
use PhpParser\Node\Expr\ArrayItem;
use PhpParser\Node\Expr\ClassConstFetch;
use PhpParser\Node\Expr\Closure;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Scalar\String_;
use PHPStan\Reflection\ParametersAcceptorSelector;
use PHPStan\Reflection\Php\PhpParameterReflection;
use PHPStan\Reflection\ReflectionProvider;
use PHPStan\Type\ObjectType;
use Rector\PhpParser\Node\Value\ValueResolver;
use Rector\Rector\AbstractRector;
use Rector\Symfony\NodeAnalyzer\SymfonyPhpClosureDetector;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @see \Rector\Symfony\Tests\Configs\Rector\Closure\ServiceArgsToServiceNamedArgRector\ServiceArgsToServiceNamedArgRectorTest
*/
final class ServiceArgsToServiceNamedArgRector extends AbstractRector
{
/**
* @readonly
* @var \Rector\Symfony\NodeAnalyzer\SymfonyPhpClosureDetector
*/
private $symfonyPhpClosureDetector;
/**
* @readonly
* @var \PHPStan\Reflection\ReflectionProvider
*/
private $reflectionProvider;
/**
* @readonly
* @var \Rector\PhpParser\Node\Value\ValueResolver
*/
private $valueResolver;
public function __construct(SymfonyPhpClosureDetector $symfonyPhpClosureDetector, ReflectionProvider $reflectionProvider, ValueResolver $valueResolver)
{
$this->symfonyPhpClosureDetector = $symfonyPhpClosureDetector;
$this->reflectionProvider = $reflectionProvider;
$this->valueResolver = $valueResolver;
}
public function getRuleDefinition() : RuleDefinition
{
return new RuleDefinition('Converts order-dependent arguments args() to named arg() call', [new CodeSample(<<<'CODE_SAMPLE'
use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
return static function (ContainerConfigurator $containerConfigurator): void {
$services = $containerConfigurator->services();
$services->set(SomeClass::class)
->args(['some_value']);
};
CODE_SAMPLE
, <<<'CODE_SAMPLE'
use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
return static function (ContainerConfigurator $containerConfigurator): void {
$services = $containerConfigurator->services();
$services->set(SomeClass::class)
->arg('$someCtorParameter', 'some_value');
};
CODE_SAMPLE
)]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes() : array
{
return [Closure::class];
}
/**
* @param Closure $node
*/
public function refactor(Node $node) : ?Node
{
if (!$this->symfonyPhpClosureDetector->detect($node)) {
return null;
}
$hasChanged = \false;
$this->traverseNodesWithCallable($node->stmts, function (Node $node) use(&$hasChanged) : ?MethodCall {
if (!$node instanceof MethodCall) {
return null;
}
if (!$this->isServiceArgsMethodCall($node)) {
return null;
}
$serviceClass = $this->resolveServiceClass($node);
// impossible to resolve
if ($serviceClass === null) {
return null;
}
if (!$this->reflectionProvider->hasClass($serviceClass)) {
return null;
}
$constructorParameterNames = $this->resolveConstructorParameterNames($serviceClass);
$mainArgMethodCall = $this->createMainArgMethodCall($node, $constructorParameterNames);
if (!$mainArgMethodCall instanceof MethodCall) {
return null;
}
$hasChanged = \true;
return $mainArgMethodCall;
});
if ($hasChanged) {
return $node;
}
return null;
}
private function resolveServiceClass(MethodCall $methodCall) : ?string
{
while ($methodCall->var instanceof MethodCall) {
$methodCall = $methodCall->var;
}
if (!$this->isName($methodCall->name, 'set')) {
return null;
}
$servicesSetArgs = $methodCall->getArgs();
foreach ($servicesSetArgs as $serviceSetArg) {
if (!$serviceSetArg->value instanceof ClassConstFetch) {
continue;
}
return $this->valueResolver->getValue($serviceSetArg->value);
}
return null;
}
/**
* @return string[]
*/
private function resolveConstructorParameterNames(string $serviceClass) : array
{
$serviceClassReflection = $this->reflectionProvider->getClass($serviceClass);
if (!$serviceClassReflection->hasConstructor()) {
return [];
}
$constructorParameterNames = [];
$extendedMethodReflection = $serviceClassReflection->getConstructor();
$parametersAcceptorWithPhpDocs = ParametersAcceptorSelector::combineAcceptors($extendedMethodReflection->getVariants());
foreach ($parametersAcceptorWithPhpDocs->getParameters() as $parameterReflectionWithPhpDoc) {
/** @var PhpParameterReflection $parameterReflectionWithPhpDoc */
$constructorParameterNames[] = '$' . $parameterReflectionWithPhpDoc->getName();
}
return $constructorParameterNames;
}
private function isServiceArgsMethodCall(MethodCall $methodCall) : bool
{
if (!$this->isName($methodCall->name, 'args')) {
return \false;
}
return $this->isObjectType($methodCall->var, new ObjectType('Symfony\\Component\\DependencyInjection\\Loader\\Configurator\\ServiceConfigurator'));
}
private function createArgMethodCall(string $parameterName, Expr $expr, ?MethodCall $argMethodCall, MethodCall $methodCall) : MethodCall
{
$argArgs = [new Arg(new String_($parameterName)), new Arg($expr)];
$callerExpr = $argMethodCall instanceof MethodCall ? $argMethodCall : $methodCall->var;
return new MethodCall($callerExpr, 'arg', $argArgs);
}
/**
* @param string[] $constructorParameterNames
*/
private function createMainArgMethodCall(MethodCall $methodCall, array $constructorParameterNames) : ?MethodCall
{
if ($constructorParameterNames === []) {
return null;
}
$argMethodCall = null;
foreach ($methodCall->getArgs() as $arg) {
if (!$arg->value instanceof Array_) {
return null;
}
$array = $arg->value;
foreach ($array->items as $key => $arrayItem) {
if (!$arrayItem instanceof ArrayItem) {
continue;
}
$arrayItemValue = $arrayItem->value;
$parameterPosition = $this->resolveParameterPosition($arrayItem, $key);
$argMethodCall = $this->createArgMethodCall($constructorParameterNames[$parameterPosition], $arrayItemValue, $argMethodCall, $methodCall);
}
}
return $argMethodCall;
}
private function resolveParameterPosition(ArrayItem $arrayItem, int $key) : int
{
if ($arrayItem->key instanceof Expr) {
return $this->valueResolver->getValue($arrayItem->key);
}
// fallbakc in case of empty array item
return $key;
}
}
?>
Did this file decode correctly?
Original Code
<?php
declare (strict_types=1);
namespace Rector\Symfony\Configs\Rector\Closure;
use PhpParser\Node;
use PhpParser\Node\Arg;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Array_;
use PhpParser\Node\Expr\ArrayItem;
use PhpParser\Node\Expr\ClassConstFetch;
use PhpParser\Node\Expr\Closure;
use PhpParser\Node\Expr\MethodCall;
use PhpParser\Node\Scalar\String_;
use PHPStan\Reflection\ParametersAcceptorSelector;
use PHPStan\Reflection\Php\PhpParameterReflection;
use PHPStan\Reflection\ReflectionProvider;
use PHPStan\Type\ObjectType;
use Rector\PhpParser\Node\Value\ValueResolver;
use Rector\Rector\AbstractRector;
use Rector\Symfony\NodeAnalyzer\SymfonyPhpClosureDetector;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @see \Rector\Symfony\Tests\Configs\Rector\Closure\ServiceArgsToServiceNamedArgRector\ServiceArgsToServiceNamedArgRectorTest
*/
final class ServiceArgsToServiceNamedArgRector extends AbstractRector
{
/**
* @readonly
* @var \Rector\Symfony\NodeAnalyzer\SymfonyPhpClosureDetector
*/
private $symfonyPhpClosureDetector;
/**
* @readonly
* @var \PHPStan\Reflection\ReflectionProvider
*/
private $reflectionProvider;
/**
* @readonly
* @var \Rector\PhpParser\Node\Value\ValueResolver
*/
private $valueResolver;
public function __construct(SymfonyPhpClosureDetector $symfonyPhpClosureDetector, ReflectionProvider $reflectionProvider, ValueResolver $valueResolver)
{
$this->symfonyPhpClosureDetector = $symfonyPhpClosureDetector;
$this->reflectionProvider = $reflectionProvider;
$this->valueResolver = $valueResolver;
}
public function getRuleDefinition() : RuleDefinition
{
return new RuleDefinition('Converts order-dependent arguments args() to named arg() call', [new CodeSample(<<<'CODE_SAMPLE'
use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
return static function (ContainerConfigurator $containerConfigurator): void {
$services = $containerConfigurator->services();
$services->set(SomeClass::class)
->args(['some_value']);
};
CODE_SAMPLE
, <<<'CODE_SAMPLE'
use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator;
return static function (ContainerConfigurator $containerConfigurator): void {
$services = $containerConfigurator->services();
$services->set(SomeClass::class)
->arg('$someCtorParameter', 'some_value');
};
CODE_SAMPLE
)]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes() : array
{
return [Closure::class];
}
/**
* @param Closure $node
*/
public function refactor(Node $node) : ?Node
{
if (!$this->symfonyPhpClosureDetector->detect($node)) {
return null;
}
$hasChanged = \false;
$this->traverseNodesWithCallable($node->stmts, function (Node $node) use(&$hasChanged) : ?MethodCall {
if (!$node instanceof MethodCall) {
return null;
}
if (!$this->isServiceArgsMethodCall($node)) {
return null;
}
$serviceClass = $this->resolveServiceClass($node);
// impossible to resolve
if ($serviceClass === null) {
return null;
}
if (!$this->reflectionProvider->hasClass($serviceClass)) {
return null;
}
$constructorParameterNames = $this->resolveConstructorParameterNames($serviceClass);
$mainArgMethodCall = $this->createMainArgMethodCall($node, $constructorParameterNames);
if (!$mainArgMethodCall instanceof MethodCall) {
return null;
}
$hasChanged = \true;
return $mainArgMethodCall;
});
if ($hasChanged) {
return $node;
}
return null;
}
private function resolveServiceClass(MethodCall $methodCall) : ?string
{
while ($methodCall->var instanceof MethodCall) {
$methodCall = $methodCall->var;
}
if (!$this->isName($methodCall->name, 'set')) {
return null;
}
$servicesSetArgs = $methodCall->getArgs();
foreach ($servicesSetArgs as $serviceSetArg) {
if (!$serviceSetArg->value instanceof ClassConstFetch) {
continue;
}
return $this->valueResolver->getValue($serviceSetArg->value);
}
return null;
}
/**
* @return string[]
*/
private function resolveConstructorParameterNames(string $serviceClass) : array
{
$serviceClassReflection = $this->reflectionProvider->getClass($serviceClass);
if (!$serviceClassReflection->hasConstructor()) {
return [];
}
$constructorParameterNames = [];
$extendedMethodReflection = $serviceClassReflection->getConstructor();
$parametersAcceptorWithPhpDocs = ParametersAcceptorSelector::combineAcceptors($extendedMethodReflection->getVariants());
foreach ($parametersAcceptorWithPhpDocs->getParameters() as $parameterReflectionWithPhpDoc) {
/** @var PhpParameterReflection $parameterReflectionWithPhpDoc */
$constructorParameterNames[] = '$' . $parameterReflectionWithPhpDoc->getName();
}
return $constructorParameterNames;
}
private function isServiceArgsMethodCall(MethodCall $methodCall) : bool
{
if (!$this->isName($methodCall->name, 'args')) {
return \false;
}
return $this->isObjectType($methodCall->var, new ObjectType('Symfony\\Component\\DependencyInjection\\Loader\\Configurator\\ServiceConfigurator'));
}
private function createArgMethodCall(string $parameterName, Expr $expr, ?MethodCall $argMethodCall, MethodCall $methodCall) : MethodCall
{
$argArgs = [new Arg(new String_($parameterName)), new Arg($expr)];
$callerExpr = $argMethodCall instanceof MethodCall ? $argMethodCall : $methodCall->var;
return new MethodCall($callerExpr, 'arg', $argArgs);
}
/**
* @param string[] $constructorParameterNames
*/
private function createMainArgMethodCall(MethodCall $methodCall, array $constructorParameterNames) : ?MethodCall
{
if ($constructorParameterNames === []) {
return null;
}
$argMethodCall = null;
foreach ($methodCall->getArgs() as $arg) {
if (!$arg->value instanceof Array_) {
return null;
}
$array = $arg->value;
foreach ($array->items as $key => $arrayItem) {
if (!$arrayItem instanceof ArrayItem) {
continue;
}
$arrayItemValue = $arrayItem->value;
$parameterPosition = $this->resolveParameterPosition($arrayItem, $key);
$argMethodCall = $this->createArgMethodCall($constructorParameterNames[$parameterPosition], $arrayItemValue, $argMethodCall, $methodCall);
}
}
return $argMethodCall;
}
private function resolveParameterPosition(ArrayItem $arrayItem, int $key) : int
{
if ($arrayItem->key instanceof Expr) {
return $this->valueResolver->getValue($arrayItem->key);
}
// fallbakc in case of empty array item
return $key;
}
}
Function Calls
None |
Stats
MD5 | fc06f9af8c8109ba571fe75350da7e2c |
Eval Count | 0 |
Decode Time | 82 ms |