Skip to content

Commit 728bb0e

Browse files
committed
Non-matched optional capture group should be null and offset -1
Fixes #3093
1 parent 31dac28 commit 728bb0e

File tree

2 files changed

+34
-3
lines changed

2 files changed

+34
-3
lines changed

src/builtin.c

Lines changed: 7 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1006,8 +1006,13 @@ static jv f_match(jq_state *jq, jv input, jv regex, jv modifiers, jv testmode) {
10061006
jv captures = jv_array();
10071007
for (int i = 1; i < region->num_regs; ++i) {
10081008
jv cap = jv_object();
1009-
cap = jv_object_set(cap, jv_string("offset"), jv_number(idx));
1010-
cap = jv_object_set(cap, jv_string("string"), jv_string(""));
1009+
if (region->beg[i] == -1) {
1010+
cap = jv_object_set(cap, jv_string("offset"), jv_number(-1));
1011+
cap = jv_object_set(cap, jv_string("string"), jv_null());
1012+
} else {
1013+
cap = jv_object_set(cap, jv_string("offset"), jv_number(idx));
1014+
cap = jv_object_set(cap, jv_string("string"), jv_string(""));
1015+
}
10111016
cap = jv_object_set(cap, jv_string("length"), jv_number(0));
10121017
cap = jv_object_set(cap, jv_string("name"), jv_null());
10131018
captures = jv_array_append(captures, cap);

tests/onig.test

Lines changed: 27 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
# match builtin
22
[match("( )*"; "g")]
33
"abc"
4-
[{"offset":0,"length":0,"string":"","captures":[{"offset":0,"string":"","length":0,"name":null}]},{"offset":1,"length":0,"string":"","captures":[{"offset":1,"string":"","length":0,"name":null}]},{"offset":2,"length":0,"string":"","captures":[{"offset":2,"string":"","length":0,"name":null}]},{"offset":3,"length":0,"string":"","captures":[{"offset":3,"string":"","length":0,"name":null}]}]
4+
[{"offset":0,"length":0,"string":"","captures":[{"offset":-1,"string":null,"length":0,"name":null}]},{"offset":1,"length":0,"string":"","captures":[{"offset":-1,"string":null,"length":0,"name":null}]},{"offset":2,"length":0,"string":"","captures":[{"offset":-1,"string":null,"length":0,"name":null}]},{"offset":3,"length":0,"string":"","captures":[{"offset":-1,"string":null,"length":0,"name":null}]}]
55

66
[match("( )*"; "gn")]
77
"abc"
@@ -37,6 +37,32 @@
3737
"foo bar foo foo foo"
3838
[{"offset": 0, "length": 11, "string": "foo bar foo", "captures":[{"offset": 4, "length": 3, "string": "bar", "name": "bar123"}]},{"offset":12, "length": 8, "string": "foo foo", "captures":[{"offset": -1, "length": 0, "string": null, "name": "bar123"}]}]
3939

40+
# non-matched optional group
41+
"a","b","c" | capture("(?<x>a)?b?")
42+
null
43+
{"x":"a"}
44+
{"x":null}
45+
{"x":null}
46+
47+
"a","b","c" | match("(?<x>a)?b?")
48+
null
49+
{"offset":0,"length":1,"string":"a","captures":[{"offset":0,"length":1,"string":"a","name":"x"}]}
50+
{"offset":0,"length":1,"string":"b","captures":[{"offset":-1,"string":null,"length":0,"name":"x"}]}
51+
{"offset":0,"length":0,"string":"","captures":[{"offset":-1,"string":null,"length":0,"name":"x"}]}
52+
53+
# same as above but allow empty match for group
54+
"a","b","c" | capture("(?<x>a?)?b?")
55+
null
56+
{"x":"a"}
57+
{"x":""}
58+
{"x":""}
59+
60+
"a","b","c" | match("(?<x>a?)?b?")
61+
null
62+
{"offset":0,"length":1,"string":"a","captures":[{"offset":0,"length":1,"string":"a","name":"x"}]}
63+
{"offset":0,"length":1,"string":"b","captures":[{"offset":0,"string":"","length":0,"name":"x"}]}
64+
{"offset":0,"length":0,"string":"","captures":[{"offset":0,"string":"","length":0,"name":"x"}]}
65+
4066
#test builtin
4167
[test("( )*"; "gn")]
4268
"abc"

0 commit comments

Comments
 (0)