Skip to content

Commit 12a0b4e

Browse files
authored
RegexArrayShapeMatcher - turn more details immutable
1 parent e876b66 commit 12a0b4e

File tree

3 files changed

+222
-97
lines changed

3 files changed

+222
-97
lines changed

src/Type/Php/RegexArrayShapeMatcher.php

+16-86
Original file line numberDiff line numberDiff line change
@@ -15,14 +15,13 @@
1515
use PHPStan\Type\IntegerRangeType;
1616
use PHPStan\Type\IntegerType;
1717
use PHPStan\Type\NullType;
18-
use PHPStan\Type\Regex\RegexAlternation;
1918
use PHPStan\Type\Regex\RegexCapturingGroup;
2019
use PHPStan\Type\Regex\RegexExpressionHelper;
20+
use PHPStan\Type\Regex\RegexGroupList;
2121
use PHPStan\Type\Regex\RegexGroupParser;
2222
use PHPStan\Type\StringType;
2323
use PHPStan\Type\Type;
2424
use PHPStan\Type\TypeCombinator;
25-
use function array_reverse;
2625
use function count;
2726
use function in_array;
2827
use function is_string;
@@ -115,16 +114,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
115114
}
116115
[$groupList, $markVerbs] = $parseResult;
117116

118-
$trailingOptionals = 0;
119-
foreach (array_reverse($groupList) as $captureGroup) {
120-
if (!$captureGroup->isOptional()) {
121-
break;
122-
}
123-
$trailingOptionals++;
124-
}
125-
126-
$onlyOptionalTopLevelGroup = $this->getOnlyOptionalTopLevelGroup($groupList);
127-
$onlyTopLevelAlternation = $this->getOnlyTopLevelAlternation($groupList);
117+
$regexGroupList = new RegexGroupList($groupList);
118+
$trailingOptionals = $regexGroupList->countTrailingOptionals();
119+
$onlyOptionalTopLevelGroup = $regexGroupList->getOnlyOptionalTopLevelGroup();
120+
$onlyTopLevelAlternation = $regexGroupList->getOnlyTopLevelAlternation();
128121
$flags ??= 0;
129122

130123
if (
@@ -134,11 +127,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
134127
) {
135128
// if only one top level capturing optional group exists
136129
// we build a more precise tagged union of a empty-match and a match with the group
137-
138-
$onlyOptionalTopLevelGroup->forceNonOptional();
130+
$regexGroupList = $regexGroupList->forceGroupNonOptional($onlyOptionalTopLevelGroup);
139131

140132
$combiType = $this->buildArrayType(
141-
$groupList,
133+
$regexGroupList,
142134
$wasMatched,
143135
$trailingOptionals,
144136
$flags,
@@ -154,8 +146,6 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
154146
);
155147
}
156148

