rector/rules/CodeQuality/Rector/FuncCall/InlineIsAInstanceOfRector.php
2022-05-27 00:01:06 +02:00

114 lines
2.7 KiB
PHP

<?php
declare(strict_types=1);
namespace Rector\CodeQuality\Rector\FuncCall;
use PhpParser\Node;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\ClassConstFetch;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Expr\Instanceof_;
use PhpParser\Node\Name\FullyQualified;
use PHPStan\Type\Generic\GenericClassStringType;
use PHPStan\Type\ObjectType;
use PHPStan\Type\ObjectWithoutClassType;
use PHPStan\Type\TypeWithClassName;
use Rector\Core\Rector\AbstractRector;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @see \Rector\Tests\CodeQuality\Rector\FuncCall\InlineIsAInstanceOfRector\InlineIsAInstanceOfRectorTest
*/
final class InlineIsAInstanceOfRector extends AbstractRector
{
public function getRuleDefinition(): RuleDefinition
{
return new RuleDefinition('Change is_a() with object and class name check to instanceof', [
new CodeSample(
<<<'CODE_SAMPLE'
class SomeClass
{
public function run(object $object)
{
return is_a($object, SomeType::class);
}
}
CODE_SAMPLE
,
<<<'CODE_SAMPLE'
class SomeClass
{
public function run(object $object)
{
return $object instanceof SomeType;
}
}
CODE_SAMPLE
),
]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes(): array
{
return [FuncCall::class];
}
/**
* @param FuncCall $node
*/
public function refactor(Node $node): ?Node
{
if (! $this->isName($node->name, 'is_a')) {
return null;
}
$args = $node->getArgs();
$firstArgValue = $args[0]->value;
if (! $this->isFirstObjectType($firstArgValue)) {
return null;
}
$className = $this->resolveClassName($args[1]->value);
if ($className === null) {
return null;
}
return new Instanceof_($firstArgValue, new FullyQualified($className));
}
private function resolveClassName(Expr $expr): ?string
{
if (! $expr instanceof ClassConstFetch) {
return null;
}
$type = $this->getType($expr);
if ($type instanceof GenericClassStringType) {
$type = $type->getGenericType();
}
if (! $type instanceof TypeWithClassName) {
return null;
}
return $type->getClassName();
}
private function isFirstObjectType(Expr $expr): bool
{
$exprType = $this->getType($expr);
if ($exprType instanceof ObjectWithoutClassType) {
return true;
}
return $exprType instanceof ObjectType;
}
}