rector/rules/Php80/NodeResolver/SwitchExprsResolver.php
Tomas Votruba b6c77d2814 Updated Rector to commit 3fba7c625c5903edc0f8069faf5b208761a16f9a
3fba7c625c [Php80] Skip default no stmt next is case on ChangeSwitchToMatchRector (#2728)
2022-08-03 18:07:32 +00:00

127 lines
4.2 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\Php80\NodeResolver;
use PhpParser\Node\Expr;
use PhpParser\Node\Expr\Assign;
use PhpParser\Node\Stmt\Break_;
use PhpParser\Node\Stmt\Case_;
use PhpParser\Node\Stmt\Expression;
use PhpParser\Node\Stmt\Return_;
use PhpParser\Node\Stmt\Switch_;
use PhpParser\Node\Stmt\Throw_;
use Rector\Php80\Enum\MatchKind;
use Rector\Php80\ValueObject\CondAndExpr;
final class SwitchExprsResolver
{
/**
* @return CondAndExpr[]
*/
public function resolve(Switch_ $switch) : array
{
$newSwitch = clone $switch;
$condAndExpr = [];
$collectionEmptyCasesCond = [];
$this->moveDefaultCaseToLast($newSwitch);
foreach ($newSwitch->cases as $key => $case) {
\assert(\is_int($key));
if (!$this->isValidCase($case)) {
return [];
}
if ($case->stmts === [] && $case->cond instanceof Expr) {
$collectionEmptyCasesCond[$key] = $case->cond;
}
}
foreach ($newSwitch->cases as $key => $case) {
if ($case->stmts === []) {
continue;
}
$expr = $case->stmts[0];
if ($expr instanceof Expression) {
$expr = $expr->expr;
}
$condExprs = [];
if ($case->cond !== null) {
$emptyCasesCond = [];
foreach ($collectionEmptyCasesCond as $i => $collectionEmptyCaseCond) {
if ($i > $key) {
break;
}
$emptyCasesCond[$i] = $collectionEmptyCaseCond;
unset($collectionEmptyCasesCond[$i]);
}
$condExprs = $emptyCasesCond;
$condExprs[] = $case->cond;
}
if ($expr instanceof Return_) {
$returnedExpr = $expr->expr;
if (!$returnedExpr instanceof Expr) {
return [];
}
$condAndExpr[] = new CondAndExpr($condExprs, $returnedExpr, MatchKind::RETURN);
} elseif ($expr instanceof Assign) {
$condAndExpr[] = new CondAndExpr($condExprs, $expr, MatchKind::ASSIGN);
} elseif ($expr instanceof Expr) {
$condAndExpr[] = new CondAndExpr($condExprs, $expr, MatchKind::NORMAL);
} elseif ($expr instanceof Throw_) {
$throwExpr = new Expr\Throw_($expr->expr);
$condAndExpr[] = new CondAndExpr($condExprs, $throwExpr, MatchKind::THROW);
} else {
return [];
}
}
return $condAndExpr;
}
private function moveDefaultCaseToLast(Switch_ $switch) : void
{
foreach ($switch->cases as $key => $case) {
if ($case->cond instanceof Expr) {
continue;
}
// not has next? default is at the end, no need move
if (!isset($switch->cases[$key + 1])) {
return;
}
// current default has no stmt? keep as is as rely to next case
if ($case->stmts === []) {
return;
}
for ($loop = $key - 1; $loop >= 0; --$loop) {
if ($switch->cases[$loop]->stmts !== []) {
break;
}
unset($switch->cases[$loop]);
}
$caseToMove = $switch->cases[$key];
unset($switch->cases[$key]);
$switch->cases[] = $caseToMove;
break;
}
}
private function isValidCase(Case_ $case) : bool
{
// prepend to previous one
if ($case->stmts === []) {
return \true;
}
if (\count($case->stmts) === 2 && $case->stmts[1] instanceof Break_) {
return \true;
}
// default throws stmts
if (\count($case->stmts) !== 1) {
return \false;
}
// throws expressoin
if ($case->stmts[0] instanceof Throw_) {
return \true;
}
// instant return
if ($case->stmts[0] instanceof Return_) {
return \true;
}
// default value
return $case->cond === null;
}
}