rector/rules/renaming/src/Rector/ClassConstFetch/RenameClassConstantRector.php
Tomas Votruba a7705ed0a5
[Symplify 9] First update + switch to RuleDocGenerator (#4616)
* [composer] bump to Symplify 9

* [Symplify 9] Update phpstan rules

* bump to Symplify 9 BETA2

* update AbstractKernel from Tests to Testing namespace

* decoupling removing node trait

* remove fluent calls

* removing variadic

* [CodingStyle] Improve AnnotateThrowablesRector

* bump deps

* Make use of RuleDocGenerator

* first short

* [DocumentationGenerator] Drop deprecated package, RuleSetGenerator now handles it

* import namespace

* update docs
2020-11-16 17:50:38 +00:00

114 lines
3.3 KiB
PHP

<?php
declare(strict_types=1);
namespace Rector\Renaming\Rector\ClassConstFetch;
use Nette\Utils\Strings;
use PhpParser\Node;
use PhpParser\Node\Expr\ClassConstFetch;
use PhpParser\Node\Identifier;
use PhpParser\Node\Name\FullyQualified;
use Rector\Core\Contract\Rector\ConfigurableRectorInterface;
use Rector\Core\Rector\AbstractRector;
use Rector\Renaming\ValueObject\RenameClassConstant;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\ConfiguredCodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
use Webmozart\Assert\Assert;
/**
* @see \Rector\Renaming\Tests\Rector\ClassConstFetch\RenameClassConstantRector\RenameClassConstantRectorTest
*/
final class RenameClassConstantRector extends AbstractRector implements ConfigurableRectorInterface
{
/**
* @var string
*/
public const CLASS_CONSTANT_RENAME = 'constant_rename';
/**
* @var RenameClassConstant[]
*/
private $classConstantRenames = [];
public function getRuleDefinition(): RuleDefinition
{
$configuration = [
self::CLASS_CONSTANT_RENAME => [
new RenameClassConstant('SomeClass', 'OLD_CONSTANT', 'NEW_CONSTANT'),
new RenameClassConstant('SomeClass', 'OTHER_OLD_CONSTANT', 'DifferentClass::NEW_CONSTANT'),
],
];
return new RuleDefinition(
'Replaces defined class constants in their calls.',
[
new ConfiguredCodeSample(
<<<'CODE_SAMPLE'
$value = SomeClass::OLD_CONSTANT;
$value = SomeClass::OTHER_OLD_CONSTANT;
CODE_SAMPLE
,
<<<'CODE_SAMPLE'
$value = SomeClass::NEW_CONSTANT;
$value = DifferentClass::NEW_CONSTANT;
CODE_SAMPLE
,
$configuration
),
]);
}
/**
* @return string[]
*/
public function getNodeTypes(): array
{
return [ClassConstFetch::class];
}
/**
* @param ClassConstFetch $node
*/
public function refactor(Node $node): ?Node
{
foreach ($this->classConstantRenames as $classConstantRename) {
if (! $this->isObjectType($node, $classConstantRename->getOldClass())) {
continue;
}
if (! $this->isName($node->name, $classConstantRename->getOldConstant())) {
continue;
}
if (Strings::contains($classConstantRename->getNewConstant(), '::')) {
return $this->createClassConstantFetchNodeFromDoubleColonFormat($classConstantRename->getNewConstant());
}
$node->name = new Identifier($classConstantRename->getNewConstant());
return $node;
}
return $node;
}
/**
* @param mixed[] $configuration
*/
public function configure(array $configuration): void
{
$classConstantRenames = $configuration[self::CLASS_CONSTANT_RENAME] ?? [];
Assert::allIsInstanceOf($classConstantRenames, RenameClassConstant::class);
$this->classConstantRenames = $classConstantRenames;
}
private function createClassConstantFetchNodeFromDoubleColonFormat(string $constant): ClassConstFetch
{
[$constantClass, $constantName] = explode('::', $constant);
return new ClassConstFetch(new FullyQualified($constantClass), new Identifier($constantName));
}
}