#![allow(clippy::cast_possible_wrap)]
use uutests::at_and_ucmd;
use uutests::new_ucmd;
#[test]
#[cfg(target_os = "linux")]
fn test_tsort_non_utf8_paths() {
use std::os::unix::ffi::OsStringExt;
let (at, mut ucmd) = at_and_ucmd!();
let filename = std::ffi::OsString::from_vec(vec![0xFF, 0xFE]);
std::fs::write(at.plus(&filename), b"a b\nb c\n").unwrap();
ucmd.arg(&filename).succeeds().stdout_is("a\nb\nc\n");
}
#[test]
fn test_invalid_arg() {
new_ucmd!().arg("--definitely-invalid").fails_with_code(1);
}
#[test]
fn test_sort_call_graph() {
new_ucmd!()
.arg("call_graph.txt")
.succeeds()
.stdout_is_fixture("call_graph.expected");
}
#[test]
fn test_sort_self_loop() {
new_ucmd!()
.pipe_in("first first\nfirst second second second")
.succeeds()
.stdout_only("first\nsecond\n");
}
#[test]
fn test_sort_floating_nodes() {
new_ucmd!()
.pipe_in("d d\nc c\na a\nb b")
.succeeds()
.stdout_only("a\nb\nc\nd\n");
}
#[test]
fn test_no_such_file() {
new_ucmd!()
.arg("invalid_file_txt")
.fails()
.stderr_contains("No such file or directory");
}
#[test]
fn test_version_flag() {
let version_short = new_ucmd!().arg("-V").succeeds();
let version_long = new_ucmd!().arg("--version").succeeds();
assert_eq!(version_short.stdout_str(), version_long.stdout_str());
}
#[test]
fn test_help_flag() {
let help_short = new_ucmd!().arg("-h").succeeds();
let help_long = new_ucmd!().arg("--help").succeeds();
assert_eq!(help_short.stdout_str(), help_long.stdout_str());
}
#[test]
fn test_multiple_arguments() {
new_ucmd!()
.arg("call_graph.txt")
.arg("invalid_file")
.fails()
.stderr_contains("unexpected argument 'invalid_file' found");
}
#[test]
fn test_error_on_dir() {
let (at, mut ucmd) = at_and_ucmd!();
at.mkdir("tsort_test_dir");
ucmd.arg("tsort_test_dir")
.fails()
.stderr_contains("tsort: tsort_test_dir: read error: Is a directory");
}
#[test]
fn test_split_on_any_whitespace() {
new_ucmd!()
.pipe_in("a\nb\n")
.succeeds()
.stdout_only("a\nb\n");
}
#[test]
fn test_cycle() {
new_ucmd!()
.pipe_in("a b b c c d c b")
.fails_with_code(1)
.stdout_is("a\nb\nc\nd\n")
.stderr_is("tsort: -: input contains a loop:\ntsort: b\ntsort: c\n");
}
#[test]
fn test_two_cycles() {
new_ucmd!()
.pipe_in("a b b c c b b d d b")
.fails_with_code(1)
.stdout_is("a\nb\nc\nd\n")
.stderr_is("tsort: -: input contains a loop:\ntsort: b\ntsort: c\ntsort: -: input contains a loop:\ntsort: b\ntsort: d\n");
}
#[test]
fn test_long_loop_no_stack_overflow() {
use std::fmt::Write;
const N: usize = 100_000;
let mut input = String::new();
for v in 0..N {
let next = (v + 1) % N;
let _ = write!(input, "{v} {next} ");
}
new_ucmd!()
.pipe_in(input)
.fails_with_code(1)
.stderr_contains("tsort: -: input contains a loop");
}
#[test]
fn test_loop_for_iterative_dfs_correctness() {
let input = r"
A B
B C
C B
C D
D A
";
new_ucmd!()
.pipe_in(input)
.fails_with_code(1)
.stderr_contains("tsort: -: input contains a loop:\ntsort: B\ntsort: C");
}