2016-09-10 04:58:30 +02:00
|
|
|
/*!
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-09-24 22:29:14 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-09-25 21:45:51 +02:00
|
|
|
fn path(unix: &str) -> String {
|
|
|
|
if cfg!(windows) {
|
|
|
|
unix.replace("/", "\\")
|
|
|
|
} else {
|
|
|
|
unix.to_string()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-12 01:57:09 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2016-09-27 01:57:23 +02:00
|
|
|
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"))
|
|
|
|
}
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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!(columns, |wd: WorkDir, mut cmd: Command| {
|
|
|
|
cmd.arg("--column");
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "\
|
2017-01-12 01:53:35 +02:00
|
|
|
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
|
2016-09-10 04:58:30 +02:00
|
|
|
";
|
|
|
|
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 fil.e
|
|
|
|
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);
|
2016-09-10 05:24:01 +02:00
|
|
|
let expected1 = "\
|
2016-09-10 04:58:30 +02:00
|
|
|
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
|
|
|
|
";
|
2016-09-10 05:24:01 +02:00
|
|
|
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.
|
|
|
|
";
|
2016-09-26 03:26:49 +02:00
|
|
|
if lines != expected1 {
|
|
|
|
assert_eq!(lines, expected2);
|
|
|
|
} else {
|
|
|
|
assert_eq!(lines, expected1);
|
|
|
|
}
|
2016-09-10 04:58:30 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
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!(literal, "()", "file", |wd: WorkDir, mut cmd: Command| {
|
|
|
|
wd.create("file", "blib\n()\nblab\n");
|
2016-09-17 22:55:58 +02:00
|
|
|
cmd.arg("-F");
|
2016-09-10 04:58:30 +02:00
|
|
|
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!(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");
|
|
|
|
});
|
|
|
|
|
2016-09-11 19:26:53 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2016-09-11 19:26:53 +02:00
|
|
|
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
|
|
|
|
");
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2017-01-02 02:32:46 +02:00
|
|
|
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);
|
2017-01-08 05:50:38 +02:00
|
|
|
assert_eq!(sort_lines(&lines), "file.py:Sherlock\nfile.wat:Sherlock\n");
|
2017-01-02 02:32:46 +02:00
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2016-09-24 04:06:34 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-11-20 03:15:41 +02:00
|
|
|
sherlock!(files_without_matches, "Sherlock", ".",
|
|
|
|
|wd: WorkDir, mut cmd: Command| {
|
2016-11-20 01:48:59 +02:00
|
|
|
wd.create("file.py", "foo");
|
2016-11-20 03:15:41 +02:00
|
|
|
cmd.arg("--files-without-match");
|
2016-11-20 01:48:59 +02:00
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "file.py\n";
|
|
|
|
assert_eq!(lines, expected);
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2017-02-28 06:53:52 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2017-03-01 07:38:06 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-24 04:44:17 +02:00
|
|
|
sherlock!(ignore_generic, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
|
|
wd.create(".ignore", "sherlock\n");
|
|
|
|
wd.assert_err(&mut cmd);
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-11-11 13:44:23 +02:00
|
|
|
#[cfg(not(windows))]
|
2016-09-10 04:58:30 +02:00
|
|
|
sherlock!(symlink_nofollow, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
|
|
|
wd.remove("sherlock");
|
|
|
|
wd.create_dir("foo");
|
|
|
|
wd.create_dir("foo/bar");
|
2016-10-11 01:34:57 +02:00
|
|
|
wd.link_dir("foo/baz", "foo/bar/baz");
|
2016-09-10 04:58:30 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-11-11 13:44:23 +02:00
|
|
|
#[cfg(not(windows))]
|
2016-09-10 04:58:30 +02:00
|
|
|
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);
|
2016-10-11 01:34:57 +02:00
|
|
|
wd.link_dir("foo/baz", "foo/bar/baz");
|
2016-09-10 04:58:30 +02:00
|
|
|
cmd.arg("-L");
|
|
|
|
cmd.current_dir(wd.path().join("foo/bar"));
|
|
|
|
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
2016-09-25 21:45:51 +02:00
|
|
|
let expected = "\
|
2016-09-10 04:58:30 +02:00
|
|
|
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
|
|
|
|
";
|
2016-09-25 21:45:51 +02:00
|
|
|
assert_eq!(lines, path(expected));
|
2016-09-10 04:58:30 +02:00
|
|
|
});
|
|
|
|
|
2016-09-21 02:24:03 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-22 03:07:36 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2016-09-23 03:32:38 +02:00
|
|
|
sherlock!(vimgrep, "Sherlock|Watson", ".", |wd: WorkDir, mut cmd: Command| {
|
|
|
|
cmd.arg("--vimgrep");
|
|
|
|
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
let expected = "\
|
2016-09-23 23:11:09 +02:00
|
|
|
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,
|
2016-09-23 03:32:38 +02:00
|
|
|
";
|
|
|
|
assert_eq!(lines, expected);
|
|
|
|
});
|
|
|
|
|
2016-09-24 22:29:14 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-09-25 00:40:50 +02:00
|
|
|
// 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);
|
2016-09-25 21:45:51 +02:00
|
|
|
let expected = path("src/llvm/foo:test\n");
|
2016-09-25 00:40:50 +02:00
|
|
|
assert_eq!(lines, expected);
|
|
|
|
|
|
|
|
cmd.current_dir(wd.path().join("src"));
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
2016-09-25 21:45:51 +02:00
|
|
|
let expected = path("llvm/foo:test\n");
|
2016-09-25 00:40:50 +02:00
|
|
|
assert_eq!(lines, expected);
|
|
|
|
});
|
|
|
|
|
2016-09-25 01:44:06 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/30
|
|
|
|
clean!(regression_30, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
2016-09-25 21:45:51 +02:00
|
|
|
if cfg!(windows) {
|
|
|
|
wd.create(".gitignore", "vendor/**\n!vendor\\manifest");
|
|
|
|
} else {
|
|
|
|
wd.create(".gitignore", "vendor/**\n!vendor/manifest");
|
|
|
|
}
|
2016-09-25 01:44:06 +02:00
|
|
|
wd.create_dir("vendor");
|
|
|
|
wd.create("vendor/manifest", "test");
|
|
|
|
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
2016-09-25 21:45:51 +02:00
|
|
|
let expected = path("vendor/manifest:test\n");
|
2016-09-25 01:44:06 +02:00
|
|
|
assert_eq!(lines, expected);
|
|
|
|
});
|
|
|
|
|
2016-09-24 22:29:14 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-09-25 02:10:26 +02:00
|
|
|
// 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);
|
2016-09-25 21:45:51 +02:00
|
|
|
assert_eq!(lines, path("dir/bar:test\n"));
|
2016-09-25 02:10:26 +02:00
|
|
|
});
|
|
|
|
|
2016-10-11 01:16:52 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-09-26 00:31:41 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-09-26 02:10:28 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-09-27 01:57:23 +02:00
|
|
|
// 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"));
|
|
|
|
});
|
|
|
|
|
2016-09-27 01:09:59 +02:00
|
|
|
// 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);
|
2017-01-12 01:53:35 +02:00
|
|
|
assert_eq!(lines, "foo:1:3:zztest\n");
|
2016-09-27 01:09:59 +02:00
|
|
|
});
|
|
|
|
|
2016-10-11 01:16:52 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-09-23 04:59:25 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-10-11 01:16:52 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/131
|
2016-10-11 03:03:11 +02:00
|
|
|
//
|
|
|
|
// TODO(burntsushi): Darwin doesn't like this test for some reason.
|
2016-10-11 03:48:47 +02:00
|
|
|
#[cfg(not(target_os = "macos"))]
|
2016-10-11 01:16:52 +02:00
|
|
|
clean!(regression_131, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
|
|
|
wd.create(".gitignore", "TopÑapa");
|
|
|
|
wd.create("TopÑapa", "test");
|
|
|
|
wd.assert_err(&mut cmd);
|
|
|
|
});
|
|
|
|
|
2016-10-11 01:27:12 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/137
|
2016-10-11 01:34:57 +02:00
|
|
|
//
|
|
|
|
// 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))]
|
2016-10-11 01:27:12 +02:00
|
|
|
sherlock!(regression_137, "Sherlock", ".", |wd: WorkDir, mut cmd: Command| {
|
2016-10-11 01:34:57 +02:00
|
|
|
wd.link_file("sherlock", "sym1");
|
|
|
|
wd.link_file("sherlock", "sym2");
|
2016-10-11 01:27:12 +02:00
|
|
|
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));
|
|
|
|
});
|
|
|
|
|
2016-10-11 04:04:29 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-10-16 16:15:11 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-11-06 19:07:47 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-11-01 01:53:25 +02:00
|
|
|
// 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")));
|
|
|
|
});
|
|
|
|
|
Switch from Docopt to Clap.
There were two important reasons for the switch:
1. Performance. Docopt does poorly when the argv becomes large, which is
a reasonable common use case for search tools. (e.g., use with xargs)
2. Better failure modes. Clap knows a lot more about how a particular
argv might be invalid, and can therefore provide much clearer error
messages.
While both were important, (1) made it urgent.
Note that since Clap requires at least Rust 1.11, this will in turn
increase the minimum Rust version supported by ripgrep from Rust 1.9 to
Rust 1.11. It is therefore a breaking change, so the soonest release of
ripgrep with Clap will have to be 0.3.
There is also at least one subtle breaking change in real usage.
Previous to this commit, this used to work:
rg -e -foo
Where this would cause ripgrep to search for the string `-foo`. Clap
currently has problems supporting this use case
(see: https://github.com/kbknapp/clap-rs/issues/742),
but it can be worked around by using this instead:
rg -e [-]foo
or even
rg [-]foo
and this still works:
rg -- -foo
This commit also adds Bash, Fish and PowerShell completion files to the
release, fixes a bug that prevented ripgrep from working on file
paths containing invalid UTF-8 and shows short descriptions in the
output of `-h` but longer descriptions in the output of `--help`.
Fixes #136, Fixes #189, Fixes #210, Fixes #230
2016-11-13 04:48:11 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/210
|
|
|
|
#[cfg(unix)]
|
|
|
|
#[test]
|
2016-11-18 03:29:53 +02:00
|
|
|
fn regression_210() {
|
Switch from Docopt to Clap.
There were two important reasons for the switch:
1. Performance. Docopt does poorly when the argv becomes large, which is
a reasonable common use case for search tools. (e.g., use with xargs)
2. Better failure modes. Clap knows a lot more about how a particular
argv might be invalid, and can therefore provide much clearer error
messages.
While both were important, (1) made it urgent.
Note that since Clap requires at least Rust 1.11, this will in turn
increase the minimum Rust version supported by ripgrep from Rust 1.9 to
Rust 1.11. It is therefore a breaking change, so the soonest release of
ripgrep with Clap will have to be 0.3.
There is also at least one subtle breaking change in real usage.
Previous to this commit, this used to work:
rg -e -foo
Where this would cause ripgrep to search for the string `-foo`. Clap
currently has problems supporting this use case
(see: https://github.com/kbknapp/clap-rs/issues/742),
but it can be worked around by using this instead:
rg -e [-]foo
or even
rg [-]foo
and this still works:
rg -- -foo
This commit also adds Bash, Fish and PowerShell completion files to the
release, fixes a bug that prevented ripgrep from working on file
paths containing invalid UTF-8 and shows short descriptions in the
output of `-h` but longer descriptions in the output of `--help`.
Fixes #136, Fixes #189, Fixes #210, Fixes #230
2016-11-13 04:48:11 +02:00
|
|
|
use std::ffi::OsStr;
|
|
|
|
use std::os::unix::ffi::OsStrExt;
|
|
|
|
|
|
|
|
let badutf8 = OsStr::from_bytes(&b"foo\xffbar"[..]);
|
|
|
|
|
2016-11-18 03:29:53 +02:00
|
|
|
let wd = WorkDir::new("regression_210");
|
Switch from Docopt to Clap.
There were two important reasons for the switch:
1. Performance. Docopt does poorly when the argv becomes large, which is
a reasonable common use case for search tools. (e.g., use with xargs)
2. Better failure modes. Clap knows a lot more about how a particular
argv might be invalid, and can therefore provide much clearer error
messages.
While both were important, (1) made it urgent.
Note that since Clap requires at least Rust 1.11, this will in turn
increase the minimum Rust version supported by ripgrep from Rust 1.9 to
Rust 1.11. It is therefore a breaking change, so the soonest release of
ripgrep with Clap will have to be 0.3.
There is also at least one subtle breaking change in real usage.
Previous to this commit, this used to work:
rg -e -foo
Where this would cause ripgrep to search for the string `-foo`. Clap
currently has problems supporting this use case
(see: https://github.com/kbknapp/clap-rs/issues/742),
but it can be worked around by using this instead:
rg -e [-]foo
or even
rg [-]foo
and this still works:
rg -- -foo
This commit also adds Bash, Fish and PowerShell completion files to the
release, fixes a bug that prevented ripgrep from working on file
paths containing invalid UTF-8 and shows short descriptions in the
output of `-h` but longer descriptions in the output of `--help`.
Fixes #136, Fixes #189, Fixes #210, Fixes #230
2016-11-13 04:48:11 +02:00
|
|
|
let mut cmd = wd.command();
|
|
|
|
wd.create(badutf8, "test");
|
|
|
|
cmd.arg("-H").arg("test").arg(badutf8);
|
|
|
|
|
|
|
|
let out = wd.output(&mut cmd);
|
|
|
|
assert_eq!(out.stdout, b"foo\xffbar:test\n".to_vec());
|
|
|
|
}
|
|
|
|
|
2016-11-18 03:29:53 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-11-29 00:57:26 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-11-29 01:31:58 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-12-06 03:05:57 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-12-12 13:55:49 +02:00
|
|
|
// 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, "");
|
|
|
|
});
|
|
|
|
|
2017-03-12 21:51:07 +02:00
|
|
|
// 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);
|
2017-03-12 22:07:01 +02:00
|
|
|
assert_eq!(lines, format!("{}:test\n", path("bar/foo/file2.txt")));
|
2017-03-12 21:51:07 +02:00
|
|
|
});
|
|
|
|
|
Add support for additional text encodings.
This includes, but is not limited to, UTF-16, latin-1, GBK, EUC-JP and
Shift_JIS. (Courtesy of the `encoding_rs` crate.)
Specifically, this feature enables ripgrep to search files that are
encoded in an encoding other than UTF-8. The list of available encodings
is tied directly to what the `encoding_rs` crate supports, which is in
turn tied to the Encoding Standard. The full list of available encodings
can be found here: https://encoding.spec.whatwg.org/#concept-encoding-get
This pull request also introduces the notion that text encodings can be
automatically detected on a best effort basis. Currently, the only
support for this is checking for a UTF-16 bom. In all other cases, a
text encoding of `auto` (the default) implies a UTF-8 or ASCII
compatible source encoding. When a text encoding is otherwise specified,
it is unconditionally used for all files searched.
Since ripgrep's regex engine is fundamentally built on top of UTF-8,
this feature works by transcoding the files to be searched from their
source encoding to UTF-8. This transcoding only happens when:
1. `auto` is specified and a non-UTF-8 encoding is detected.
2. A specific encoding is given by end users (including UTF-8).
When transcoding occurs, errors are handled by automatically inserting
the Unicode replacement character. In this case, ripgrep's output is
guaranteed to be valid UTF-8 (excluding non-UTF-8 file paths, if they
are printed).
In all other cases, the source text is searched directly, which implies
an assumption that it is at least ASCII compatible, but where UTF-8 is
most useful. In this scenario, encoding errors are not detected. In this
case, ripgrep's output will match the input exactly, byte-for-byte.
This design may not be optimal in all cases, but it has some advantages:
1. In the happy path ("UTF-8 everywhere") remains happy. I have not been
able to witness any performance regressions.
2. In the non-UTF-8 path, implementation complexity is kept relatively
low. The cost here is transcoding itself. A potentially superior
implementation might build decoding of any encoding into the regex
engine itself. In particular, the fundamental problem with
transcoding everything first is that literal optimizations are nearly
negated.
Future work should entail improving the user experience. For example, we
might want to auto-detect more text encodings. A more elaborate UX
experience might permit end users to specify multiple text encodings,
although this seems hard to pull off in an ergonomic way.
Fixes #1
2017-03-09 03:22:48 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-11-09 13:07:53 +02:00
|
|
|
// 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);
|
Fix a bug with handling --ignore-file.
Namely, passing a directory to --ignore-file caused ripgrep to allocate
memory without bound.
The issue was that I got a bit overzealous with partial error
reporting. Namely, when processing a gitignore file, we should try
to use every pattern even if some patterns are invalid globs (e.g.,
a**b). In the process, I applied the same logic to I/O errors. In this
case, it manifest by attempting to read lines from a directory, which
appears to yield Results forever, where each Result is an error of the
form "you can't read from a directory silly." Since I treated it as a
partial error, ripgrep was just spinning and accruing each error in
memory, which caused the OOM killer to kick in.
Fixes #228
2016-11-09 23:45:21 +02:00
|
|
|
});
|
|
|
|
|
2016-09-25 01:23:19 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/20
|
2016-09-28 22:30:57 +02:00
|
|
|
sherlock!(feature_20_no_filename, "Sherlock", ".",
|
|
|
|
|wd: WorkDir, mut cmd: Command| {
|
2016-09-25 01:23:19 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-10-12 01:57:09 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-09-25 03:31:24 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/68
|
2016-09-28 22:30:57 +02:00
|
|
|
clean!(feature_68_no_ignore_vcs, "test", ".", |wd: WorkDir, mut cmd: Command| {
|
2016-09-25 03:31:24 +02:00
|
|
|
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");
|
|
|
|
});
|
|
|
|
|
2016-09-25 03:51:04 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/70
|
2016-09-28 22:30:57 +02:00
|
|
|
sherlock!(feature_70_smart_case, "sherlock", ".",
|
|
|
|
|wd: WorkDir, mut cmd: Command| {
|
2016-09-25 03:51:04 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-27 01:21:17 +02:00
|
|
|
// 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
|
2016-11-20 01:48:59 +02:00
|
|
|
sherlock!(feature_89_files_without_matches, "Sherlock", ".",
|
|
|
|
|wd: WorkDir, mut cmd: Command| {
|
|
|
|
wd.create("file.py", "foo");
|
2016-11-20 03:15:41 +02:00
|
|
|
cmd.arg("--null").arg("--files-without-match");
|
2016-11-20 01:48:59 +02:00
|
|
|
|
|
|
|
let lines: String = wd.stdout(&mut cmd);
|
|
|
|
assert_eq!(lines, "file.py\x00");
|
|
|
|
});
|
|
|
|
|
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/89
|
2016-09-27 01:21:17 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-27 05:56:15 +02:00
|
|
|
// See: https://github.com/BurntSushi/ripgrep/issues/109
|
2016-09-28 22:30:57 +02:00
|
|
|
clean!(feature_109_max_depth, "far", ".", |wd: WorkDir, mut cmd: Command| {
|
2016-09-27 05:56:15 +02:00
|
|
|
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);
|
|
|
|
});
|
|
|
|
|
2016-09-28 22:30:57 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2017-02-02 16:29:50 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2016-11-06 20:09:53 +02:00
|
|
|
// 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);
|
|
|
|
});
|
|
|
|
|
2017-01-12 01:53:35 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2017-01-07 05:43:59 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2017-01-11 01:16:15 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2017-03-28 18:06:30 +02:00
|
|
|
// 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");
|
|
|
|
});
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
#[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);
|
2016-09-26 03:26:49 +02:00
|
|
|
assert_eq!(lines, "foo\x00bar\nfoo\x00baz\n");
|
2016-09-10 04:58:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[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);
|
2016-09-25 21:45:51 +02:00
|
|
|
assert!(lines == path("file\ndir/file\n")
|
|
|
|
|| lines == path("dir/file\nfile\n"));
|
2016-09-10 04:58:30 +02:00
|
|
|
}
|
|
|
|
|
2016-09-25 02:22:02 +02:00
|
|
|
// 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);
|
2016-09-25 21:45:51 +02:00
|
|
|
assert_eq!(lines, path("foo/abc\n"));
|
2016-09-25 02:22:02 +02:00
|
|
|
}
|
|
|
|
|
2016-12-07 00:29:34 +02:00
|
|
|
// 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"));
|
|
|
|
}
|
|
|
|
|
2017-03-04 10:40:19 +02:00
|
|
|
// 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");
|
|
|
|
}
|
|
|
|
|
2016-09-10 04:58:30 +02:00
|
|
|
#[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());
|
|
|
|
}
|