mirror of
https://github.com/BurntSushi/ripgrep.git
synced 2025-01-19 05:49:14 +02:00
8cb5833ef9
We use the new AppSettings::AllArgsOverrideSelf to permit all flags to be specified multiple times. This removes the need for our previous work-around where we would enable `multiple` for every flag and then just extract the last value when consuming clap's matches. We also add a couple regression tests that ensure repeated switches and flags work as expected.
2073 lines
65 KiB
Rust
2073 lines
65 KiB
Rust
/*!
|
|
This module contains *integration* tests. Their purpose is to test the CLI
|
|
interface. Namely, that passing a flag does what it says on the tin.
|
|
|
|
Tests for more fine grained behavior (like the search or the globber) should be
|
|
unit tests in their respective modules.
|
|
*/
|
|
|
|
#![allow(dead_code, unused_imports)]
|
|
|
|
use std::process::Command;
|
|
|
|
use workdir::WorkDir;
|
|
|
|
mod hay;
|
|
mod workdir;
|
|
|
|
macro_rules! sherlock {
|
|
($name:ident, $fun:expr) => {
|
|
sherlock!($name, "Sherlock", $fun);
|
|
};
|
|
($name:ident, $query:expr, $fun:expr) => {
|
|
sherlock!($name, $query, "sherlock", $fun);
|
|
};
|
|
($name:ident, $query:expr, $path:expr, $fun:expr) => {
|
|
#[test]
|
|
fn $name() {
|
|
let wd = WorkDir::new(stringify!($name));
|
|
wd.create("sherlock", hay::SHERLOCK);
|
|
let mut cmd = wd.command();
|
|
cmd.arg($query).arg($path);
|
|
$fun(wd, cmd);
|
|
}
|
|
};
|
|
}
|
|
|
|
macro_rules! clean {
|
|
($name:ident, $query:expr, $path:expr, $fun:expr) => {
|
|
#[test]
|
|
fn $name() {
|
|
let wd = WorkDir::new(stringify!($name));
|
|
let mut cmd = wd.command();
|
|
cmd.arg($query).arg($path);
|
|
$fun(wd, cmd);
|
|
}
|
|
};
|
|
}
|
|
|
|
fn path(unix: &str) -> String {
|
|
if cfg!(windows) {
|
|
unix.replace("/", "\\")
|
|
} else {
|
|
unix.to_string()
|
|
}
|
|
}
|
|
|
|
fn paths(unix: &[&str]) -> Vec<String> {
|
|
let mut xs: Vec<_> = unix.iter().map(|s| path(s)).collect();
|
|
xs.sort();
|
|
xs
|
|
}
|
|
|
|
fn paths_from_stdout(stdout: String) -> Vec<String> {
|
|
let mut paths: Vec<_> = stdout.lines().map(|s| {
|
|
s.split(':').next().unwrap().to_string()
|
|
}).collect();
|
|
paths.sort();
|
|
paths
|
|
}
|
|
|
|
fn sort_lines(lines: &str) -> String {
|
|
let mut lines: Vec<String> =
|
|
lines.trim().lines().map(|s| s.to_owned()).collect();
|
|
lines.sort();
|
|
format!("{}\n", lines.join("\n"))
|
|
}
|
|
|
|
fn cmd_exists(name: &str) -> bool {
|
|
Command::new(name).arg("--help").output().is_ok()
|
|
}
|
|
|
|
sherlock!(single_file, |wd: WorkDir, mut cmd| {
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(dir, "Sherlock", ".", |wd: WorkDir, mut cmd| {
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(line_numbers, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-n");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
3:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(line_number_width, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-n");
|
|
cmd.arg("--line-number-width").arg("2");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = " 1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
3:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(line_number_width_padding_character_error, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-n");
|
|
cmd.arg("--line-number-width").arg("02");
|
|
wd.assert_non_empty_stderr(&mut cmd);
|
|
});
|
|
|
|
sherlock!(columns, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--column");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
1:57:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
3:49:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(with_filename, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-H");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(with_heading, |wd: WorkDir, mut cmd: Command| {
|
|
// This forces the issue since --with-filename is disabled by default
|
|
// when searching one file.
|
|
cmd.arg("--with-filename").arg("--heading");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(with_heading_default, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// Search two or more and get --with-filename enabled by default.
|
|
// Use -j1 to get deterministic results.
|
|
wd.create("foo", "Sherlock Holmes lives on Baker Street.");
|
|
cmd.arg("-j1").arg("--heading");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected1 = "\
|
|
foo
|
|
Sherlock Holmes lives on Baker Street.
|
|
|
|
sherlock
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
let expected2 = "\
|
|
sherlock
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
|
|
foo
|
|
Sherlock Holmes lives on Baker Street.
|
|
";
|
|
if lines != expected1 {
|
|
assert_eq!(lines, expected2);
|
|
} else {
|
|
assert_eq!(lines, expected1);
|
|
}
|
|
});
|
|
|
|
sherlock!(inverted, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-v");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
Holmeses, success in the province of detective work must always
|
|
can extract a clew from a wisp of straw or a flake of cigar ash;
|
|
but Doctor Watson has to have it taken out for him and dusted,
|
|
and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(inverted_line_numbers, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-n").arg("-v");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
2:Holmeses, success in the province of detective work must always
|
|
4:can extract a clew from a wisp of straw or a flake of cigar ash;
|
|
5:but Doctor Watson has to have it taken out for him and dusted,
|
|
6:and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(case_insensitive, "sherlock", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-i");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(word, "as", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-w");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(line, "Watson|and exhibited clearly, with a label attached.",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-x");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(literal, "()", "file", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file", "blib\n()\nblab\n");
|
|
cmd.arg("-F");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "()\n");
|
|
});
|
|
|
|
sherlock!(quiet, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-q");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert!(lines.is_empty());
|
|
});
|
|
|
|
sherlock!(replace, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-r").arg("FooBar");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the FooBar
|
|
be, to a very large extent, the result of luck. FooBar Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(replace_groups, "([A-Z][a-z]+) ([A-Z][a-z]+)",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-r").arg("$2, $1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Watsons, Doctor of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Holmes, Sherlock
|
|
but Watson, Doctor has to have it taken out for him and dusted,
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(replace_named_groups, "(?P<first>[A-Z][a-z]+) (?P<last>[A-Z][a-z]+)",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-r").arg("$last, $first");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Watsons, Doctor of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Holmes, Sherlock
|
|
but Watson, Doctor has to have it taken out for him and dusted,
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(replace_with_only_matching, "of (\\w+)",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-o").arg("-r").arg("$1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
this
|
|
detective
|
|
luck
|
|
straw
|
|
cigar
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(file_types, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
cmd.arg("-t").arg("rust");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.rs:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(file_types_all, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
cmd.arg("-t").arg("all");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.py:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(file_types_negate, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
cmd.arg("-T").arg("rust");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.py:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(file_types_negate_all, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
cmd.arg("-T").arg("all");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
");
|
|
});
|
|
|
|
sherlock!(file_type_clear, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
cmd.arg("--type-clear").arg("rust").arg("-t").arg("rust");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(file_type_add, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
wd.create("file.wat", "Sherlock");
|
|
cmd.arg("--type-add").arg("wat:*.wat").arg("-t").arg("wat");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.wat:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(file_type_add_compose, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
wd.create("file.wat", "Sherlock");
|
|
cmd.arg("--type-add").arg("wat:*.wat");
|
|
cmd.arg("--type-add").arg("combo:include:wat,py").arg("-t").arg("combo");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
println!("{}", lines);
|
|
assert_eq!(sort_lines(&lines), "file.py:Sherlock\nfile.wat:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(glob, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
cmd.arg("-g").arg("*.rs");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.rs:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(glob_negate, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create("file.py", "Sherlock");
|
|
wd.create("file.rs", "Sherlock");
|
|
cmd.arg("-g").arg("!*.rs");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.py:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(iglob, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.HTML", "Sherlock");
|
|
cmd.arg("--iglob").arg("*.html");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.HTML:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(csglob, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file1.HTML", "Sherlock");
|
|
wd.create("file2.html", "Sherlock");
|
|
cmd.arg("--glob").arg("*.html");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file2.html:Sherlock\n");
|
|
});
|
|
|
|
sherlock!(count, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--count");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "sherlock:2\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(files_with_matches, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--files-with-matches");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "sherlock\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(files_without_matches, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "foo");
|
|
cmd.arg("--files-without-match");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "file.py\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(after_context, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-A").arg("1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
Holmeses, success in the province of detective work must always
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
can extract a clew from a wisp of straw or a flake of cigar ash;
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(after_context_line_numbers, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-A").arg("1").arg("-n");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
2-Holmeses, success in the province of detective work must always
|
|
3:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
4-can extract a clew from a wisp of straw or a flake of cigar ash;
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(before_context, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-B").arg("1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
Holmeses, success in the province of detective work must always
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(before_context_line_numbers, |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-B").arg("1").arg("-n");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
2-Holmeses, success in the province of detective work must always
|
|
3:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(context, "world|attached", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-C").arg("1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
Holmeses, success in the province of detective work must always
|
|
--
|
|
but Doctor Watson has to have it taken out for him and dusted,
|
|
and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(context_line_numbers, "world|attached",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-C").arg("1").arg("-n");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
2-Holmeses, success in the province of detective work must always
|
|
--
|
|
5-but Doctor Watson has to have it taken out for him and dusted,
|
|
6:and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(max_filesize_parse_error_length, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--max-filesize").arg("44444444444444444444");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(max_filesize_parse_error_suffix, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--max-filesize").arg("45k");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(max_filesize_parse_no_suffix, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_size("foo", 40);
|
|
wd.create_size("bar", 60);
|
|
|
|
cmd.arg("--max-filesize").arg("50").arg("--files");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
foo
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(max_filesize_parse_k_suffix, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_size("foo", 3048);
|
|
wd.create_size("bar", 4100);
|
|
|
|
cmd.arg("--max-filesize").arg("4K").arg("--files");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
foo
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(max_filesize_parse_m_suffix, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_size("foo", 1000000);
|
|
wd.create_size("bar", 1400000);
|
|
|
|
cmd.arg("--max-filesize").arg("1M").arg("--files");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
foo
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(max_filesize_suffix_overflow, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_size("foo", 1000000);
|
|
|
|
// 2^35 * 2^30 would otherwise overflow
|
|
cmd.arg("--max-filesize").arg("34359738368G").arg("--files");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(ignore_hidden, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create(".sherlock", hay::SHERLOCK);
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(no_ignore_hidden, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create(".sherlock", hay::SHERLOCK);
|
|
|
|
cmd.arg("--hidden");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
.sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
.sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(ignore_git, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "sherlock\n");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(ignore_generic, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".ignore", "sherlock\n");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(ignore_ripgrep, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".rgignore", "sherlock\n");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(no_ignore, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "sherlock\n");
|
|
cmd.arg("--no-ignore");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(ignore_git_parent, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create(".gitignore", "sherlock\n");
|
|
wd.create_dir(".git");
|
|
wd.create_dir("foo");
|
|
wd.create("foo/sherlock", hay::SHERLOCK);
|
|
// Even though we search in foo/, which has no .gitignore, ripgrep will
|
|
// search parent directories and respect the gitignore files found.
|
|
cmd.current_dir(wd.path().join("foo"));
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(ignore_git_parent_stop, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// This tests that searching parent directories for .gitignore files stops
|
|
// after it sees a .git directory. To test this, we create this directory
|
|
// hierarchy:
|
|
//
|
|
// .gitignore (contains `sherlock`)
|
|
// foo/
|
|
// .git
|
|
// bar/
|
|
// sherlock
|
|
//
|
|
// And we perform the search inside `foo/bar/`. ripgrep will stop looking
|
|
// for .gitignore files after it sees `foo/.git/`, and therefore not
|
|
// respect the top-level `.gitignore` containing `sherlock`.
|
|
wd.remove("sherlock");
|
|
wd.create(".gitignore", "sherlock\n");
|
|
wd.create_dir("foo");
|
|
wd.create_dir("foo/.git");
|
|
wd.create_dir("foo/bar");
|
|
wd.create("foo/bar/sherlock", hay::SHERLOCK);
|
|
cmd.current_dir(wd.path().join("foo").join("bar"));
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(ignore_ripgrep_parent_no_stop, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// This is like the `ignore_git_parent_stop` test, except it checks that
|
|
// ripgrep *doesn't* stop checking for .rgignore files.
|
|
wd.remove("sherlock");
|
|
wd.create(".rgignore", "sherlock\n");
|
|
wd.create_dir("foo");
|
|
wd.create_dir("foo/.git");
|
|
wd.create_dir("foo/bar");
|
|
wd.create("foo/bar/sherlock", hay::SHERLOCK);
|
|
cmd.current_dir(wd.path().join("foo").join("bar"));
|
|
// The top-level .rgignore applies.
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
sherlock!(no_parent_ignore_git, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// Set up a directory hierarchy like this:
|
|
//
|
|
// .gitignore
|
|
// foo/
|
|
// .gitignore
|
|
// sherlock
|
|
// watson
|
|
//
|
|
// Where `.gitignore` contains `sherlock` and `foo/.gitignore` contains
|
|
// `watson`.
|
|
//
|
|
// Now *do the search* from the foo directory. By default, ripgrep will
|
|
// search parent directories for .gitignore files. The --no-ignore-parent
|
|
// flag should prevent that. At the same time, the `foo/.gitignore` file
|
|
// will still be respected (since the search is happening in `foo/`).
|
|
//
|
|
// In other words, we should only see results from `sherlock`, not from
|
|
// `watson`.
|
|
wd.remove("sherlock");
|
|
wd.create(".gitignore", "sherlock\n");
|
|
wd.create_dir("foo");
|
|
wd.create("foo/.gitignore", "watson\n");
|
|
wd.create("foo/sherlock", hay::SHERLOCK);
|
|
wd.create("foo/watson", hay::SHERLOCK);
|
|
cmd.current_dir(wd.path().join("foo"));
|
|
cmd.arg("--no-ignore-parent");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
#[cfg(not(windows))]
|
|
sherlock!(symlink_nofollow, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_dir("foo");
|
|
wd.create_dir("foo/bar");
|
|
wd.link_dir("foo/baz", "foo/bar/baz");
|
|
wd.create_dir("foo/baz");
|
|
wd.create("foo/baz/sherlock", hay::SHERLOCK);
|
|
cmd.current_dir(wd.path().join("foo/bar"));
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
#[cfg(not(windows))]
|
|
sherlock!(symlink_follow, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_dir("foo");
|
|
wd.create_dir("foo/bar");
|
|
wd.create_dir("foo/baz");
|
|
wd.create("foo/baz/sherlock", hay::SHERLOCK);
|
|
wd.link_dir("foo/baz", "foo/bar/baz");
|
|
cmd.arg("-L");
|
|
cmd.current_dir(wd.path().join("foo/bar"));
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
baz/sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
baz/sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, path(expected));
|
|
});
|
|
|
|
sherlock!(unrestricted1, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "sherlock\n");
|
|
cmd.arg("-u");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(unrestricted2, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create(".sherlock", hay::SHERLOCK);
|
|
cmd.arg("-uu");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
.sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
.sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(unrestricted3, "foo", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file", "foo\x00bar\nfoo\x00baz\n");
|
|
cmd.arg("-uuu");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file:foo\x00bar\nfile:foo\x00baz\n");
|
|
});
|
|
|
|
sherlock!(vimgrep, "Sherlock|Watson", ".", |wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--vimgrep");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:1:16:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:1:57:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:3:49:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
sherlock:5:12:but Doctor Watson has to have it taken out for him and dusted,
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/16
|
|
clean!(regression_16, "xyz", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "ghi/");
|
|
wd.create_dir("ghi");
|
|
wd.create_dir("def/ghi");
|
|
wd.create("ghi/toplevel.txt", "xyz");
|
|
wd.create("def/ghi/subdir.txt", "xyz");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/25
|
|
clean!(regression_25, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "/llvm/");
|
|
wd.create_dir("src/llvm");
|
|
wd.create("src/llvm/foo", "test");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = path("src/llvm/foo:test\n");
|
|
assert_eq!(lines, expected);
|
|
|
|
cmd.current_dir(wd.path().join("src"));
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = path("llvm/foo:test\n");
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/30
|
|
clean!(regression_30, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
if cfg!(windows) {
|
|
wd.create(".gitignore", "vendor/**\n!vendor\\manifest");
|
|
} else {
|
|
wd.create(".gitignore", "vendor/**\n!vendor/manifest");
|
|
}
|
|
wd.create_dir("vendor");
|
|
wd.create("vendor/manifest", "test");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = path("vendor/manifest:test\n");
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/49
|
|
clean!(regression_49, "xyz", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "foo/bar");
|
|
wd.create_dir("test/foo/bar");
|
|
wd.create("test/foo/bar/baz", "test");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/50
|
|
clean!(regression_50, "xyz", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "XXX/YYY/");
|
|
wd.create_dir("abc/def/XXX/YYY");
|
|
wd.create_dir("ghi/XXX/YYY");
|
|
wd.create("abc/def/XXX/YYY/bar", "test");
|
|
wd.create("ghi/XXX/YYY/bar", "test");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/65
|
|
clean!(regression_65, "xyz", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "a/");
|
|
wd.create_dir("a");
|
|
wd.create("a/foo", "xyz");
|
|
wd.create("a/bar", "xyz");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/67
|
|
clean!(regression_67, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "/*\n!/dir");
|
|
wd.create_dir("dir");
|
|
wd.create_dir("foo");
|
|
wd.create("foo/bar", "test");
|
|
wd.create("dir/bar", "test");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, path("dir/bar:test\n"));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/87
|
|
clean!(regression_87, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "foo\n**no-vcs**");
|
|
wd.create("foo", "test");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/90
|
|
clean!(regression_90, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "!.foo");
|
|
wd.create(".foo", "test");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, ".foo:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/93
|
|
clean!(regression_93, r"(\d{1,3}\.){3}\d{1,3}", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "192.168.1.1");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:192.168.1.1\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/99
|
|
clean!(regression_99, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo1", "test");
|
|
wd.create("foo2", "zzz");
|
|
wd.create("bar", "test");
|
|
cmd.arg("-j1").arg("--heading");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(sort_lines(&lines), sort_lines("bar\ntest\n\nfoo1\ntest\n"));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/105
|
|
clean!(regression_105_part1, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "zztest");
|
|
cmd.arg("--vimgrep");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:1:3:zztest\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/105
|
|
clean!(regression_105_part2, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "zztest");
|
|
cmd.arg("--column");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:1:3:zztest\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/127
|
|
clean!(regression_127, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
// Set up a directory hierarchy like this:
|
|
//
|
|
// .gitignore
|
|
// foo/
|
|
// sherlock
|
|
// watson
|
|
//
|
|
// Where `.gitignore` contains `foo/sherlock`.
|
|
//
|
|
// ripgrep should ignore 'foo/sherlock' giving us results only from
|
|
// 'foo/watson' but on Windows ripgrep will include both 'foo/sherlock' and
|
|
// 'foo/watson' in the search results.
|
|
wd.create(".gitignore", "foo/sherlock\n");
|
|
wd.create_dir("foo");
|
|
wd.create("foo/sherlock", hay::SHERLOCK);
|
|
wd.create("foo/watson", hay::SHERLOCK);
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = format!("\
|
|
{path}:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
{path}:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
", path=path("foo/watson"));
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/128
|
|
clean!(regression_128, "x", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_bytes("foo", b"01234567\x0b\n\x0b\n\x0b\n\x0b\nx");
|
|
cmd.arg("-n");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:5:x\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/131
|
|
//
|
|
// TODO(burntsushi): Darwin doesn't like this test for some reason.
|
|
#[cfg(not(target_os = "macos"))]
|
|
clean!(regression_131, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "TopÑapa");
|
|
wd.create("TopÑapa", "test");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/137
|
|
//
|
|
// TODO(burntsushi): Figure out why Windows gives "access denied" errors
|
|
// when trying to create a file symlink. For now, disable test on Windows.
|
|
#[cfg(not(windows))]
|
|
sherlock!(regression_137, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.link_file("sherlock", "sym1");
|
|
wd.link_file("sherlock", "sym2");
|
|
cmd.arg("sym1");
|
|
cmd.arg("sym2");
|
|
cmd.arg("-j1");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
sym1:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sym1:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
sym2:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sym2:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, path(expected));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/156
|
|
clean!(
|
|
regression_156,
|
|
r#"#(?:parse|include)\s*\(\s*(?:"|')[./A-Za-z_-]+(?:"|')"#,
|
|
"testcase.txt",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
const TESTCASE: &'static str = r#"#parse('widgets/foo_bar_macros.vm')
|
|
#parse ( 'widgets/mobile/foo_bar_macros.vm' )
|
|
#parse ("widgets/foobarhiddenformfields.vm")
|
|
#parse ( "widgets/foo_bar_legal.vm" )
|
|
#include( 'widgets/foo_bar_tips.vm' )
|
|
#include('widgets/mobile/foo_bar_macros.vm')
|
|
#include ("widgets/mobile/foo_bar_resetpw.vm")
|
|
#parse('widgets/foo-bar-macros.vm')
|
|
#parse ( 'widgets/mobile/foo-bar-macros.vm' )
|
|
#parse ("widgets/foo-bar-hiddenformfields.vm")
|
|
#parse ( "widgets/foo-bar-legal.vm" )
|
|
#include( 'widgets/foo-bar-tips.vm' )
|
|
#include('widgets/mobile/foo-bar-macros.vm')
|
|
#include ("widgets/mobile/foo-bar-resetpw.vm")
|
|
"#;
|
|
wd.create("testcase.txt", TESTCASE);
|
|
cmd.arg("-N");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, TESTCASE);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/184
|
|
clean!(regression_184, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", ".*");
|
|
wd.create_dir("foo/bar");
|
|
wd.create("foo/bar/baz", "test");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, format!("{}:test\n", path("foo/bar/baz")));
|
|
|
|
cmd.current_dir(wd.path().join("./foo/bar"));
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "baz:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/199
|
|
clean!(regression_199, r"\btest\b", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "tEsT");
|
|
cmd.arg("--smart-case");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:tEsT\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/206
|
|
clean!(regression_206, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("foo");
|
|
wd.create("foo/bar.txt", "test");
|
|
cmd.arg("-g").arg("*.txt");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, format!("{}:test\n", path("foo/bar.txt")));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/210
|
|
#[cfg(unix)]
|
|
#[test]
|
|
fn regression_210() {
|
|
use std::ffi::OsStr;
|
|
use std::os::unix::ffi::OsStrExt;
|
|
|
|
let badutf8 = OsStr::from_bytes(&b"foo\xffbar"[..]);
|
|
|
|
let wd = WorkDir::new("regression_210");
|
|
// APFS does not support creating files with invalid UTF-8 bytes.
|
|
// https://github.com/BurntSushi/ripgrep/issues/559
|
|
if wd.try_create(badutf8, "test").is_ok() {
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-H").arg("test").arg(badutf8);
|
|
|
|
let out = wd.output(&mut cmd);
|
|
assert_eq!(out.stdout, b"foo\xffbar:test\n".to_vec());
|
|
}
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/228
|
|
clean!(regression_228, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("foo");
|
|
cmd.arg("--ignore-file").arg("foo");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/229
|
|
clean!(regression_229, "[E]conomie", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "economie");
|
|
cmd.arg("-S");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/251
|
|
clean!(regression_251, "привет", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "привет\nПривет\nПрИвЕт");
|
|
cmd.arg("-i");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:привет\nfoo:Привет\nfoo:ПрИвЕт\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/256
|
|
#[cfg(not(windows))]
|
|
clean!(regression_256, "test", "foo", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("bar");
|
|
wd.create("bar/baz", "test");
|
|
wd.link_dir("bar", "foo");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo/baz:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/256
|
|
#[cfg(not(windows))]
|
|
clean!(regression_256_j1, "test", "foo", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("bar");
|
|
wd.create("bar/baz", "test");
|
|
wd.link_dir("bar", "foo");
|
|
cmd.arg("-j1");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo/baz:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/279
|
|
clean!(regression_279, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test");
|
|
cmd.arg("-q");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/405
|
|
clean!(regression_405, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("foo/bar");
|
|
wd.create_dir("bar/foo");
|
|
wd.create("foo/bar/file1.txt", "test");
|
|
wd.create("bar/foo/file2.txt", "test");
|
|
cmd.arg("-g").arg("!/foo/**");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, format!("{}:test\n", path("bar/foo/file2.txt")));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/428
|
|
#[cfg(not(windows))]
|
|
clean!(regression_428_color_context_path, "foo", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("sherlock", "foo\nbar");
|
|
cmd.arg("-A1").arg("-H").arg("--no-heading").arg("-N")
|
|
.arg("--colors=match:none").arg("--color=always");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = format!(
|
|
"{colored_path}:foo\n{colored_path}-bar\n",
|
|
colored_path=format!(
|
|
"\x1b\x5b\x30\x6d\x1b\x5b\x33\x35\x6d{path}\x1b\x5b\x30\x6d",
|
|
path=path("sherlock")));
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/428
|
|
clean!(regression_428_unrecognized_style, "Sherlok", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--colors=match:style:");
|
|
wd.assert_err(&mut cmd);
|
|
|
|
let output = cmd.output().unwrap();
|
|
let err = String::from_utf8_lossy(&output.stderr);
|
|
let expected = "\
|
|
Unrecognized style attribute ''. Choose from: nobold, bold, nointense, intense.
|
|
";
|
|
assert_eq!(err, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/493
|
|
clean!(regression_493, " 're ", "input.txt", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("input.txt", "peshwaship 're seminomata");
|
|
cmd.arg("-o").arg("-w");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, " 're \n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/553
|
|
sherlock!(regression_553_switch, "sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-i");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
|
|
// This repeats the `-i` flag.
|
|
cmd.arg("-i");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(regression_553_flag, "world|attached",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-C").arg("1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
Holmeses, success in the province of detective work must always
|
|
--
|
|
but Doctor Watson has to have it taken out for him and dusted,
|
|
and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
|
|
cmd.arg("-C").arg("0");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
and exhibited clearly, with a label attached.
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/599
|
|
clean!(regression_599, "^$", "input.txt", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("input.txt", "\n\ntest\n");
|
|
cmd.args(&[
|
|
"--color", "ansi",
|
|
"--colors", "path:none",
|
|
"--colors", "line:none",
|
|
"--colors", "match:fg:red",
|
|
"--colors", "match:style:nobold",
|
|
"--line-number",
|
|
]);
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
// Technically, the expected output should only be two lines, but:
|
|
// https://github.com/BurntSushi/ripgrep/issues/441
|
|
let expected = "\
|
|
[0m1[0m:[0m[31m[0m
|
|
[0m2[0m:[0m[31m[0m
|
|
[0m4[0m:
|
|
";
|
|
assert_eq!(expected, lines);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
clean!(feature_1_sjis, "Шерлок Холмс", ".", |wd: WorkDir, mut cmd: Command| {
|
|
let sherlock =
|
|
b"\x84Y\x84u\x84\x82\x84|\x84\x80\x84{ \x84V\x84\x80\x84|\x84}\x84\x83";
|
|
wd.create_bytes("foo", &sherlock[..]);
|
|
cmd.arg("-Esjis");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:Шерлок Холмс\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
clean!(feature_1_utf16_auto, "Шерлок Холмс", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
let sherlock =
|
|
b"\xff\xfe(\x045\x04@\x04;\x04>\x04:\x04 \x00%\x04>\x04;\x04<\x04A\x04";
|
|
wd.create_bytes("foo", &sherlock[..]);
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:Шерлок Холмс\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
clean!(feature_1_utf16_explicit, "Шерлок Холмс", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
let sherlock =
|
|
b"\xff\xfe(\x045\x04@\x04;\x04>\x04:\x04 \x00%\x04>\x04;\x04<\x04A\x04";
|
|
wd.create_bytes("foo", &sherlock[..]);
|
|
cmd.arg("-Eutf-16le");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:Шерлок Холмс\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
clean!(feature_1_eucjp, "Шерлок Холмс", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
let sherlock =
|
|
b"\xa7\xba\xa7\xd6\xa7\xe2\xa7\xdd\xa7\xe0\xa7\xdc \xa7\xb7\xa7\xe0\xa7\xdd\xa7\xde\xa7\xe3";
|
|
wd.create_bytes("foo", &sherlock[..]);
|
|
cmd.arg("-Eeuc-jp");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:Шерлок Холмс\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
sherlock!(feature_1_unknown_encoding, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-Efoobar");
|
|
wd.assert_non_empty_stderr(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/1
|
|
// Specific: https://github.com/BurntSushi/ripgrep/pull/398/files#r111109265
|
|
sherlock!(feature_1_replacement_encoding, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-Ecsiso2022kr");
|
|
wd.assert_non_empty_stderr(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/7
|
|
sherlock!(feature_7, "-fpat", "sherlock", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("pat", "Sherlock\nHolmes");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
Holmeses, success in the province of detective work must always
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/7
|
|
sherlock!(feature_7_dash, "-f-", ".", |wd: WorkDir, mut cmd: Command| {
|
|
let output = wd.pipe(&mut cmd, "Sherlock");
|
|
let lines = String::from_utf8_lossy(&output.stdout);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/20
|
|
sherlock!(feature_20_no_filename, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--no-filename");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/34
|
|
sherlock!(feature_34_only_matching, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--only-matching");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:Sherlock
|
|
sherlock:Sherlock
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/34
|
|
sherlock!(feature_34_only_matching_line_column, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--only-matching").arg("--column").arg("--line-number");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:1:57:Sherlock
|
|
sherlock:3:49:Sherlock
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/45
|
|
sherlock!(feature_45_relative_cwd, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".not-an-ignore", "foo\n/bar");
|
|
wd.create_dir("bar");
|
|
wd.create_dir("baz/bar");
|
|
wd.create_dir("baz/baz/bar");
|
|
wd.create("bar/test", "test");
|
|
wd.create("baz/bar/test", "test");
|
|
wd.create("baz/baz/bar/test", "test");
|
|
wd.create("baz/foo", "test");
|
|
wd.create("baz/test", "test");
|
|
wd.create("foo", "test");
|
|
wd.create("test", "test");
|
|
|
|
// First, get a baseline without applying ignore rules.
|
|
let lines = paths_from_stdout(wd.stdout(&mut cmd));
|
|
assert_eq!(lines, paths(&[
|
|
"bar/test", "baz/bar/test", "baz/baz/bar/test", "baz/foo",
|
|
"baz/test", "foo", "test",
|
|
]));
|
|
|
|
// Now try again with the ignore file activated.
|
|
cmd.arg("--ignore-file").arg(".not-an-ignore");
|
|
let lines = paths_from_stdout(wd.stdout(&mut cmd));
|
|
assert_eq!(lines, paths(&[
|
|
"baz/bar/test", "baz/baz/bar/test", "baz/test", "test",
|
|
]));
|
|
|
|
// Now do it again, but inside the baz directory.
|
|
// Since the ignore file is interpreted relative to the CWD, this will
|
|
// cause the /bar anchored pattern to filter out baz/bar, which is a
|
|
// subtle difference between true parent ignore files and manually
|
|
// specified ignore files.
|
|
let mut cmd = wd.command();
|
|
cmd.arg("test").arg(".").arg("--ignore-file").arg("../.not-an-ignore");
|
|
cmd.current_dir(wd.path().join("baz"));
|
|
let lines = paths_from_stdout(wd.stdout(&mut cmd));
|
|
assert_eq!(lines, paths(&["baz/bar/test", "test"]));
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/45
|
|
sherlock!(feature_45_precedence_with_others, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".not-an-ignore", "*.log");
|
|
wd.create(".ignore", "!imp.log");
|
|
wd.create("imp.log", "test");
|
|
wd.create("wat.log", "test");
|
|
|
|
cmd.arg("--ignore-file").arg(".not-an-ignore");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "imp.log:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/45
|
|
sherlock!(feature_45_precedence_internal, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".not-an-ignore1", "*.log");
|
|
wd.create(".not-an-ignore2", "!imp.log");
|
|
wd.create("imp.log", "test");
|
|
wd.create("wat.log", "test");
|
|
|
|
cmd.arg("--ignore-file").arg(".not-an-ignore1");
|
|
cmd.arg("--ignore-file").arg(".not-an-ignore2");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "imp.log:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/68
|
|
clean!(feature_68_no_ignore_vcs, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create(".gitignore", "foo");
|
|
wd.create(".ignore", "bar");
|
|
wd.create("foo", "test");
|
|
wd.create("bar", "test");
|
|
cmd.arg("--no-ignore-vcs");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/70
|
|
sherlock!(feature_70_smart_case, "sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--smart-case");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock:For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock:be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
|
sherlock!(feature_89_files_with_matches, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--null").arg("--files-with-matches");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "sherlock\x00");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
|
sherlock!(feature_89_files_without_matches, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("file.py", "foo");
|
|
cmd.arg("--null").arg("--files-without-match");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "file.py\x00");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
|
sherlock!(feature_89_count, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--null").arg("--count");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "sherlock\x002\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
|
sherlock!(feature_89_files, "NADA", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--null").arg("--files");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "sherlock\x00");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
|
sherlock!(feature_89_match, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--null").arg("-C1");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
sherlock\x00For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
sherlock\x00Holmeses, success in the province of detective work must always
|
|
sherlock\x00be, to a very large extent, the result of luck. Sherlock Holmes
|
|
sherlock\x00can extract a clew from a wisp of straw or a flake of cigar ash;
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/109
|
|
clean!(feature_109_max_depth, "far", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("one");
|
|
wd.create("one/pass", "far");
|
|
wd.create_dir("one/too");
|
|
wd.create("one/too/many", "far");
|
|
|
|
cmd.arg("--maxdepth").arg("2");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = path("one/pass:far\n");
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/124
|
|
clean!(feature_109_case_sensitive_part1, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "tEsT");
|
|
cmd.arg("--smart-case").arg("--case-sensitive");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/124
|
|
clean!(feature_109_case_sensitive_part2, "test", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "tEsT");
|
|
cmd.arg("--ignore-case").arg("--case-sensitive");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/129
|
|
clean!(feature_129_matches, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test\ntest abcdefghijklmnopqrstuvwxyz test");
|
|
cmd.arg("-M26");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "foo:test\nfoo:[Omitted long line with 2 matches]\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/129
|
|
clean!(feature_129_context, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test\nabcdefghijklmnopqrstuvwxyz");
|
|
cmd.arg("-M20").arg("-C1");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "foo:test\nfoo-[Omitted long context line]\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/129
|
|
clean!(feature_129_replace, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test\ntest abcdefghijklmnopqrstuvwxyz test");
|
|
cmd.arg("-M26").arg("-rfoo");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "foo:foo\nfoo:[Omitted long line with 2 replacements]\n";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/159
|
|
clean!(feature_159_works, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test\ntest");
|
|
cmd.arg("-m1");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/159
|
|
clean!(feature_159_zero_max, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test\ntest");
|
|
cmd.arg("-m0");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/243
|
|
clean!(feature_243_column_line, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test");
|
|
cmd.arg("--column");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo:1:1:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/263
|
|
clean!(feature_263_sort_files, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create("foo", "test");
|
|
wd.create("abc", "test");
|
|
wd.create("zoo", "test");
|
|
wd.create("bar", "test");
|
|
cmd.arg("--sort-files");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "abc:test\nbar:test\nfoo:test\nzoo:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/275
|
|
clean!(feature_275_pathsep, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
wd.create_dir("foo");
|
|
wd.create("foo/bar", "test");
|
|
cmd.arg("--path-separator").arg("Z");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "fooZbar:test\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/362
|
|
sherlock!(feature_362_dfa_size_limit, r"For\s",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// This should fall back to the nfa engine but should still produce the
|
|
// expected result.
|
|
cmd.arg("--dfa-size-limit").arg("10");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
sherlock!(feature_362_exceeds_regex_size_limit, r"[0-9]\w+",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("--regex-size-limit").arg("10K");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
#[cfg(target_pointer_width = "32")]
|
|
sherlock!(feature_362_u64_to_narrow_usize_suffix_overflow, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.remove("sherlock");
|
|
wd.create_size("foo", 1000000);
|
|
|
|
// 2^35 * 2^20 is ok for u64, but not for usize
|
|
cmd.arg("--dfa-size-limit").arg("34359738368M").arg("--files");
|
|
wd.assert_err(&mut cmd);
|
|
});
|
|
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/419
|
|
sherlock!(feature_419_zero_as_shortcut_for_null, "Sherlock", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
cmd.arg("-0").arg("--count");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "sherlock\x002\n");
|
|
});
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/709
|
|
clean!(suggest_fixed_strings_for_invalid_regex, "foo(", ".",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
wd.assert_non_empty_stderr(&mut cmd);
|
|
|
|
let output = cmd.output().unwrap();
|
|
let err = String::from_utf8_lossy(&output.stderr);
|
|
assert_eq!(err.contains("--fixed-strings"), true);
|
|
});
|
|
|
|
#[test]
|
|
fn compressed_gzip() {
|
|
if !cmd_exists("gzip") {
|
|
return;
|
|
}
|
|
let gzip_file = include_bytes!("./data/sherlock.gz");
|
|
|
|
let wd = WorkDir::new("feature_search_compressed");
|
|
wd.create_bytes("sherlock.gz", gzip_file);
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-z").arg("Sherlock").arg("sherlock.gz");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
#[test]
|
|
fn compressed_bzip2() {
|
|
if !cmd_exists("bzip2") {
|
|
return;
|
|
}
|
|
let bzip2_file = include_bytes!("./data/sherlock.bz2");
|
|
|
|
let wd = WorkDir::new("feature_search_compressed");
|
|
wd.create_bytes("sherlock.bz2", bzip2_file);
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-z").arg("Sherlock").arg("sherlock.bz2");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
#[test]
|
|
fn compressed_xz() {
|
|
if !cmd_exists("xz") {
|
|
return;
|
|
}
|
|
let xz_file = include_bytes!("./data/sherlock.xz");
|
|
|
|
let wd = WorkDir::new("feature_search_compressed");
|
|
wd.create_bytes("sherlock.xz", xz_file);
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-z").arg("Sherlock").arg("sherlock.xz");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
#[test]
|
|
fn compressed_lzma() {
|
|
if !cmd_exists("xz") {
|
|
return;
|
|
}
|
|
let lzma_file = include_bytes!("./data/sherlock.lzma");
|
|
|
|
let wd = WorkDir::new("feature_search_compressed");
|
|
wd.create_bytes("sherlock.lzma", lzma_file);
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-z").arg("Sherlock").arg("sherlock.lzma");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
#[test]
|
|
fn compressed_failing_gzip() {
|
|
if !cmd_exists("gzip") {
|
|
return;
|
|
}
|
|
let wd = WorkDir::new("feature_search_compressed");
|
|
wd.create("sherlock.gz", hay::SHERLOCK);
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-z").arg("Sherlock").arg("sherlock.gz");
|
|
|
|
wd.assert_non_empty_stderr(&mut cmd);
|
|
|
|
let output = cmd.output().unwrap();
|
|
let err = String::from_utf8_lossy(&output.stderr);
|
|
assert_eq!(err.contains("not in gzip format"), true);
|
|
}
|
|
|
|
sherlock!(feature_196_persistent_config, "sherlock",
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
// Make sure we get no matches by default.
|
|
wd.assert_err(&mut cmd);
|
|
|
|
// Now add our config file, and make sure it impacts ripgrep.
|
|
wd.create(".ripgreprc", "--ignore-case");
|
|
cmd.env("RIPGREP_CONFIG_PATH", ".ripgreprc");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
For the Doctor Watsons of this world, as opposed to the Sherlock
|
|
be, to a very large extent, the result of luck. Sherlock Holmes
|
|
";
|
|
assert_eq!(lines, expected);
|
|
});
|
|
|
|
#[test]
|
|
fn feature_740_passthru() {
|
|
let wd = WorkDir::new("feature_740");
|
|
wd.create("file", "\nfoo\nbar\nfoobar\n\nbaz\n");
|
|
wd.create("patterns", "foo\n\nbar\n");
|
|
|
|
// We can't assume that the way colour specs are translated to ANSI
|
|
// sequences will remain stable, and --replace doesn't currently work with
|
|
// pass-through, so for now we don't actually test the match sub-strings
|
|
let common_args = &["-n", "--passthru"];
|
|
let expected = "\
|
|
1:
|
|
2:foo
|
|
3:bar
|
|
4:foobar
|
|
5:
|
|
6:baz
|
|
";
|
|
|
|
// With single pattern
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args).arg("foo").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, expected);
|
|
|
|
// With multiple -e patterns
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args)
|
|
.arg("-e").arg("foo").arg("-e").arg("bar").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, expected);
|
|
|
|
// With multiple -f patterns
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args).arg("-f").arg("patterns").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, expected);
|
|
|
|
// -c should override
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args).arg("-c").arg("foo").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "2\n");
|
|
|
|
// -o should conflict
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args).arg("-o").arg("foo").arg("file");
|
|
wd.assert_err(&mut cmd);
|
|
|
|
// -r should conflict
|
|
let mut cmd = wd.command();
|
|
cmd.args(common_args).arg("-r").arg("$0").arg("foo").arg("file");
|
|
wd.assert_err(&mut cmd);
|
|
}
|
|
|
|
#[test]
|
|
fn binary_nosearch() {
|
|
let wd = WorkDir::new("binary_nosearch");
|
|
wd.create("file", "foo\x00bar\nfoo\x00baz\n");
|
|
let mut cmd = wd.command();
|
|
cmd.arg("foo").arg("file");
|
|
wd.assert_err(&mut cmd);
|
|
}
|
|
|
|
// The following two tests show a discrepancy in search results between
|
|
// searching with memory mapped files and stream searching. Stream searching
|
|
// uses a heuristic (that GNU grep also uses) where NUL bytes are replaced with
|
|
// the EOL terminator, which tends to avoid allocating large amounts of memory
|
|
// for really long "lines." The memory map searcher has no need to worry about
|
|
// such things, and more than that, it would be pretty hard for it to match
|
|
// the semantics of streaming search in this case.
|
|
//
|
|
// Binary files with lots of NULs aren't really part of the use case of ripgrep
|
|
// (or any other grep-like tool for that matter), so we shouldn't feel too bad
|
|
// about it.
|
|
#[test]
|
|
fn binary_search_mmap() {
|
|
let wd = WorkDir::new("binary_search_mmap");
|
|
wd.create("file", "foo\x00bar\nfoo\x00baz\n");
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-a").arg("--mmap").arg("foo").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo\x00bar\nfoo\x00baz\n");
|
|
}
|
|
|
|
#[test]
|
|
fn binary_search_no_mmap() {
|
|
let wd = WorkDir::new("binary_search_no_mmap");
|
|
wd.create("file", "foo\x00bar\nfoo\x00baz\n");
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-a").arg("--no-mmap").arg("foo").arg("file");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo\x00bar\nfoo\x00baz\n");
|
|
}
|
|
|
|
#[test]
|
|
fn files() {
|
|
let wd = WorkDir::new("files");
|
|
wd.create("file", "");
|
|
wd.create_dir("dir");
|
|
wd.create("dir/file", "");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--files");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert!(lines == path("file\ndir/file\n")
|
|
|| lines == path("dir/file\nfile\n"));
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/64
|
|
#[test]
|
|
fn regression_64() {
|
|
let wd = WorkDir::new("regression_64");
|
|
wd.create_dir("dir");
|
|
wd.create_dir("foo");
|
|
wd.create("dir/abc", "");
|
|
wd.create("foo/abc", "");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--files").arg("foo");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, path("foo/abc\n"));
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/270
|
|
#[test]
|
|
fn regression_270() {
|
|
let wd = WorkDir::new("regression_270");
|
|
wd.create("foo", "-test");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-e").arg("-test");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, path("foo:-test\n"));
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/391
|
|
#[test]
|
|
fn regression_391() {
|
|
let wd = WorkDir::new("regression_391");
|
|
wd.create_dir(".git");
|
|
wd.create("lock", "");
|
|
wd.create("bar.py", "");
|
|
wd.create(".git/packed-refs", "");
|
|
wd.create(".git/description", "");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--no-ignore").arg("--hidden").arg("--follow").arg("--files")
|
|
.arg("--glob")
|
|
.arg("!{.git,node_modules,plugged}/**")
|
|
.arg("--glob")
|
|
.arg("*.{js,json,php,md,styl,scss,sass,pug,html,config,py,cpp,c,go,hs}");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "bar.py\n");
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/451
|
|
#[test]
|
|
fn regression_451_only_matching_as_in_issue() {
|
|
let wd = WorkDir::new("regression_451_only_matching");
|
|
let path = "digits.txt";
|
|
wd.create(path, "1 2 3\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("[0-9]+").arg(path).arg("--only-matching");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "\
|
|
1
|
|
2
|
|
3
|
|
";
|
|
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/451
|
|
#[test]
|
|
fn regression_451_only_matching() {
|
|
let wd = WorkDir::new("regression_451_only_matching");
|
|
let path = "digits.txt";
|
|
wd.create(path, "1 2 3\n123\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("[0-9]").arg(path)
|
|
.arg("--only-matching")
|
|
.arg("--column");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "\
|
|
1:1:1
|
|
1:3:2
|
|
1:5:3
|
|
2:1:1
|
|
2:2:2
|
|
2:3:3
|
|
";
|
|
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/483
|
|
#[test]
|
|
fn regression_483_matching_no_stdout() {
|
|
let wd = WorkDir::new("regression_483_matching_no_stdout");
|
|
wd.create("file.py", "");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--quiet")
|
|
.arg("--files")
|
|
.arg("--glob").arg("*.py");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert!(lines.is_empty());
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/483
|
|
#[test]
|
|
fn regression_483_non_matching_exit_code() {
|
|
let wd = WorkDir::new("regression_483_non_matching_exit_code");
|
|
wd.create("file.rs", "");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--quiet")
|
|
.arg("--files")
|
|
.arg("--glob").arg("*.py");
|
|
|
|
wd.assert_err(&mut cmd);
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/506
|
|
#[test]
|
|
fn regression_506_word_boundaries_not_parenthesized() {
|
|
let wd = WorkDir::new("regression_506_word_boundaries_not_parenthesized");
|
|
let path = "wb.txt";
|
|
wd.create(path, "min minimum amin\n\
|
|
max maximum amax");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-w").arg("min|max").arg(path).arg("--only-matching");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "min\nmax\n";
|
|
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/568
|
|
#[test]
|
|
fn regression_568_leading_hyphen_option_arguments() {
|
|
let wd = WorkDir::new("regression_568_leading_hyphen_option_arguments");
|
|
let path = "file";
|
|
wd.create(path, "foo bar -baz\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-e-baz").arg("-e").arg("-baz").arg(path);
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo bar -baz\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-rni").arg("bar").arg(path);
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo ni -baz\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-r").arg("-n").arg("-i").arg("bar").arg(path);
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
assert_eq!(lines, "foo -n -baz\n");
|
|
}
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/693
|
|
#[test]
|
|
fn regression_693_context_option_in_contextless_mode() {
|
|
let wd = WorkDir::new("regression_693_context_option_in_contextless_mode");
|
|
|
|
wd.create("foo", "xyz\n");
|
|
wd.create("bar", "xyz\n");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("-C1").arg("-c").arg("--sort-files").arg("xyz");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
let expected = "\
|
|
bar:1
|
|
foo:1
|
|
";
|
|
assert_eq!(lines, expected);
|
|
}
|
|
|
|
#[test]
|
|
fn type_list() {
|
|
let wd = WorkDir::new("type_list");
|
|
|
|
let mut cmd = wd.command();
|
|
cmd.arg("--type-list");
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
// This can change over time, so just make sure we print something.
|
|
assert!(!lines.is_empty());
|
|
}
|