mirror of
https://github.com/rectorphp/rector.git
synced 2025-01-17 21:38:22 +01:00
7f755fff28
2d862a7c77
[CodingStyle] Skip equal modifier on direct parent method compatible on MakeInheritedMethodVisibilitySameAsParentRector (#6406)
104 lines
3.4 KiB
PHP
104 lines
3.4 KiB
PHP
<?php
|
|
|
|
declare (strict_types=1);
|
|
namespace Rector\Php55;
|
|
|
|
use RectorPrefix202411\Nette\Utils\Strings;
|
|
use PhpParser\Node\Expr;
|
|
use PhpParser\Node\Expr\BinaryOp\Concat;
|
|
use PhpParser\Node\Scalar\String_;
|
|
final class RegexMatcher
|
|
{
|
|
/**
|
|
* @var string
|
|
* @see https://regex101.com/r/Ok4wuE/1
|
|
*/
|
|
private const LAST_E_REGEX = '#(\\w+)?e(\\w+)?$#';
|
|
/**
|
|
* @var string
|
|
* @see https://regex101.com/r/2NWVwT/1
|
|
*/
|
|
private const LETTER_SUFFIX_REGEX = '#(?<modifiers>\\w+)$#';
|
|
/**
|
|
* @var string[]
|
|
* @see https://www.php.net/manual/en/reference.pcre.pattern.modifiers.php
|
|
*/
|
|
private const ALL_MODIFIERS_VALUES = ['i', 'm', 's', 'x', 'e', 'A', 'D', 'S', 'U', 'X', 'J', 'u'];
|
|
/**
|
|
* @return \PhpParser\Node\Expr\BinaryOp\Concat|\PhpParser\Node\Scalar\String_|null
|
|
*/
|
|
public function resolvePatternExpressionWithoutEIfFound(Expr $expr)
|
|
{
|
|
if ($expr instanceof String_) {
|
|
$pattern = $expr->value;
|
|
$delimiter = $pattern[0];
|
|
switch ($delimiter) {
|
|
case '(':
|
|
$delimiter = ')';
|
|
break;
|
|
case '{':
|
|
$delimiter = '}';
|
|
break;
|
|
case '[':
|
|
$delimiter = ']';
|
|
break;
|
|
case '<':
|
|
$delimiter = '>';
|
|
break;
|
|
default:
|
|
$delimiter = $delimiter;
|
|
break;
|
|
}
|
|
/** @var string $modifiers */
|
|
$modifiers = $this->resolveModifiers((string) Strings::after($pattern, $delimiter, -1));
|
|
if (\strpos($modifiers, 'e') === \false) {
|
|
return null;
|
|
}
|
|
$expr->value = $this->createPatternWithoutE($pattern, $delimiter, $modifiers);
|
|
return $expr;
|
|
}
|
|
if ($expr instanceof Concat) {
|
|
return $this->matchConcat($expr);
|
|
}
|
|
return null;
|
|
}
|
|
private function resolveModifiers(string $modifiersCandidate) : string
|
|
{
|
|
$modifiers = '';
|
|
for ($modifierIndex = 0; $modifierIndex < \strlen($modifiersCandidate); ++$modifierIndex) {
|
|
if (!\in_array($modifiersCandidate[$modifierIndex], self::ALL_MODIFIERS_VALUES, \true)) {
|
|
$modifiers = '';
|
|
continue;
|
|
}
|
|
$modifiers .= $modifiersCandidate[$modifierIndex];
|
|
}
|
|
return $modifiers;
|
|
}
|
|
private function createPatternWithoutE(string $pattern, string $delimiter, string $modifiers) : string
|
|
{
|
|
$modifiersWithoutE = \str_replace('e', '', $modifiers);
|
|
return Strings::before($pattern, $delimiter, -1) . $delimiter . $modifiersWithoutE;
|
|
}
|
|
private function matchConcat(Concat $concat) : ?Concat
|
|
{
|
|
// cause parse error
|
|
if (!$concat->left instanceof Concat) {
|
|
return null;
|
|
}
|
|
$lastItem = $concat->right;
|
|
if (!$lastItem instanceof String_) {
|
|
return null;
|
|
}
|
|
$matches = Strings::match($lastItem->value, self::LETTER_SUFFIX_REGEX);
|
|
if (!isset($matches['modifiers'])) {
|
|
return null;
|
|
}
|
|
if (\strpos((string) $matches['modifiers'], 'e') === \false) {
|
|
return null;
|
|
}
|
|
// replace last "e" in the code
|
|
$lastItem->value = Strings::replace($lastItem->value, self::LAST_E_REGEX, '$1$2');
|
|
return $concat;
|
|
}
|
|
}
|