mirror of
https://github.com/rectorphp/rector.git
synced 2025-01-17 21:38:22 +01:00
bdfd24339f
2e977bd3aa
deploy to original repository
129 lines
5.0 KiB
PHP
129 lines
5.0 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\DeadCode;
|
|
|
|
use PhpParser\Node\Arg;
|
|
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
|
|
{
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\NodeNameResolver\NodeNameResolver
|
|
*/
|
|
private $nodeNameResolver;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\Php\PhpVersionProvider
|
|
*/
|
|
private $phpVersionProvider;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\PhpParser\Node\Value\ValueResolver
|
|
*/
|
|
private $valueResolver;
|
|
/**
|
|
* @readonly
|
|
* @var \Rector\Core\Util\PhpVersionFactory
|
|
*/
|
|
private $phpVersionFactory;
|
|
public function __construct(NodeNameResolver $nodeNameResolver, PhpVersionProvider $phpVersionProvider, ValueResolver $valueResolver, PhpVersionFactory $phpVersionFactory)
|
|
{
|
|
$this->nodeNameResolver = $nodeNameResolver;
|
|
$this->phpVersionProvider = $phpVersionProvider;
|
|
$this->valueResolver = $valueResolver;
|
|
$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(Expr $expr) : bool
|
|
{
|
|
if (!$expr instanceof FuncCall) {
|
|
return \false;
|
|
}
|
|
return $this->nodeNameResolver->isName($expr, '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]) && $funcCall->args[2] instanceof Arg) {
|
|
$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
|
|
{
|
|
if (!isset($funcCall->args[$argumentPosition])) {
|
|
return null;
|
|
}
|
|
if (!$funcCall->args[$argumentPosition] instanceof Arg) {
|
|
return null;
|
|
}
|
|
$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;
|
|
}
|
|
}
|