rector/rules/DeadCode/ConditionResolver.php

154 lines
4.7 KiB
PHP

<?php
declare(strict_types=1);
namespace Rector\DeadCode;
use PhpParser\Node;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\BinaryOp\Equal;
use PhpParser\Node\Expr\BinaryOp\Identical;
use PhpParser\Node\Expr\BinaryOp\NotEqual;
use PhpParser\Node\Expr\BinaryOp\NotIdentical;
use PhpParser\Node\Expr\FuncCall;
use Rector\Core\Php\PhpVersionProvider;
use Rector\Core\PhpParser\Node\Value\ValueResolver;
use Rector\Core\Util\PhpVersionFactory;
use Rector\DeadCode\Contract\ConditionInterface;
use Rector\DeadCode\ValueObject\BinaryToVersionCompareCondition;
use Rector\DeadCode\ValueObject\VersionCompareCondition;
use Rector\NodeNameResolver\NodeNameResolver;
final class ConditionResolver
{
/**
* @var NodeNameResolver
*/
private $nodeNameResolver;
/**
* @var ValueResolver
*/
private $valueResolver;
/**
* @var PhpVersionProvider
*/
private $phpVersionProvider;
/**
* @var PhpVersionFactory
*/
private $phpVersionFactory;
public function __construct(
NodeNameResolver $nodeNameResolver,
PhpVersionProvider $phpVersionProvider,
ValueResolver $valueResolver,
PhpVersionFactory $phpVersionFactory
) {
$this->nodeNameResolver = $nodeNameResolver;
$this->valueResolver = $valueResolver;
$this->phpVersionProvider = $phpVersionProvider;
$this->phpVersionFactory = $phpVersionFactory;
}
public function resolveFromExpr(Expr $expr): ?ConditionInterface
{
if ($this->isVersionCompareFuncCall($expr)) {
/** @var FuncCall $expr */
return $this->resolveVersionCompareConditionForFuncCall($expr);
}
if (! $expr instanceof Identical && ! $expr instanceof Equal && ! $expr instanceof NotIdentical && ! $expr instanceof NotEqual) {
return null;
}
$binaryClass = get_class($expr);
if ($this->isVersionCompareFuncCall($expr->left)) {
/** @var FuncCall $funcCall */
$funcCall = $expr->left;
return $this->resolveFuncCall($funcCall, $expr->right, $binaryClass);
}
if ($this->isVersionCompareFuncCall($expr->right)) {
/** @var FuncCall $funcCall */
$funcCall = $expr->right;
$versionCompareCondition = $this->resolveVersionCompareConditionForFuncCall($funcCall);
if (! $versionCompareCondition instanceof VersionCompareCondition) {
return null;
}
$expectedValue = $this->valueResolver->getValue($expr->left);
return new BinaryToVersionCompareCondition($versionCompareCondition, $binaryClass, $expectedValue);
}
return null;
}
private function isVersionCompareFuncCall(Node $node): bool
{
if (! $node instanceof FuncCall) {
return false;
}
return $this->nodeNameResolver->isName($node, 'version_compare');
}
private function resolveVersionCompareConditionForFuncCall(FuncCall $funcCall): ?VersionCompareCondition
{
$firstVersion = $this->resolveArgumentValue($funcCall, 0);
if ($firstVersion === null) {
return null;
}
$secondVersion = $this->resolveArgumentValue($funcCall, 1);
if ($secondVersion === null) {
return null;
}
// includes compare sign as 3rd argument
$versionCompareSign = null;
if (isset($funcCall->args[2])) {
$versionCompareSign = $this->valueResolver->getValue($funcCall->args[2]->value);
}
return new VersionCompareCondition($firstVersion, $secondVersion, $versionCompareSign);
}
private function resolveFuncCall(
FuncCall $funcCall,
Expr $expr,
string $binaryClass
): ?BinaryToVersionCompareCondition {
$versionCompareCondition = $this->resolveVersionCompareConditionForFuncCall($funcCall);
if (! $versionCompareCondition instanceof VersionCompareCondition) {
return null;
}
$expectedValue = $this->valueResolver->getValue($expr);
return new BinaryToVersionCompareCondition($versionCompareCondition, $binaryClass, $expectedValue);
}
private function resolveArgumentValue(FuncCall $funcCall, int $argumentPosition): ?int
{
$firstArgValue = $funcCall->args[$argumentPosition]->value;
/** @var mixed|null $version */
$version = $this->valueResolver->getValue($firstArgValue);
if (in_array($version, ['PHP_VERSION', 'PHP_VERSION_ID'], true)) {
return $this->phpVersionProvider->provide();
}
if (is_string($version)) {
return $this->phpVersionFactory->createIntVersion($version);
}
return $version;
}
}