diff options
author | Andrew Gallant <jamslam@gmail.com> | 2020-05-08 08:09:26 -0400 |
---|---|---|
committer | Andrew Gallant <jamslam@gmail.com> | 2020-05-08 23:24:40 -0400 |
commit | 7ed9a31819aa4f1c1b25f1fa95bdf602232ddbb0 (patch) | |
tree | 70790f5efb9e96b85b00bef4a035abf38f121320 | |
parent | a2e6aec7a4d9382941932245e8854f0ae5703a5e (diff) |
printer: fix --count-matches output
In order to implement --count-matches, we simply re-execute the regex on
the spans reported by the searcher. The spans always correspond to the
lines that participated in the match. This is the correct thing to do,
except when the regex contains look-ahead (or look-behind).
In particular, the look-around permits the regex's match success to
depends on an arbitrary point before or after the lines actually
reported as participating in the match. Since only the matched lines are
reported to the printer, it is possible for subsequent searching on
those lines to fail.
A true fix for this would somehow make the total span available to the
printer. But that seems tricky since it isn't always available. For
PCRE2's case in multiline mode, it is available because we force it to
be so for correctness.
For now, we simply detect this corner case heuristically. If the match
count is zero, then it necessarily means there is some kind of
look-around that isn't matching. So we set the match count to 1. This is
probably incorrect in some cases, although my brain can't quite come up
with a concrete example. Nevertheless, this is strictly better than the
status quo.
Fixes #1573
-rw-r--r-- | CHANGELOG.md | 2 | ||||
-rw-r--r-- | crates/printer/src/summary.rs | 13 | ||||
-rw-r--r-- | tests/regression.rs | 42 |
3 files changed, 57 insertions, 0 deletions
diff --git a/CHANGELOG.md b/CHANGELOG.md index b0438110..3414bbd9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,8 @@ Bug fixes: * [BUG #1537](https://github.com/BurntSushi/ripgrep/issues/1537): Fix match bug caused by inner literal optimization. +* [BUG #1573](https://github.com/BurntSushi/ripgrep/issues/1573): + Fix incorrect `--count-matches` output when using look-around. 12.0.1 (2020-03-29) diff --git a/crates/printer/src/summary.rs b/crates/printer/src/summary.rs index a27fc005..5ce087ef 100644 --- a/crates/printer/src/summary.rs +++ b/crates/printer/src/summary.rs @@ -591,6 +591,19 @@ impl<'p, 's, M: Matcher, W: WriteColor> Sink for SummarySink<'p, 's, M, W> { true }) .map_err(io::Error::error_message)?; + if match_count == 0 { + // It is possible for the match count to be zero when + // look-around is used. Since `SinkMatch` won't necessarily + // contain the look-around in its match span, the search here + // could fail to find anything. + // + // It seems likely that setting match_count=1 here is probably + // wrong in some cases, but I don't think we can do any + // better. (Because this printer cannot assume that subsequent + // contents have been loaded into memory, so we have no way of + // increasing the search span here.) + match_count = 1; + } stats.add_matches(match_count); stats.add_matched_lines(mat.lines().count() as u64); } else if self.summary.config.kind.quit_early() { diff --git a/tests/regression.rs b/tests/regression.rs index 1bfd4f0f..b34fb0ff 100644 --- a/tests/regression.rs +++ b/tests/regression.rs @@ -822,3 +822,45 @@ foo: TaskID int `json:\"taskID\"` "; eqnice!(expected, cmd.arg("TaskID +int").stdout()); }); + +// See: https://github.com/BurntSushi/ripgrep/issues/1573 +// +// Tests that if look-ahead is used, then --count-matches is correct. +rgtest!(r1573, |dir: Dir, mut cmd: TestCommand| { + // Only PCRE2 supports look-ahead. + if !dir.is_pcre2() { + return; + } + + dir.create_bytes("foo", b"\xFF\xFE\x00\x62"); + dir.create( + "foo", + "\ +def A; +def B; +use A; +use B; +", + ); + + // Check that normal --count is correct. + cmd.args(&[ + "--pcre2", + "--multiline", + "--count", + r"(?s)def (\w+);(?=.*use \w+)", + "foo", + ]); + eqnice!("2\n", cmd.stdout()); + + // Now check --count-matches. + let mut cmd = dir.command(); + cmd.args(&[ + "--pcre2", + "--multiline", + "--count-matches", + r"(?s)def (\w+);(?=.*use \w+)", + "foo", + ]); + eqnice!("2\n", cmd.stdout()); +}); |