15
15
use PHPStan \Type \IntegerRangeType ;
16
16
use PHPStan \Type \IntegerType ;
17
17
use PHPStan \Type \NullType ;
18
- use PHPStan \Type \Regex \RegexAlternation ;
19
18
use PHPStan \Type \Regex \RegexCapturingGroup ;
20
19
use PHPStan \Type \Regex \RegexExpressionHelper ;
20
+ use PHPStan \Type \Regex \RegexGroupList ;
21
21
use PHPStan \Type \Regex \RegexGroupParser ;
22
22
use PHPStan \Type \StringType ;
23
23
use PHPStan \Type \Type ;
24
24
use PHPStan \Type \TypeCombinator ;
25
- use function array_reverse ;
26
25
use function count ;
27
26
use function in_array ;
28
27
use function is_string ;
@@ -115,16 +114,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
115
114
}
116
115
[$ groupList , $ markVerbs ] = $ parseResult ;
117
116
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 ();
128
121
$ flags ??= 0 ;
129
122
130
123
if (
@@ -134,11 +127,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
134
127
) {
135
128
// if only one top level capturing optional group exists
136
129
// 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 );
139
131
140
132
$ combiType = $ this ->buildArrayType (
141
- $ groupList ,
133
+ $ regexGroupList ,
142
134
$ wasMatched ,
143
135
$ trailingOptionals ,
144
136
$ flags ,
@@ -154,8 +146,6 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
154
146
);
155
147
}
156
148
157
- $ onlyOptionalTopLevelGroup ->clearOverrides ();
158
-
159
149
return $ combiType ;
160
150
} elseif (
161
151
!$ matchesAll
@@ -168,24 +158,24 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
168
158
$ combiTypes = [];
169
159
$ isOptionalAlternation = false ;
170
160
foreach ($ onlyTopLevelAlternation ->getGroupCombinations () as $ groupCombo ) {
171
- $ comboList = $ groupList ;
161
+ $ comboList = new RegexGroupList ( $ groupList) ;
172
162
173
163
$ 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 )) {
176
166
$ isOptionalAlternation = $ group ->inOptionalAlternation ();
177
- $ group -> forceNonOptional ( );
167
+ $ comboList = $ comboList -> forceGroupNonOptional ( $ group );
178
168
$ beforeCurrentCombo = false ;
179
169
} elseif ($ beforeCurrentCombo && !$ group ->resetsGroupCounter ()) {
180
- $ group -> forceNonOptional ();
181
- $ group-> forceType (
170
+ $ comboList = $ comboList -> forceGroupTypeAndNonOptional (
171
+ $ group,
182
172
$ this ->containsUnmatchedAsNull ($ flags , $ matchesAll ) ? new NullType () : new ConstantStringType ('' ),
183
173
);
184
174
} elseif (
185
175
$ group ->getAlternationId () === $ onlyTopLevelAlternation ->getId ()
186
176
&& !$ this ->containsUnmatchedAsNull ($ flags , $ matchesAll )
187
177
) {
188
- unset( $ comboList[ $ groupId ] );
178
+ $ comboList = $ comboList -> removeGroup ( $ group );
189
179
}
190
180
}
191
181
@@ -199,11 +189,6 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
199
189
);
200
190
201
191
$ combiTypes [] = $ combiType ;
202
-
203
- foreach ($ groupCombo as $ groupId ) {
204
- $ group = $ comboList [$ groupId ];
205
- $ group ->clearOverrides ();
206
- }
207
192
}
208
193
209
194
if (
@@ -223,7 +208,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
223
208
// the general case, which should work in all cases but does not yield the most
224
209
// precise result possible in some cases
225
210
return $ this ->buildArrayType (
226
- $ groupList ,
211
+ $ regexGroupList ,
227
212
$ wasMatched ,
228
213
$ trailingOptionals ,
229
214
$ flags ,
@@ -233,65 +218,10 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
233
218
}
234
219
235
220
/**
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
291
221
* @param list<string> $markVerbs
292
222
*/
293
223
private function buildArrayType (
294
- array $ captureGroups ,
224
+ RegexGroupList $ captureGroups ,
295
225
TrinaryLogic $ wasMatched ,
296
226
int $ trailingOptionals ,
297
227
int $ flags ,
0 commit comments