2020-03-26 21:23:52 +01:00
|
|
|
<?php
|
|
|
|
|
2021-05-09 20:15:43 +00:00
|
|
|
declare (strict_types=1);
|
2020-03-26 21:23:52 +01:00
|
|
|
namespace Rector\DeadCode;
|
|
|
|
|
2021-07-01 12:49:39 +00:00
|
|
|
use PhpParser\Node;
|
2020-03-26 21:23:52 +01:00
|
|
|
use PhpParser\Node\Expr;
|
|
|
|
use PhpParser\Node\Expr\Array_;
|
|
|
|
use PhpParser\Node\Expr\BinaryOp\NotEqual;
|
|
|
|
use PhpParser\Node\Expr\BinaryOp\NotIdentical;
|
|
|
|
use PhpParser\Node\Expr\BooleanNot;
|
|
|
|
use PhpParser\Node\Expr\Empty_;
|
2021-07-01 12:49:39 +00:00
|
|
|
use PhpParser\Node\Expr\Variable;
|
|
|
|
use PhpParser\Node\Param;
|
2020-03-26 21:23:52 +01:00
|
|
|
use PhpParser\Node\Stmt\If_;
|
2021-07-01 12:49:39 +00:00
|
|
|
use PHPStan\Type\ArrayType;
|
2021-07-14 07:23:54 +00:00
|
|
|
use Rector\Core\NodeAnalyzer\ParamAnalyzer;
|
2021-02-19 13:01:23 +01:00
|
|
|
use Rector\Core\PhpParser\Comparing\NodeComparator;
|
2021-07-01 12:49:39 +00:00
|
|
|
use Rector\Core\PhpParser\Node\BetterNodeFinder;
|
2020-04-08 17:14:21 +02:00
|
|
|
use Rector\NodeTypeResolver\NodeTypeResolver;
|
2020-03-26 21:23:52 +01:00
|
|
|
final class UselessIfCondBeforeForeachDetector
|
|
|
|
{
|
|
|
|
/**
|
2021-12-04 12:47:17 +00:00
|
|
|
* @readonly
|
2021-05-10 23:39:21 +00:00
|
|
|
* @var \Rector\NodeTypeResolver\NodeTypeResolver
|
2020-03-26 21:23:52 +01:00
|
|
|
*/
|
2021-02-19 13:01:23 +01:00
|
|
|
private $nodeTypeResolver;
|
2020-04-08 17:14:21 +02:00
|
|
|
/**
|
2021-12-04 12:47:17 +00:00
|
|
|
* @readonly
|
2021-05-10 23:39:21 +00:00
|
|
|
* @var \Rector\Core\PhpParser\Comparing\NodeComparator
|
2020-04-08 17:14:21 +02:00
|
|
|
*/
|
2021-02-19 13:01:23 +01:00
|
|
|
private $nodeComparator;
|
2021-07-01 12:49:39 +00:00
|
|
|
/**
|
2021-12-04 12:47:17 +00:00
|
|
|
* @readonly
|
2021-07-01 12:49:39 +00:00
|
|
|
* @var \Rector\Core\PhpParser\Node\BetterNodeFinder
|
|
|
|
*/
|
|
|
|
private $betterNodeFinder;
|
2021-07-14 07:23:54 +00:00
|
|
|
/**
|
2021-12-04 12:47:17 +00:00
|
|
|
* @readonly
|
2021-07-14 07:23:54 +00:00
|
|
|
* @var \Rector\Core\NodeAnalyzer\ParamAnalyzer
|
|
|
|
*/
|
|
|
|
private $paramAnalyzer;
|
|
|
|
public function __construct(\Rector\NodeTypeResolver\NodeTypeResolver $nodeTypeResolver, \Rector\Core\PhpParser\Comparing\NodeComparator $nodeComparator, \Rector\Core\PhpParser\Node\BetterNodeFinder $betterNodeFinder, \Rector\Core\NodeAnalyzer\ParamAnalyzer $paramAnalyzer)
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
2020-04-08 17:14:21 +02:00
|
|
|
$this->nodeTypeResolver = $nodeTypeResolver;
|
2021-02-19 13:01:23 +01:00
|
|
|
$this->nodeComparator = $nodeComparator;
|
2021-07-01 12:49:39 +00:00
|
|
|
$this->betterNodeFinder = $betterNodeFinder;
|
2021-07-14 07:23:54 +00:00
|
|
|
$this->paramAnalyzer = $paramAnalyzer;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Matches:
|
|
|
|
* !empty($values)
|
|
|
|
*/
|
2021-05-10 22:23:08 +00:00
|
|
|
public function isMatchingNotEmpty(\PhpParser\Node\Stmt\If_ $if, \PhpParser\Node\Expr $foreachExpr) : bool
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
|
|
|
$cond = $if->cond;
|
2021-05-10 22:23:08 +00:00
|
|
|
if (!$cond instanceof \PhpParser\Node\Expr\BooleanNot) {
|
2021-05-09 20:15:43 +00:00
|
|
|
return \false;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
2021-05-10 22:23:08 +00:00
|
|
|
if (!$cond->expr instanceof \PhpParser\Node\Expr\Empty_) {
|
2021-05-09 20:15:43 +00:00
|
|
|
return \false;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
/** @var Empty_ $empty */
|
|
|
|
$empty = $cond->expr;
|
2021-05-09 20:15:43 +00:00
|
|
|
if (!$this->nodeComparator->areNodesEqual($empty->expr, $foreachExpr)) {
|
|
|
|
return \false;
|
2020-04-08 17:14:21 +02:00
|
|
|
}
|
|
|
|
// is array though?
|
2021-10-07 19:40:12 +00:00
|
|
|
$arrayType = $this->nodeTypeResolver->getType($empty->expr);
|
2021-07-01 12:49:39 +00:00
|
|
|
if (!$arrayType instanceof \PHPStan\Type\ArrayType) {
|
|
|
|
return \false;
|
|
|
|
}
|
|
|
|
$previousParam = $this->fromPreviousParam($foreachExpr);
|
|
|
|
if (!$previousParam instanceof \PhpParser\Node\Param) {
|
|
|
|
return \true;
|
|
|
|
}
|
2021-07-14 07:23:54 +00:00
|
|
|
if ($this->paramAnalyzer->isNullable($previousParam)) {
|
|
|
|
return \false;
|
|
|
|
}
|
|
|
|
return !$this->paramAnalyzer->hasDefaultNull($previousParam);
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
/**
|
|
|
|
* Matches:
|
|
|
|
* $values !== []
|
|
|
|
* $values != []
|
|
|
|
* [] !== $values
|
|
|
|
* [] != $values
|
|
|
|
*/
|
2021-05-10 22:23:08 +00:00
|
|
|
public function isMatchingNotIdenticalEmptyArray(\PhpParser\Node\Stmt\If_ $if, \PhpParser\Node\Expr $foreachExpr) : bool
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
2021-05-10 22:23:08 +00:00
|
|
|
if (!$if->cond instanceof \PhpParser\Node\Expr\BinaryOp\NotIdentical && !$if->cond instanceof \PhpParser\Node\Expr\BinaryOp\NotEqual) {
|
2021-05-09 20:15:43 +00:00
|
|
|
return \false;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
/** @var NotIdentical|NotEqual $notIdentical */
|
|
|
|
$notIdentical = $if->cond;
|
|
|
|
return $this->isMatchingNotBinaryOp($notIdentical, $foreachExpr);
|
|
|
|
}
|
2021-07-01 12:49:39 +00:00
|
|
|
private function fromPreviousParam(\PhpParser\Node\Expr $expr) : ?\PhpParser\Node
|
|
|
|
{
|
|
|
|
return $this->betterNodeFinder->findFirstPreviousOfNode($expr, function (\PhpParser\Node $node) use($expr) : bool {
|
|
|
|
if (!$node instanceof \PhpParser\Node\Param) {
|
|
|
|
return \false;
|
|
|
|
}
|
|
|
|
if (!$node->var instanceof \PhpParser\Node\Expr\Variable) {
|
|
|
|
return \false;
|
|
|
|
}
|
|
|
|
return $this->nodeComparator->areNodesEqual($node->var, $expr);
|
|
|
|
});
|
|
|
|
}
|
2020-03-26 21:23:52 +01:00
|
|
|
/**
|
2021-10-30 14:18:31 +00:00
|
|
|
* @param \PhpParser\Node\Expr\BinaryOp\NotEqual|\PhpParser\Node\Expr\BinaryOp\NotIdentical $binaryOp
|
2020-03-26 21:23:52 +01:00
|
|
|
*/
|
2021-06-29 14:24:45 +00:00
|
|
|
private function isMatchingNotBinaryOp($binaryOp, \PhpParser\Node\Expr $foreachExpr) : bool
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
|
|
|
if ($this->isEmptyArrayAndForeachedVariable($binaryOp->left, $binaryOp->right, $foreachExpr)) {
|
2021-05-09 20:15:43 +00:00
|
|
|
return \true;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
return $this->isEmptyArrayAndForeachedVariable($binaryOp->right, $binaryOp->left, $foreachExpr);
|
|
|
|
}
|
2021-05-10 22:23:08 +00:00
|
|
|
private function isEmptyArrayAndForeachedVariable(\PhpParser\Node\Expr $leftExpr, \PhpParser\Node\Expr $rightExpr, \PhpParser\Node\Expr $foreachExpr) : bool
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
2021-05-09 20:15:43 +00:00
|
|
|
if (!$this->isEmptyArray($leftExpr)) {
|
|
|
|
return \false;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
2021-02-19 13:01:23 +01:00
|
|
|
return $this->nodeComparator->areNodesEqual($foreachExpr, $rightExpr);
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
2021-05-10 22:23:08 +00:00
|
|
|
private function isEmptyArray(\PhpParser\Node\Expr $expr) : bool
|
2020-03-26 21:23:52 +01:00
|
|
|
{
|
2021-05-10 22:23:08 +00:00
|
|
|
if (!$expr instanceof \PhpParser\Node\Expr\Array_) {
|
2021-05-09 20:15:43 +00:00
|
|
|
return \false;
|
2020-03-26 21:23:52 +01:00
|
|
|
}
|
|
|
|
return $expr->items === [];
|
|
|
|
}
|
|
|
|
}
|