5 releases

0.1.4 Mar 15, 2024
0.1.3 Jan 31, 2024
0.1.2 Jan 27, 2024
0.1.1 Jan 25, 2024
0.1.0 Jan 25, 2024

#137 in Text processing

Download history 1078/week @ 2024-08-19 997/week @ 2024-08-26 1035/week @ 2024-09-02 1004/week @ 2024-09-09 1172/week @ 2024-09-16 1041/week @ 2024-09-23 1150/week @ 2024-09-30 1097/week @ 2024-10-07 1132/week @ 2024-10-14 839/week @ 2024-10-21 886/week @ 2024-10-28 788/week @ 2024-11-04 909/week @ 2024-11-11 1219/week @ 2024-11-18 1025/week @ 2024-11-25 1011/week @ 2024-12-02

4,239 downloads per month
Used in tracexec

MIT/Apache

645KB
7.5K SLoC

regex-cursor

This crate provides routines for searching discontiguous strings for matches of a [regular expression] (aka "regex"). It is based on [regex-automata] and most of the code is adapted from the various crates in the regex repository.

It is intended as a prototype for upstream support for "streaming regex". The cursor based API in this crate is very similar to the API already exposed by regex/regex-automata. To that end a generic Cursor trait is provided that collections can implement.

A sketch of the cursor API is shown below. The string is yielded in multiple byte chunks. Calling advance moves the cursor to the next chunk. Calling backtrack moves the cursor a chunk back. Backtracking is required by this create. That makes it unsuitable for searching fully unbuffered streams like bytes send over a TCP connection.

pub trait Cursor {
    fn chunk(&self) -> &[u8] { .. }
    fn advance(&mut self) -> bool { .. }
    fn bracktrack(&mut self) -> bool { .. }
}

Working on this crate showed me that regex backtracks a lot more than expected with most functionality fundamentally requiring backtracking. For network usecases that do not buffer their input the primary usecase would likely be detecting a match (without necessarily requiring the matched byte range). Such usecases can be covered by manually feeding bytes into the hybrid and DFA engines from the regex-automata crate. This approach also has the advantage of allowing the caller to pause the match (async) while waiting for more data allowing the caller to drive the search instead of the engine itself.

The only part of this crate that could be applied to the fully streaming case is the streaming PikeVM implementation. However, there are some limitations:

  • only a single search can be run since the PikeVM may look ahead multiple bytes to disambiguate alternative matches
  • Prefilters longer than one byte can not work
  • utf-8 mode can not be supported (empty matches may occur between unicode boundaries)

Currently, the PikeVM implementation is not written with this use case in mind and may call backtrack unnecessarily, but that could be addressed in the future, but especially the first point is very limiting. The pikevm also does not allow the user to drive the search and would block on network calls for example (no async).

Dependencies

~2.5–3.5MB
~68K SLoC