@@ -107,12 +107,17 @@ private function matchPatternType(Type $patternType, ?Type $flagsType, TrinaryLo
107
107
*/
108
108
private function matchRegex (string $ regex , ?int $ flags , TrinaryLogic $ wasMatched , bool $ matchesAll ): ?Type
109
109
{
110
- $ parseResult = $ this ->regexGroupParser ->parseGroups ($ regex );
111
- if ($ parseResult === null ) {
110
+ $ astWalkResult = $ this ->regexGroupParser ->parseGroups ($ regex );
111
+ if ($ astWalkResult === null ) {
112
112
// regex could not be parsed by Hoa/Regex
113
113
return null ;
114
114
}
115
- [$ groupList , $ markVerbs ] = $ parseResult ;
115
+ $ groupList = $ astWalkResult ->getCapturingGroups ();
116
+ $ markVerbs = $ astWalkResult ->getMarkVerbs ();
117
+ $ subjectBaseType = new StringType ();
118
+ if ($ wasMatched ->yes ()) {
119
+ $ subjectBaseType = $ astWalkResult ->getSubjectBaseType ();
120
+ }
116
121
117
122
$ regexGroupList = new RegexGroupList ($ groupList );
118
123
$ trailingOptionals = $ regexGroupList ->countTrailingOptionals ();
@@ -130,6 +135,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
130
135
$ regexGroupList = $ regexGroupList ->forceGroupNonOptional ($ onlyOptionalTopLevelGroup );
131
136
132
137
$ combiType = $ this ->buildArrayType (
138
+ $ subjectBaseType ,
133
139
$ regexGroupList ,
134
140
$ wasMatched ,
135
141
$ trailingOptionals ,
@@ -141,7 +147,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
141
147
if (!$ this ->containsUnmatchedAsNull ($ flags , $ matchesAll )) {
142
148
// positive match has a subject but not any capturing group
143
149
$ combiType = TypeCombinator::union (
144
- new ConstantArrayType ([new ConstantIntegerType (0 )], [$ this ->createSubjectValueType ($ flags , $ matchesAll )], [1 ], [], TrinaryLogic::createYes ()),
150
+ new ConstantArrayType ([new ConstantIntegerType (0 )], [$ this ->createSubjectValueType ($ subjectBaseType , $ flags , $ matchesAll )], [1 ], [], TrinaryLogic::createYes ()),
145
151
$ combiType ,
146
152
);
147
153
}
@@ -180,6 +186,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
180
186
}
181
187
182
188
$ combiType = $ this ->buildArrayType (
189
+ $ subjectBaseType ,
183
190
$ comboList ,
184
191
$ wasMatched ,
185
192
$ trailingOptionals ,
@@ -199,7 +206,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
199
206
)
200
207
) {
201
208
// positive match has a subject but not any capturing group
202
- $ combiTypes [] = new ConstantArrayType ([new ConstantIntegerType (0 )], [$ this ->createSubjectValueType ($ flags , $ matchesAll )], [1 ], [], TrinaryLogic::createYes ());
209
+ $ combiTypes [] = new ConstantArrayType ([new ConstantIntegerType (0 )], [$ this ->createSubjectValueType ($ subjectBaseType , $ flags , $ matchesAll )], [1 ], [], TrinaryLogic::createYes ());
203
210
}
204
211
205
212
return TypeCombinator::union (...$ combiTypes );
@@ -208,6 +215,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
208
215
// the general case, which should work in all cases but does not yield the most
209
216
// precise result possible in some cases
210
217
return $ this ->buildArrayType (
218
+ $ subjectBaseType ,
211
219
$ regexGroupList ,
212
220
$ wasMatched ,
213
221
$ trailingOptionals ,
@@ -221,6 +229,7 @@ private function matchRegex(string $regex, ?int $flags, TrinaryLogic $wasMatched
221
229
* @param list<string> $markVerbs
222
230
*/
223
231
private function buildArrayType (
232
+ Type $ subjectBaseType ,
224
233
RegexGroupList $ captureGroups ,
225
234
TrinaryLogic $ wasMatched ,
226
235
int $ trailingOptionals ,
@@ -234,7 +243,7 @@ private function buildArrayType(
234
243
// first item in matches contains the overall match.
235
244
$ builder ->setOffsetValueType (
236
245
$ this ->getKeyType (0 ),
237
- $ this ->createSubjectValueType ($ flags , $ matchesAll ),
246
+ $ this ->createSubjectValueType ($ subjectBaseType , $ flags , $ matchesAll ),
238
247
$ this ->isSubjectOptional ($ wasMatched , $ matchesAll ),
239
248
);
240
249
@@ -298,13 +307,21 @@ private function isSubjectOptional(TrinaryLogic $wasMatched, bool $matchesAll):
298
307
return !$ wasMatched ->yes ();
299
308
}
300
309
301
- private function createSubjectValueType (int $ flags , bool $ matchesAll ): Type
310
+ /**
311
+ * @param Type $baseType A string type (or string variant) representing the subject of the match
312
+ */
313
+ private function createSubjectValueType (Type $ baseType , int $ flags , bool $ matchesAll ): Type
302
314
{
303
- $ subjectValueType = TypeCombinator::removeNull ($ this ->getValueType (new StringType () , $ flags , $ matchesAll ));
315
+ $ subjectValueType = TypeCombinator::removeNull ($ this ->getValueType ($ baseType , $ flags , $ matchesAll ));
304
316
305
317
if ($ matchesAll ) {
318
+ $ subjectValueType = TypeCombinator::removeNull ($ this ->getValueType (new StringType (), $ flags , $ matchesAll ));
319
+
306
320
if ($ this ->containsPatternOrder ($ flags )) {
307
- $ subjectValueType = TypeCombinator::intersect (new ArrayType (new IntegerType (), $ subjectValueType ), new AccessoryArrayListType ());
321
+ $ subjectValueType = TypeCombinator::intersect (
322
+ new ArrayType (new IntegerType (), $ subjectValueType ),
323
+ new AccessoryArrayListType (),
324
+ );
308
325
}
309
326
}
310
327
0 commit comments