/*! 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 { let mut xs: Vec<_> = unix.iter().map(|s| path(s)).collect(); xs.sort(); xs } fn paths_from_stdout(stdout: String) -> Vec { 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 = 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[A-Z][a-z]+) (?P[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| { 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, \ nounderline, underline. "; 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 = "\ 1: 2: 4: "; assert_eq!(expected, lines); }); // See: https://github.com/BurntSushi/ripgrep/issues/807 clean!(regression_807, "test", ".", |wd: WorkDir, mut cmd: Command| { wd.create(".gitignore", ".a/b"); wd.create_dir(".a/b"); wd.create_dir(".a/c"); wd.create(".a/b/file", "test"); wd.create(".a/c/file", "test"); cmd.arg("--hidden"); let lines: String = wd.stdout(&mut cmd); assert_eq!(lines, format!("{}:test\n", path(".a/c/file"))); }); // 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()); }