157-
$onlyOptionalTopLevelGroup->clearOverrides();
158-
159149
return $combiType;
160150
} elseif (
161151
!$matchesAll
@@ -168,24 +158,24 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
168158
$combiTypes = [];
169159
$isOptionalAlternation = false;
170160
foreach ($onlyTopLevelAlternation->getGroupCombinations() as $groupCombo) {
171-
$comboList = $groupList;
161+
$comboList = new RegexGroupList($groupList);
172162

173163
$beforeCurrentCombo = true;
174-
foreach ($comboList as $groupId => $group) {
175-
if (in_array($groupId, $groupCombo, true)) {
164+
foreach ($comboList as $group) {
165+
if (in_array($group->getId(), $groupCombo, true)) {
176166
$isOptionalAlternation = $group->inOptionalAlternation();
177-
$group->forceNonOptional();
167+
$comboList = $comboList->forceGroupNonOptional($group);
178168
$beforeCurrentCombo = false;
179169
} elseif ($beforeCurrentCombo && !$group->resetsGroupCounter()) {
180-
$group->forceNonOptional();
181-
$group->forceType(
170+
$comboList = $comboList->forceGroupTypeAndNonOptional(
171+
$group,
182172
$this->containsUnmatchedAsNull($flags, $matchesAll) ? new NullType() : new ConstantStringType(''),
183173
);
184174
} elseif (
185175
$group->getAlternationId() === $onlyTopLevelAlternation->getId()
186176
&& !$this->containsUnmatchedAsNull($flags, $matchesAll)
187177
) {
188-
unset($comboList[$groupId]);
178+
$comboList = $comboList->removeGroup($group);
189179
}
190180
}
191181

@@ -199,11 +189,6 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
199189
);
200190

201191
$combiTypes[] = $combiType;
202-
203-
foreach ($groupCombo as $groupId) {
204-
$group = $comboList[$groupId];
205-
$group->clearOverrides();
206-
}
207192
}
208193

209194
if (
@@ -223,7 +208,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
223208
// the general case, which should work in all cases but does not yield the most
224209
// precise result possible in some cases
225210
return $this->buildArrayType(
226-
$groupList,
211+
$regexGroupList,
227212
$wasMatched,
228213
$trailingOptionals,
229214
$flags,
@@ -233,65 +218,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
233218
}
234219

235220
/**
236-
* @param array<int, RegexCapturingGroup> $captureGroups
237-
*/
238-
private function getOnlyOptionalTopLevelGroup(array $captureGroups): ?RegexCapturingGroup
239-
{
240-
$group = null;
241-
foreach ($captureGroups as $captureGroup) {
242-
if (!$captureGroup->isTopLevel()) {
243-
continue;
244-
}
245-
246-
if (!$captureGroup->isOptional()) {
247-
return null;
248-
}
249-
250-
if ($group !== null) {
251-
return null;
252-
}
253-
254-
$group = $captureGroup;
255-
}
256-
257-
return $group;
258-
}
259-
260-
/**
261-
* @param array<int, RegexCapturingGroup> $captureGroups
262-
*/
263-
private function getOnlyTopLevelAlternation(array $captureGroups): ?RegexAlternation
264-
{
265-
$alternation = null;
266-
foreach ($captureGroups as $captureGroup) {
267-
if (!$captureGroup->isTopLevel()) {
268-
continue;
269-
}
270-
271-
if (!$captureGroup->inAlternation()) {
272-
return null;
273-
}
274-
275-
if ($captureGroup->inOptionalQuantification()) {
276-
return null;
277-
}
278-
279-
if ($alternation === null) {
280-
$alternation = $captureGroup->getAlternation();
281-
} elseif ($alternation->getId() !== $captureGroup->getAlternation()->getId()) {
282-
return null;
283-
}
284-
}
285-
286-
return $alternation;
287-
}
288-
289-
/**
290-
* @param array<RegexCapturingGroup> $captureGroups
291221
* @param list<string> $markVerbs
292222
*/
293223
private function buildArrayType(
294-
array $captureGroups,
224+
RegexGroupList $captureGroups,
295225
TrinaryLogic $wasMatched,
296226
int $trailingOptionals,
297227
int $flags,

src/Type/Regex/RegexCapturingGroup.php

+40-11
Original file line numberDiff line numberDiff line change
@@ -7,17 +7,15 @@
77
final class RegexCapturingGroup
88
{
99

10-
private bool $forceNonOptional = false;
11-
12-
private ?Type $forceType = null;
13-
1410
public function __construct(
1511
private readonly int $id,
1612
private readonly ?string $name,
1713
private readonly ?RegexAlternation $alternation,
1814
private readonly bool $inOptionalQuantification,
1915
private readonly RegexCapturingGroup|RegexNonCapturingGroup|null $parent,
2016
private readonly Type $type,
17+
private readonly bool $forceNonOptional = false,
18+
private readonly ?Type $forceType = null,
2119
)
2220
{
2321
}
@@ -27,20 +25,46 @@ public function getId(): int
2725
return $this->id;
2826
}
2927

30-
public function forceNonOptional(): void
28+
public function forceNonOptional(): self
3129
{
32-
$this->forceNonOptional = true;
30+
return new self(
31+
$this->id,
32+
$this->name,
33+
$this->alternation,
34+
$this->inOptionalQuantification,
35+
$this->parent,
36+
$this->type,
37+
true,
38+
$this->forceType,
39+
);
3340
}
3441

35-
public function forceType(Type $type): void
42+
public function forceType(Type $type): self
3643
{
37-
$this->forceType = $type;
44+
return new self(
45+
$this->id,
46+
$this->name,
47+
$this->alternation,
48+
$this->inOptionalQuantification,
49+
$this->parent,
50+
$type,
51+
$this->forceNonOptional,
52+
$this->forceType,
53+
);
3854
}
3955

40-
public function clearOverrides(): void
56+
public function withParent(RegexCapturingGroup|RegexNonCapturingGroup $parent): self
4157
{
42-
$this->forceNonOptional = false;
43-
$this->forceType = null;
58+
return new self(
59+
$this->id,
60+
$this->name,
61+
$this->alternation,
62+
$this->inOptionalQuantification,
63+
$parent,
64+
$this->type,
65+
$this->forceNonOptional,
66+
$this->forceType,
67+
);
4468
}
4569

4670
public function resetsGroupCounter(): bool
@@ -128,4 +152,9 @@ public function getType(): Type
128152
return $this->type;
129153
}
130154

155+
public function getParent(): RegexCapturingGroup|RegexNonCapturingGroup|null
156+
{
157+
return $this->parent;
158+
}
159+
131160
}

0 commit comments

Comments
 (0)