Skip to content

Commit 9437056

Browse files
authored
RegexArrayShapeMatcher - Fix matching literal dot character
1 parent b3c25b8 commit 9437056

File tree

2 files changed

+49
-5
lines changed

2 files changed

+49
-5
lines changed

src/Type/Regex/RegexGroupParser.php

+7-5
Original file line numberDiff line numberDiff line change
@@ -445,14 +445,14 @@ private function walkGroupAst(
445445
foreach ($children as $child) {
446446
$oldLiterals = $onlyLiterals;
447447

448-
$this->getLiteralValue($child, $oldLiterals, true, $patternModifiers);
448+
$this->getLiteralValue($child, $oldLiterals, true, $patternModifiers, true);
449449
foreach ($oldLiterals ?? [] as $oldLiteral) {
450450
$newLiterals[] = $oldLiteral;
451451
}
452452
}
453453
$onlyLiterals = $newLiterals;
454454
} elseif ($ast->getId() === 'token') {
455-
$literalValue = $this->getLiteralValue($ast, $onlyLiterals, !$inClass, $patternModifiers);
455+
$literalValue = $this->getLiteralValue($ast, $onlyLiterals, !$inClass, $patternModifiers, false);
456456
if ($literalValue !== null) {
457457
if (Strings::match($literalValue, '/^\d+$/') === null) {
458458
$isNumeric = TrinaryLogic::createNo();
@@ -508,7 +508,7 @@ private function isMaybeEmptyNode(TreeNode $node, string $patternModifiers, bool
508508
}
509509
}
510510

511-
$literal = $this->getLiteralValue($node, $onlyLiterals, false, $patternModifiers);
511+
$literal = $this->getLiteralValue($node, $onlyLiterals, false, $patternModifiers, false);
512512
if ($literal !== null) {
513513
if ($literal !== '' && $literal !== '0') {
514514
$isNonFalsy = true;
@@ -528,7 +528,7 @@ private function isMaybeEmptyNode(TreeNode $node, string $patternModifiers, bool
528528
/**
529529
* @param array<string>|null $onlyLiterals
530530
*/
531-
private function getLiteralValue(TreeNode $node, ?array &$onlyLiterals, bool $appendLiterals, string $patternModifiers): ?string
531+
private function getLiteralValue(TreeNode $node, ?array &$onlyLiterals, bool $appendLiterals, string $patternModifiers, bool $inCharacterClass): ?string
532532
{
533533
if ($node->getId() !== 'token') {
534534
return null;
@@ -551,15 +551,17 @@ private function getLiteralValue(TreeNode $node, ?array &$onlyLiterals, bool $ap
551551
return null;
552552
}
553553

554+
$isEscaped = false;
554555
if (strlen($value) > 1 && $value[0] === '\\') {
555556
$value = substr($value, 1) ?: '';
557+
$isEscaped = true;
556558
}
557559

558560
if (
559561
$appendLiterals
560562
&& in_array($token, ['literal', 'range', 'class_', '_class_literal'], true)
561563
&& $onlyLiterals !== null
562-
&& !in_array($value, ['.'], true)
564+
&& (!in_array($value, ['.'], true) || $isEscaped || $inCharacterClass)
563565
) {
564566
if ($onlyLiterals === []) {
565567
$onlyLiterals = [$value];
+42
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
<?php declare(strict_types = 1);
2+
3+
namespace Bug11699;
4+
5+
use function PHPStan\Testing\assertType;
6+
7+
function doFoo():void {
8+
$string = 'Foo.bar';
9+
$match = [];
10+
$result = preg_match('~(?<AB>[\~,\?\.])~', $string, $match);
11+
if ($result === 1) {
12+
assertType("','|'.'|'?'|'~'", $match['AB']);
13+
}
14+
}
15+
16+
function doFoo1():void {
17+
$string = 'Foo.bar';
18+
$match = [];
19+
$result = preg_match('~(?<AB>[\~,\?.])~', $string, $match); // dot in character class does not need to be escaped
20+
if ($result === 1) {
21+
assertType("','|'.'|'?'|'~'", $match['AB']);
22+
}
23+
}
24+
25+
function doFoo2():void {
26+
$string = 'Foo.bar';
27+
$match = [];
28+
$result = preg_match('~(?<AB>.)~', $string, $match);
29+
if ($result === 1) {
30+
assertType("non-empty-string", $match['AB']);
31+
}
32+
}
33+
34+
35+
function doFoo3():void {
36+
$string = 'Foo.bar';
37+
$match = [];
38+
$result = preg_match('~(?<AB>\.)~', $string, $match);
39+
if ($result === 1) {
40+
assertType("'.'", $match['AB']);
41+
}
42+
}

0 commit comments

Comments
 (0)