mirror of
https://github.com/rectorphp/rector.git
synced 2025-02-18 23:05:25 +01:00
154 lines
4.7 KiB
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;
|
|
}
|
|
}
|