mirror of
https://github.com/uutils/coreutils
synced 2024-12-13 23:02:38 +00:00
23d3e58f33
* Change data structure from Vec to HashMap in order to better keep track of files while watching them with `--follow=name`. E.g. file paths that were removed while watching them and exit if no files are remaining, etc. * Move all logic related to file handling into a FileHandling trait * Simplify handling of the verbose flag.
718 lines
21 KiB
Rust
718 lines
21 KiB
Rust
// * This file is part of the uutils coreutils package.
|
|
// *
|
|
// * For the full copyright and license information, please view the LICENSE
|
|
// * file that was distributed with this source code.
|
|
|
|
// spell-checker:ignore (ToDO) abcdefghijklmnopqrstuvwxyz efghijklmnopqrstuvwxyz vwxyz emptyfile logfile
|
|
// spell-checker:ignore (libs) kqueue
|
|
|
|
extern crate tail;
|
|
|
|
use crate::common::util::*;
|
|
use std::char::from_digit;
|
|
use std::io::{Read, Write};
|
|
use std::thread::sleep;
|
|
use std::time::Duration;
|
|
|
|
static FOOBAR_TXT: &str = "foobar.txt";
|
|
static FOOBAR_2_TXT: &str = "foobar2.txt";
|
|
static FOOBAR_WITH_NULL_TXT: &str = "foobar_with_null.txt";
|
|
static FOLLOW_NAME_TXT: &str = "follow_name.txt";
|
|
static FOLLOW_NAME_SHORT_EXP: &str = "follow_name_short.expected";
|
|
static FOLLOW_NAME_EXP: &str = "follow_name.expected";
|
|
|
|
#[test]
|
|
fn test_stdin_default() {
|
|
new_ucmd!()
|
|
.pipe_in_fixture(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_stdin_default.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_stdin_explicit() {
|
|
new_ucmd!()
|
|
.pipe_in_fixture(FOOBAR_TXT)
|
|
.arg("-")
|
|
.run()
|
|
.stdout_is_fixture("foobar_stdin_default.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_single_default() {
|
|
new_ucmd!()
|
|
.arg(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_single_default.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_n_greater_than_number_of_lines() {
|
|
new_ucmd!()
|
|
.arg("-n")
|
|
.arg("99999999")
|
|
.arg(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture(FOOBAR_TXT);
|
|
}
|
|
|
|
#[test]
|
|
fn test_null_default() {
|
|
new_ucmd!()
|
|
.arg("-z")
|
|
.arg(FOOBAR_WITH_NULL_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_with_null_default.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_follow() {
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
|
|
let mut child = ucmd.arg("-f").arg(FOOBAR_TXT).run_no_wait();
|
|
|
|
let expected = at.read("foobar_single_default.expected");
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
// We write in a temporary copy of foobar.txt
|
|
let expected = "line1\nline2\n";
|
|
at.append(FOOBAR_TXT, expected);
|
|
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
child.kill().unwrap();
|
|
}
|
|
|
|
#[test]
|
|
fn test_follow_multiple() {
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
let mut child = ucmd
|
|
.arg("-f")
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.run_no_wait();
|
|
|
|
let expected = at.read("foobar_follow_multiple.expected");
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
let first_append = "trois\n";
|
|
at.append(FOOBAR_2_TXT, first_append);
|
|
assert_eq!(read_size(&mut child, first_append.len()), first_append);
|
|
|
|
let second_append = "twenty\nthirty\n";
|
|
let expected = at.read("foobar_follow_multiple_appended.expected");
|
|
at.append(FOOBAR_TXT, second_append);
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
child.kill().unwrap();
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_multiple() {
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
let mut child = ucmd
|
|
.arg("--follow=name")
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.run_no_wait();
|
|
|
|
let expected = at.read("foobar_follow_multiple.expected");
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
let first_append = "trois\n";
|
|
at.append(FOOBAR_2_TXT, first_append);
|
|
assert_eq!(read_size(&mut child, first_append.len()), first_append);
|
|
|
|
let second_append = "twenty\nthirty\n";
|
|
let expected = at.read("foobar_follow_multiple_appended.expected");
|
|
at.append(FOOBAR_TXT, second_append);
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
child.kill().unwrap();
|
|
}
|
|
|
|
#[test]
|
|
fn test_follow_stdin() {
|
|
new_ucmd!()
|
|
.arg("-f")
|
|
.pipe_in_fixture(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture("follow_stdin.expected");
|
|
}
|
|
|
|
// FixME: test PASSES for usual windows builds, but fails for coverage testing builds (likely related to the specific RUSTFLAGS '-Zpanic_abort_tests -Cpanic=abort') This test also breaks tty settings under bash requiring a 'stty sane' or reset. // spell-checker:disable-line
|
|
#[cfg(disable_until_fixed)]
|
|
#[test]
|
|
fn test_follow_with_pid() {
|
|
use std::process::{Command, Stdio};
|
|
use std::thread::sleep;
|
|
use std::time::Duration;
|
|
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
|
|
#[cfg(unix)]
|
|
let dummy_cmd = "sh";
|
|
#[cfg(windows)]
|
|
let dummy_cmd = "cmd";
|
|
|
|
let mut dummy = Command::new(dummy_cmd)
|
|
.stdout(Stdio::null())
|
|
.spawn()
|
|
.unwrap();
|
|
let pid = dummy.id();
|
|
|
|
let mut child = ucmd
|
|
.arg("-f")
|
|
.arg(format!("--pid={}", pid))
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.run_no_wait();
|
|
|
|
let expected = at.read("foobar_follow_multiple.expected");
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
let first_append = "trois\n";
|
|
at.append(FOOBAR_2_TXT, first_append);
|
|
assert_eq!(read_size(&mut child, first_append.len()), first_append);
|
|
|
|
let second_append = "twenty\nthirty\n";
|
|
let expected = at.read("foobar_follow_multiple_appended.expected");
|
|
at.append(FOOBAR_TXT, second_append);
|
|
assert_eq!(read_size(&mut child, expected.len()), expected);
|
|
|
|
// kill the dummy process and give tail time to notice this
|
|
dummy.kill().unwrap();
|
|
let _ = dummy.wait();
|
|
sleep(Duration::from_secs(1));
|
|
|
|
let third_append = "should\nbe\nignored\n";
|
|
at.append(FOOBAR_TXT, third_append);
|
|
assert_eq!(read_size(&mut child, 1), "\u{0}");
|
|
|
|
// On Unix, trying to kill a process that's already dead is fine; on Windows it's an error.
|
|
#[cfg(unix)]
|
|
child.kill().unwrap();
|
|
#[cfg(windows)]
|
|
assert_eq!(child.kill().is_err(), true);
|
|
}
|
|
|
|
#[test]
|
|
fn test_single_big_args() {
|
|
const FILE: &str = "single_big_args.txt";
|
|
const EXPECTED_FILE: &str = "single_big_args_expected.txt";
|
|
const LINES: usize = 1_000_000;
|
|
const N_ARG: usize = 100_000;
|
|
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
|
|
let mut big_input = at.make_file(FILE);
|
|
for i in 0..LINES {
|
|
writeln!(&mut big_input, "Line {}", i).expect("Could not write to FILE");
|
|
}
|
|
big_input.flush().expect("Could not flush FILE");
|
|
|
|
let mut big_expected = at.make_file(EXPECTED_FILE);
|
|
for i in (LINES - N_ARG)..LINES {
|
|
writeln!(&mut big_expected, "Line {}", i).expect("Could not write to EXPECTED_FILE");
|
|
}
|
|
big_expected.flush().expect("Could not flush EXPECTED_FILE");
|
|
|
|
ucmd.arg(FILE)
|
|
.arg("-n")
|
|
.arg(format!("{}", N_ARG))
|
|
.run()
|
|
.stdout_is(at.read(EXPECTED_FILE));
|
|
}
|
|
|
|
#[test]
|
|
fn test_bytes_single() {
|
|
new_ucmd!()
|
|
.arg("-c")
|
|
.arg("10")
|
|
.arg(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_bytes_single.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_bytes_stdin() {
|
|
new_ucmd!()
|
|
.arg("-c")
|
|
.arg("13")
|
|
.pipe_in_fixture(FOOBAR_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_bytes_stdin.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_bytes_big() {
|
|
const FILE: &str = "test_bytes_big.txt";
|
|
const EXPECTED_FILE: &str = "test_bytes_big_expected.txt";
|
|
const BYTES: usize = 1_000_000;
|
|
const N_ARG: usize = 100_000;
|
|
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
|
|
let mut big_input = at.make_file(FILE);
|
|
for i in 0..BYTES {
|
|
let digit = from_digit((i % 10) as u32, 10).unwrap();
|
|
write!(&mut big_input, "{}", digit).expect("Could not write to FILE");
|
|
}
|
|
big_input.flush().expect("Could not flush FILE");
|
|
|
|
let mut big_expected = at.make_file(EXPECTED_FILE);
|
|
for i in (BYTES - N_ARG)..BYTES {
|
|
let digit = from_digit((i % 10) as u32, 10).unwrap();
|
|
write!(&mut big_expected, "{}", digit).expect("Could not write to EXPECTED_FILE");
|
|
}
|
|
big_expected.flush().expect("Could not flush EXPECTED_FILE");
|
|
|
|
let result = ucmd
|
|
.arg(FILE)
|
|
.arg("-c")
|
|
.arg(format!("{}", N_ARG))
|
|
.succeeds()
|
|
.stdout_move_str();
|
|
let expected = at.read(EXPECTED_FILE);
|
|
|
|
assert_eq!(result.len(), expected.len());
|
|
for (actual_char, expected_char) in result.chars().zip(expected.chars()) {
|
|
assert_eq!(actual_char, expected_char);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_lines_with_size_suffix() {
|
|
const FILE: &str = "test_lines_with_size_suffix.txt";
|
|
const EXPECTED_FILE: &str = "test_lines_with_size_suffix_expected.txt";
|
|
const LINES: usize = 3_000;
|
|
const N_ARG: usize = 2 * 1024;
|
|
|
|
let (at, mut ucmd) = at_and_ucmd!();
|
|
|
|
let mut big_input = at.make_file(FILE);
|
|
for i in 0..LINES {
|
|
writeln!(&mut big_input, "Line {}", i).expect("Could not write to FILE");
|
|
}
|
|
big_input.flush().expect("Could not flush FILE");
|
|
|
|
let mut big_expected = at.make_file(EXPECTED_FILE);
|
|
for i in (LINES - N_ARG)..LINES {
|
|
writeln!(&mut big_expected, "Line {}", i).expect("Could not write to EXPECTED_FILE");
|
|
}
|
|
big_expected.flush().expect("Could not flush EXPECTED_FILE");
|
|
|
|
ucmd.arg(FILE)
|
|
.arg("-n")
|
|
.arg("2K")
|
|
.run()
|
|
.stdout_is_fixture(EXPECTED_FILE);
|
|
}
|
|
|
|
#[test]
|
|
fn test_multiple_input_files() {
|
|
new_ucmd!()
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.run()
|
|
.stdout_is_fixture("foobar_follow_multiple.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_multiple_input_files_missing() {
|
|
new_ucmd!()
|
|
.arg(FOOBAR_TXT)
|
|
.arg("missing1")
|
|
.arg(FOOBAR_2_TXT)
|
|
.arg("missing2")
|
|
.run()
|
|
.stdout_is_fixture("foobar_follow_multiple.expected")
|
|
.stderr_is(
|
|
"tail: cannot open 'missing1': No such file or directory\n\
|
|
tail: cannot open 'missing2': No such file or directory",
|
|
)
|
|
.code_is(1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_multiple_input_files_with_suppressed_headers() {
|
|
new_ucmd!()
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.arg("-q")
|
|
.run()
|
|
.stdout_is_fixture("foobar_multiple_quiet.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_multiple_input_quiet_flag_overrides_verbose_flag_for_suppressing_headers() {
|
|
new_ucmd!()
|
|
.arg(FOOBAR_TXT)
|
|
.arg(FOOBAR_2_TXT)
|
|
.arg("-v")
|
|
.arg("-q")
|
|
.run()
|
|
.stdout_is_fixture("foobar_multiple_quiet.expected");
|
|
}
|
|
|
|
#[test]
|
|
fn test_negative_indexing() {
|
|
let positive_lines_index = new_ucmd!().arg("-n").arg("5").arg(FOOBAR_TXT).run();
|
|
|
|
let negative_lines_index = new_ucmd!().arg("-n").arg("-5").arg(FOOBAR_TXT).run();
|
|
|
|
let positive_bytes_index = new_ucmd!().arg("-c").arg("20").arg(FOOBAR_TXT).run();
|
|
|
|
let negative_bytes_index = new_ucmd!().arg("-c").arg("-20").arg(FOOBAR_TXT).run();
|
|
|
|
assert_eq!(positive_lines_index.stdout(), negative_lines_index.stdout());
|
|
assert_eq!(positive_bytes_index.stdout(), negative_bytes_index.stdout());
|
|
}
|
|
|
|
#[test]
|
|
fn test_sleep_interval() {
|
|
new_ucmd!().arg("-s").arg("10").arg(FOOBAR_TXT).succeeds();
|
|
new_ucmd!().arg("-s").arg(".1").arg(FOOBAR_TXT).succeeds();
|
|
new_ucmd!().arg("-s.1").arg(FOOBAR_TXT).succeeds();
|
|
new_ucmd!().arg("-s").arg("-1").arg(FOOBAR_TXT).fails();
|
|
new_ucmd!()
|
|
.arg("-s")
|
|
.arg("1..1")
|
|
.arg(FOOBAR_TXT)
|
|
.fails()
|
|
.stderr_contains("invalid number of seconds: '1..1'")
|
|
.code_is(1);
|
|
}
|
|
|
|
/// Test for reading all but the first NUM bytes: `tail -c +3`.
|
|
#[test]
|
|
fn test_positive_bytes() {
|
|
new_ucmd!()
|
|
.args(&["-c", "+3"])
|
|
.pipe_in("abcde")
|
|
.succeeds()
|
|
.stdout_is("cde");
|
|
}
|
|
|
|
/// Test for reading all bytes, specified by `tail -c +0`.
|
|
#[test]
|
|
fn test_positive_zero_bytes() {
|
|
new_ucmd!()
|
|
.args(&["-c", "+0"])
|
|
.pipe_in("abcde")
|
|
.succeeds()
|
|
.stdout_is("abcde");
|
|
}
|
|
|
|
/// Test for reading all but the first NUM lines: `tail -n +3`.
|
|
#[test]
|
|
fn test_positive_lines() {
|
|
new_ucmd!()
|
|
.args(&["-n", "+3"])
|
|
.pipe_in("a\nb\nc\nd\ne\n")
|
|
.succeeds()
|
|
.stdout_is("c\nd\ne\n");
|
|
}
|
|
|
|
/// Test for reading all lines, specified by `tail -n +0`.
|
|
#[test]
|
|
fn test_positive_zero_lines() {
|
|
new_ucmd!()
|
|
.args(&["-n", "+0"])
|
|
.pipe_in("a\nb\nc\nd\ne\n")
|
|
.succeeds()
|
|
.stdout_is("a\nb\nc\nd\ne\n");
|
|
}
|
|
|
|
#[test]
|
|
fn test_tail_invalid_num() {
|
|
new_ucmd!()
|
|
.args(&["-c", "1024R", "emptyfile.txt"])
|
|
.fails()
|
|
.stderr_is("tail: invalid number of bytes: '1024R'");
|
|
new_ucmd!()
|
|
.args(&["-n", "1024R", "emptyfile.txt"])
|
|
.fails()
|
|
.stderr_is("tail: invalid number of lines: '1024R'");
|
|
#[cfg(not(target_pointer_width = "128"))]
|
|
new_ucmd!()
|
|
.args(&["-c", "1Y", "emptyfile.txt"])
|
|
.fails()
|
|
.stderr_is("tail: invalid number of bytes: '1Y': Value too large for defined data type");
|
|
#[cfg(not(target_pointer_width = "128"))]
|
|
new_ucmd!()
|
|
.args(&["-n", "1Y", "emptyfile.txt"])
|
|
.fails()
|
|
.stderr_is("tail: invalid number of lines: '1Y': Value too large for defined data type");
|
|
#[cfg(target_pointer_width = "32")]
|
|
{
|
|
let sizes = ["1000G", "10T"];
|
|
for size in &sizes {
|
|
new_ucmd!()
|
|
.args(&["-c", size])
|
|
.fails()
|
|
.code_is(1)
|
|
.stderr_only(format!(
|
|
"tail: invalid number of bytes: '{}': Value too large for defined data type",
|
|
size
|
|
));
|
|
}
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn test_tail_num_with_undocumented_sign_bytes() {
|
|
// tail: '-' is not documented (8.32 man pages)
|
|
// head: '+' is not documented (8.32 man pages)
|
|
const ALPHABET: &str = "abcdefghijklmnopqrstuvwxyz";
|
|
new_ucmd!()
|
|
.args(&["-c", "5"])
|
|
.pipe_in(ALPHABET)
|
|
.succeeds()
|
|
.stdout_is("vwxyz");
|
|
new_ucmd!()
|
|
.args(&["-c", "-5"])
|
|
.pipe_in(ALPHABET)
|
|
.succeeds()
|
|
.stdout_is("vwxyz");
|
|
new_ucmd!()
|
|
.args(&["-c", "+5"])
|
|
.pipe_in(ALPHABET)
|
|
.succeeds()
|
|
.stdout_is("efghijklmnopqrstuvwxyz");
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(unix)]
|
|
fn test_tail_bytes_for_funny_files() {
|
|
// gnu/tests/tail-2/tail-c.sh
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
for &file in &["/proc/version", "/sys/kernel/profiling"] {
|
|
if !at.file_exists(file) {
|
|
continue;
|
|
}
|
|
let args = ["--bytes", "1", file];
|
|
let result = ts.ucmd().args(&args).run();
|
|
let exp_result = unwrap_or_return!(expected_result(&ts, &args));
|
|
result
|
|
.stdout_is(exp_result.stdout_str())
|
|
.stderr_is(exp_result.stderr_str())
|
|
.code_is(exp_result.code());
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_remove() {
|
|
// This test triggers a remove event while `tail --follow=name logfile` is running.
|
|
// ((sleep 1 && rm logfile &)>/dev/null 2>&1 &) ; tail --follow=name logfile
|
|
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
|
|
let source = FOLLOW_NAME_TXT;
|
|
let source_canonical = &at.plus(source);
|
|
|
|
let expected_stdout = at.read(FOLLOW_NAME_SHORT_EXP);
|
|
let expected_stderr = format!(
|
|
"{}: {}: No such file or directory\n{0}: no files remaining\n",
|
|
ts.util_name, source
|
|
);
|
|
|
|
let args = ["--follow=name", source];
|
|
let mut p = ts.ucmd().args(&args).run_no_wait();
|
|
|
|
let delay = 1000;
|
|
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::remove_file(source_canonical).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
|
|
p.kill().unwrap();
|
|
|
|
let mut buf_stdout = String::new();
|
|
let mut p_stdout = p.stdout.take().unwrap();
|
|
p_stdout.read_to_string(&mut buf_stdout).unwrap();
|
|
assert_eq!(buf_stdout, expected_stdout);
|
|
|
|
let mut buf_stderr = String::new();
|
|
let mut p_stderr = p.stderr.take().unwrap();
|
|
p_stderr.read_to_string(&mut buf_stderr).unwrap();
|
|
assert_eq!(buf_stderr, expected_stderr);
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_truncate() {
|
|
// This test triggers a truncate event while `tail --follow=name logfile` is running.
|
|
// cp logfile backup && head logfile > logfile && sleep 1 && cp backup logfile
|
|
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
|
|
let source = FOLLOW_NAME_TXT;
|
|
let source_canonical = &at.plus(source);
|
|
let backup = at.plus_as_string("backup");
|
|
|
|
let expected_stdout = at.read(FOLLOW_NAME_EXP);
|
|
let expected_stderr = format!("{}: {}: file truncated\n", ts.util_name, source);
|
|
|
|
let args = ["--follow=name", source];
|
|
let mut p = ts.ucmd().args(&args).run_no_wait();
|
|
|
|
let delay = 1000;
|
|
|
|
std::fs::copy(&source_canonical, &backup).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
let _ = std::fs::File::create(source_canonical).unwrap(); // trigger truncate
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::copy(&backup, &source_canonical).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
|
|
p.kill().unwrap();
|
|
|
|
let mut buf_stdout = String::new();
|
|
let mut p_stdout = p.stdout.take().unwrap();
|
|
p_stdout.read_to_string(&mut buf_stdout).unwrap();
|
|
assert_eq!(buf_stdout, expected_stdout);
|
|
|
|
let mut buf_stderr = String::new();
|
|
let mut p_stderr = p.stderr.take().unwrap();
|
|
p_stderr.read_to_string(&mut buf_stderr).unwrap();
|
|
assert_eq!(buf_stderr, expected_stderr);
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_remove_polling() {
|
|
// This test triggers a remove event while `tail --follow=name ---disable-inotify logfile` is running.
|
|
// ((sleep 1 && rm logfile &)>/dev/null 2>&1 &) ; tail --follow=name ---disable-inotify logfile
|
|
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
|
|
let source = FOLLOW_NAME_TXT;
|
|
let source_canonical = &at.plus(source);
|
|
|
|
let expected_stdout = at.read(FOLLOW_NAME_SHORT_EXP);
|
|
let expected_stderr = format!(
|
|
"{}: {}: No such file or directory\n{0}: no files remaining\n",
|
|
ts.util_name, source
|
|
);
|
|
|
|
let args = ["--follow=name", "--disable-inotify", source];
|
|
let mut p = ts.ucmd().args(&args).run_no_wait();
|
|
|
|
let delay = 1000;
|
|
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::remove_file(source_canonical).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
|
|
p.kill().unwrap();
|
|
|
|
let mut buf_stdout = String::new();
|
|
let mut p_stdout = p.stdout.take().unwrap();
|
|
p_stdout.read_to_string(&mut buf_stdout).unwrap();
|
|
assert_eq!(buf_stdout, expected_stdout);
|
|
|
|
let mut buf_stderr = String::new();
|
|
let mut p_stderr = p.stderr.take().unwrap();
|
|
p_stderr.read_to_string(&mut buf_stderr).unwrap();
|
|
assert_eq!(buf_stderr, expected_stderr);
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_move_create() {
|
|
// This test triggers a move/create event while `tail --follow=name logfile` is running.
|
|
// ((sleep 1 && mv logfile backup && sleep 1 && cp backup logfile &)>/dev/null 2>&1 &) ; tail --follow=name logfile
|
|
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
|
|
let source = FOLLOW_NAME_TXT;
|
|
let source_canonical = &at.plus(source);
|
|
let backup = at.plus_as_string("backup");
|
|
|
|
#[cfg(target_os = "linux")]
|
|
let expected_stdout = at.read(FOLLOW_NAME_EXP);
|
|
#[cfg(target_os = "linux")]
|
|
let expected_stderr = format!(
|
|
"{}: {}: No such file or directory\n{0}: '{1}' has appeared; following new file\n",
|
|
ts.util_name, source
|
|
);
|
|
|
|
// NOTE: We are less strict if not on Linux (inotify backend).
|
|
#[cfg(not(target_os = "linux"))]
|
|
let expected_stdout = at.read(FOLLOW_NAME_SHORT_EXP);
|
|
#[cfg(not(target_os = "linux"))]
|
|
let expected_stderr = format!("{}: {}: No such file or directory\n", ts.util_name, source);
|
|
|
|
let args = ["--follow=name", source];
|
|
let mut p = ts.ucmd().args(&args).run_no_wait();
|
|
|
|
let delay = 1000;
|
|
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::rename(&source_canonical, &backup).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::copy(&backup, &source_canonical).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
|
|
p.kill().unwrap();
|
|
|
|
let mut buf_stdout = String::new();
|
|
let mut p_stdout = p.stdout.take().unwrap();
|
|
p_stdout.read_to_string(&mut buf_stdout).unwrap();
|
|
assert_eq!(buf_stdout, expected_stdout);
|
|
|
|
let mut buf_stderr = String::new();
|
|
let mut p_stderr = p.stderr.take().unwrap();
|
|
p_stderr.read_to_string(&mut buf_stderr).unwrap();
|
|
assert_eq!(buf_stderr, expected_stderr);
|
|
}
|
|
|
|
#[test]
|
|
#[cfg(not(windows))]
|
|
fn test_follow_name_move_polling() {
|
|
// This test triggers a move event while `tail --follow=name --disable-inotify logfile` is running.
|
|
// ((sleep 1 && mv logfile backup && sleep 1 && cp backup logfile &)>/dev/null 2>&1 &) ; tail --follow=name ---disable-inotify logfile
|
|
// NOTE: GNU's tail does not recognize this move event for `---disable-inotify`
|
|
|
|
let ts = TestScenario::new(util_name!());
|
|
let at = &ts.fixtures;
|
|
|
|
let source = FOLLOW_NAME_TXT;
|
|
let source_canonical = &at.plus(source);
|
|
let backup = at.plus_as_string("backup");
|
|
|
|
let expected_stdout = at.read(FOLLOW_NAME_SHORT_EXP);
|
|
let expected_stderr = format!(
|
|
"{}: {}: No such file or directory\n{0}: no files remaining\n",
|
|
ts.util_name, source
|
|
);
|
|
|
|
let args = ["--follow=name", "--disable-inotify", source];
|
|
let mut p = ts.ucmd().args(&args).run_no_wait();
|
|
|
|
let delay = 1000;
|
|
|
|
sleep(Duration::from_millis(delay));
|
|
std::fs::rename(&source_canonical, &backup).unwrap();
|
|
sleep(Duration::from_millis(delay));
|
|
|
|
p.kill().unwrap();
|
|
|
|
let mut buf_stdout = String::new();
|
|
let mut p_stdout = p.stdout.take().unwrap();
|
|
p_stdout.read_to_string(&mut buf_stdout).unwrap();
|
|
assert_eq!(buf_stdout, expected_stdout);
|
|
|
|
let mut buf_stderr = String::new();
|
|
let mut p_stderr = p.stderr.take().unwrap();
|
|
p_stderr.read_to_string(&mut buf_stderr).unwrap();
|
|
assert_eq!(buf_stderr, expected_stderr);
|
|
}
|