rector/vendor/rector/rector-downgrade-php/rules/DowngradePhp73/Rector/FuncCall/SetCookieOptionsArrayToArgumentsRector.php
2023-05-13 16:44:37 +00:00

139 lines
4.3 KiB
PHP

<?php
declare (strict_types=1);
namespace Rector\DowngradePhp73\Rector\FuncCall;
use PhpParser\BuilderHelpers;
use PhpParser\Node;
use PhpParser\Node\Arg;
use PhpParser\Node\Expr\Array_;
use PhpParser\Node\Expr\ArrayItem;
use PhpParser\Node\Expr\FuncCall;
use PhpParser\Node\Scalar\String_;
use Rector\Core\Exception\ShouldNotHappenException;
use Rector\Core\Rector\AbstractRector;
use Symplify\RuleDocGenerator\ValueObject\CodeSample\CodeSample;
use Symplify\RuleDocGenerator\ValueObject\RuleDefinition;
/**
* @see \Rector\Tests\DowngradePhp73\Rector\FuncCall\SetCookieOptionsArrayToArgumentsRector\SetCookieOptionsArrayToArgumentsRectorTest
*/
final class SetCookieOptionsArrayToArgumentsRector extends AbstractRector
{
/**
* Conversion table from argument index to options name
* @var array<string, int>
*/
private const ARGUMENT_ORDER = ['expires' => 2, 'path' => 3, 'domain' => 4, 'secure' => 5, 'httponly' => 6];
/**
* Conversion table from argument index to options name
* @var array<int, int|string|bool>
*/
private const ARGUMENT_DEFAULT_VALUES = [2 => 0, 3 => '', 4 => '', 5 => \false, 6 => \false];
/**
* @var int
*/
private $highestIndex = 1;
public function getRuleDefinition() : RuleDefinition
{
return new RuleDefinition('Convert setcookie option array to arguments', [new CodeSample(<<<'CODE_SAMPLE'
setcookie('name', $value, ['expires' => 360]);
CODE_SAMPLE
, <<<'CODE_SAMPLE'
setcookie('name', $value, 360);
CODE_SAMPLE
)]);
}
/**
* @return array<class-string<Node>>
*/
public function getNodeTypes() : array
{
return [FuncCall::class];
}
/**
* @param FuncCall $node
*/
public function refactor(Node $node) : ?Node
{
if ($this->shouldSkip($node)) {
return null;
}
$node->args = $this->composeNewArgs($node);
return $node;
}
private function shouldSkip(FuncCall $funcCall) : bool
{
if (!$this->isNames($funcCall, ['setcookie', 'setrawcookie'])) {
return \true;
}
if (\count($funcCall->getArgs()) < 3) {
return \true;
}
$thirdArg = $funcCall->getArgs()[2];
return !$thirdArg->value instanceof Array_;
}
/**
* @return Arg[]
*/
private function composeNewArgs(FuncCall $funcCall) : array
{
$this->highestIndex = 1;
if (\count($funcCall->getArgs()) < 3) {
return [];
}
$firstArg = $funcCall->getArgs()[0];
$secondArg = $funcCall->getArgs()[1];
$newArgs = [$firstArg, $secondArg];
$thirdArg = $funcCall->getArgs()[2];
/** @var Array_ $optionsArray */
$optionsArray = $thirdArg->value;
foreach ($optionsArray->items as $arrayItem) {
if (!$arrayItem instanceof ArrayItem) {
continue;
}
$value = $arrayItem->value;
/** @var String_ $key */
$key = $arrayItem->key;
$name = $key->value;
if (!$this->isMappableArrayKey($name)) {
continue;
}
$order = self::ARGUMENT_ORDER[$name];
if ($order > $this->highestIndex) {
$this->highestIndex = $order;
}
$newArgs[$order] = new Arg($value);
}
$newArgs = $this->fillMissingArgumentsWithDefaultValues($newArgs);
\ksort($newArgs);
return $newArgs;
}
private function isMappableArrayKey(string $key) : bool
{
return isset(self::ARGUMENT_ORDER[$key]);
}
/**
* @param array<int, Arg> $args
* @return array<int, Arg>
*/
private function fillMissingArgumentsWithDefaultValues(array $args) : array
{
for ($i = 1; $i < $this->highestIndex; ++$i) {
if (isset($args[$i])) {
continue;
}
$args[$i] = $this->createDefaultValueArg($i);
}
return $args;
}
private function createDefaultValueArg(int $argumentIndex) : Arg
{
if (!\array_key_exists($argumentIndex, self::ARGUMENT_DEFAULT_VALUES)) {
throw new ShouldNotHappenException();
}
$argumentDefaultValue = self::ARGUMENT_DEFAULT_VALUES[$argumentIndex];
$expr = BuilderHelpers::normalizeValue($argumentDefaultValue);
return new Arg($expr);
}
}