Skip to content

Commit 641ef3f

Browse files
committed
Add regex.matchPos
A convenience function, which does the slicing and position adjustments for you, to match at a given position within the haystack string.
1 parent 676d76d commit 641ef3f

File tree

3 files changed

+32
-1
lines changed

3 files changed

+32
-1
lines changed

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ The provided Regex type allows 64 'operations' and 8 unique ASCII character sets
1111
Drop the file into your project, or use the Zig build system:
1212

1313
```zig
14-
zig fetch --save "https://github.com/mnemnion/mvzr/archive/refs/tags/v0.1.2.tar.gz"
14+
zig fetch --save "https://github.com/mnemnion/mvzr/archive/refs/tags/v0.2.0.tar.gz"
1515
```
1616

1717
I'll do my best to keep that URL fresh, but it pays to check over here: ➔

build.zig

+3
Original file line numberDiff line numberDiff line change
@@ -12,12 +12,15 @@ pub fn build(b: *std.Build) void {
1212
// set a preferred release mode, allowing the user to decide how to optimize.
1313
const optimize = b.standardOptimizeOption(.{});
1414

15+
const test_filters = b.option([]const []const u8, "test-filter", "Skip tests that do not match any filter") orelse &[0][]const u8{};
16+
1517
// Creates a step for unit testing. This only builds the test executable
1618
// but does not run it.
1719
const lib_unit_tests = b.addTest(.{
1820
.root_source_file = b.path("src/mvzr.zig"),
1921
.target = target,
2022
.optimize = optimize,
23+
.filters = test_filters,
2124
});
2225

2326
const run_lib_unit_tests = b.addRunArtifact(lib_unit_tests);

src/mvzr.zig

+28
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,25 @@ pub fn SizedRegex(ops: comptime_int, char_sets: comptime_int) type {
124124
}
125125
}
126126

127+
/// Match a regex pattern in `haystack` after `pos`. Returns a `Match`
128+
/// if the pattern is found.
129+
pub fn matchPos(regex: *const SizedRegexT, pos: usize, haystack: []const u8) ?Match {
130+
const substack = haystack[pos..];
131+
if (substack.len == 0) return null;
132+
const maybe_matched = regex.matchInternal(substack);
133+
if (maybe_matched) |m| {
134+
const m1 = pos + m[0];
135+
const m2 = pos + m[1];
136+
return Match{
137+
.slice = haystack[m1..m2],
138+
.start = m1,
139+
.end = m2,
140+
};
141+
} else {
142+
return null;
143+
}
144+
}
145+
127146
/// Boolean test if the regex matches in the haystack.
128147
pub fn isMatch(regex: *const SizedRegexT, haystack: []const u8) bool {
129148
const maybe_matched = regex.matchInternal(haystack);
@@ -2119,6 +2138,7 @@ test "workshop" {
21192138
// printRegexString("^[a-zA-Z0-9_!#$%&.-]+@([a-zA-Z0-9.-])+$");
21202139
// try testMatchAll("^[a-zA-Z0-9_!#$%&.-]+@([a-zA-Z0-9.-])+$", "[email protected]");
21212140
//
2141+
try testMatchAll("(a[bc]){3,5}ac", "abacabacac");
21222142
}
21232143

21242144
test "heap allocated regex and match" {
@@ -2167,6 +2187,14 @@ test "iteration" {
21672187
try expectEqual(null, r_iter.next());
21682188
}
21692189

2190+
test "matchPos" {
2191+
const regex = Regex.compile("abcd").?;
2192+
const matched = regex.matchPos(4, "abcdabcd").?;
2193+
try expectEqual(4, matched.start);
2194+
try expectEqualStrings("abcd", matched.slice);
2195+
try expectEqual(8, matched.end);
2196+
}
2197+
21702198
test "comptime regex" {
21712199
const comp_regex = comptime compile("foo+").?;
21722200
const run_match = comp_regex.match("foofoofoo");

0 commit comments

Comments
 (0)