// 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 (words) autoformat nocheck use crate::common::util::TestScenario; #[cfg(any(target_os = "linux", target_os = "freebsd", target_os = "netbsd"))] use std::fs::OpenOptions; #[cfg(unix)] use std::{ffi::OsStr, os::unix::ffi::OsStrExt}; #[cfg(windows)] use std::{ffi::OsString, os::windows::ffi::OsStringExt}; #[test] fn test_invalid_arg() { new_ucmd!().arg("--definitely-invalid").fails().code_is(1); } #[test] fn empty_files() { new_ucmd!() .arg("empty.txt") .arg("empty.txt") .succeeds() .stdout_only(""); new_ucmd!() .arg("empty.txt") .arg("fields_1.txt") .succeeds() .stdout_only(""); new_ucmd!() .arg("fields_1.txt") .arg("empty.txt") .succeeds() .stdout_only(""); } #[test] fn empty_intersection() { new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .arg("-2") .arg("2") .succeeds() .stdout_only(""); } #[test] fn default_arguments() { new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .succeeds() .stdout_only_fixture("default.expected"); } #[test] fn different_fields() { new_ucmd!() .arg("fields_2.txt") .arg("fields_4.txt") .arg("-j") .arg("2") .succeeds() .stdout_only_fixture("different_fields.expected"); new_ucmd!() .arg("fields_2.txt") .arg("fields_4.txt") .arg("-1") .arg("2") .arg("-2") .arg("2") .succeeds() .stdout_only_fixture("different_fields.expected"); } #[test] fn different_field() { new_ucmd!() .arg("fields_2.txt") .arg("fields_3.txt") .arg("-2") .arg("2") .succeeds() .stdout_only_fixture("different_field.expected"); } #[test] fn out_of_bounds_fields() { new_ucmd!() .arg("fields_1.txt") .arg("fields_4.txt") .arg("-1") .arg("3") .arg("-2") .arg("5") .succeeds() .stdout_only_fixture("out_of_bounds_fields.expected"); new_ucmd!() .arg("fields_1.txt") .arg("fields_4.txt") .arg("-j") .arg("100000000000000000000") // > usize::MAX for 64 bits .succeeds() .stdout_only_fixture("out_of_bounds_fields.expected"); } #[test] fn unpaired_lines() { new_ucmd!() .arg("fields_2.txt") .arg("fields_3.txt") .arg("-a") .arg("1") .succeeds() .stdout_only_fixture("fields_2.txt"); new_ucmd!() .arg("fields_3.txt") .arg("fields_2.txt") .arg("-1") .arg("2") .arg("-a") .arg("2") .succeeds() .stdout_only_fixture("unpaired_lines.expected"); new_ucmd!() .arg("fields_3.txt") .arg("fields_2.txt") .arg("-1") .arg("2") .arg("-a") .arg("1") .arg("-a") .arg("2") .succeeds() .stdout_only_fixture("unpaired_lines_outer.expected"); } #[test] fn suppress_joined() { new_ucmd!() .arg("fields_3.txt") .arg("fields_2.txt") .arg("-1") .arg("2") .arg("-v") .arg("2") .succeeds() .stdout_only_fixture("suppress_joined.expected"); new_ucmd!() .arg("fields_3.txt") .arg("fields_2.txt") .arg("-1") .arg("2") .arg("-a") .arg("1") .arg("-v") .arg("2") .succeeds() .stdout_only_fixture("suppress_joined_outer.expected"); } #[test] fn case_insensitive() { new_ucmd!() .arg("capitalized.txt") .arg("fields_3.txt") .arg("-i") .succeeds() .stdout_only_fixture("case_insensitive.expected"); } #[test] fn semicolon_separated() { new_ucmd!() .arg("semicolon_fields_1.txt") .arg("semicolon_fields_2.txt") .arg("-t") .arg(";") .succeeds() .stdout_only_fixture("semicolon_separated.expected"); } #[test] fn new_line_separated() { new_ucmd!() .arg("-") .arg("fields_2.txt") .arg("-t") .arg("") .pipe_in("1 a\n1 b\n8 h\n") .succeeds() .stdout_only("1 a\n8 h\n"); } #[test] fn tab_multi_character() { new_ucmd!() .arg("semicolon_fields_1.txt") .arg("semicolon_fields_2.txt") .arg("-t") .arg("э") .fails() .stderr_is("join: multi-character tab э\n"); } #[test] fn default_format() { new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .arg("-o") .arg("1.1 2.2") .succeeds() .stdout_only_fixture("default.expected"); new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .arg("-o") .arg("0 2.2") .succeeds() .stdout_only_fixture("default.expected"); } #[test] fn unpaired_lines_format() { new_ucmd!() .arg("fields_2.txt") .arg("fields_3.txt") .arg("-a") .arg("2") .arg("-o") .arg("1.2 1.1 2.4 2.3 2.2 0") .succeeds() .stdout_only_fixture("unpaired_lines_format.expected"); } #[test] fn autoformat() { new_ucmd!() .arg("fields_2.txt") .arg("different_lengths.txt") .arg("-o") .arg("auto") .succeeds() .stdout_only_fixture("autoformat.expected"); new_ucmd!() .arg("-") .arg("fields_2.txt") .arg("-o") .arg("auto") .pipe_in("1 x y z\n2 p") .succeeds() .stdout_only("1 x y z a\n2 p b\n"); new_ucmd!() .arg("-") .arg("fields_2.txt") .arg("-a") .arg("1") .arg("-o") .arg("auto") .arg("-e") .arg(".") .pipe_in("1 x y z\n2 p\n99 a b\n") .succeeds() .stdout_only("1 x y z a\n2 p . . b\n99 a b . .\n"); } #[test] fn empty_format() { new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .arg("-o") .arg("") .fails() .stderr_is("join: invalid file number in field spec: ''\n"); } #[test] fn empty_key() { new_ucmd!() .arg("fields_1.txt") .arg("empty.txt") .arg("-j") .arg("2") .arg("-a") .arg("1") .arg("-e") .arg("x") .succeeds() .stdout_only_fixture("empty_key.expected"); } #[test] fn missing_format_fields() { new_ucmd!() .arg("fields_2.txt") .arg("different_lengths.txt") .arg("-o") .arg("0 1.2 2.4") .arg("-e") .arg("x") .succeeds() .stdout_only_fixture("missing_format_fields.expected"); } #[test] fn nocheck_order() { new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .arg("--nocheck-order") .succeeds() .stdout_only_fixture("default.expected"); } #[test] fn wrong_line_order() { let ts = TestScenario::new(util_name!()); new_ucmd!() .arg("fields_2.txt") .arg("fields_4.txt") .fails() .stdout_contains("7 g f 4 fg") .stderr_is(format!( "{0} {1}: fields_4.txt:5: is not sorted: 11 g 5 gh\n{0} {1}: input is not in sorted order\n", ts.bin_path.to_string_lossy(), ts.util_name )); new_ucmd!() .arg("--check-order") .arg("fields_2.txt") .arg("fields_4.txt") .fails() .stdout_does_not_contain("7 g f 4 fg") .stderr_is(format!( "{0}: fields_4.txt:5: is not sorted: 11 g 5 gh\n", ts.util_name )); } #[test] fn both_files_wrong_line_order() { let ts = TestScenario::new(util_name!()); new_ucmd!() .arg("fields_4.txt") .arg("fields_5.txt") .fails() .stdout_contains("5 e 3 ef") .stderr_is(format!( "{0} {1}: fields_5.txt:4: is not sorted: 3\n{0} {1}: fields_4.txt:5: is not sorted: 11 g 5 gh\n{0} {1}: input is not in sorted order\n", ts.bin_path.to_string_lossy(), ts.util_name )); new_ucmd!() .arg("--check-order") .arg("fields_4.txt") .arg("fields_5.txt") .fails() .stdout_does_not_contain("5 e 3 ef") .stderr_is(format!( "{0}: fields_5.txt:4: is not sorted: 3\n", ts.util_name )); } #[test] fn headers() { new_ucmd!() .arg("header_1.txt") .arg("header_2.txt") .arg("--header") .succeeds() .stdout_only_fixture("header.expected"); } #[test] fn headers_autoformat() { new_ucmd!() .arg("header_1.txt") .arg("header_2.txt") .arg("--header") .arg("-o") .arg("auto") .succeeds() .stdout_only_fixture("header_autoformat.expected"); } #[test] fn single_file_with_header() { new_ucmd!() .arg("capitalized.txt") .arg("empty.txt") .arg("--header") .succeeds() .stdout_is("A 1\n"); new_ucmd!() .arg("empty.txt") .arg("capitalized.txt") .arg("--header") .succeeds() .stdout_is("A 1\n"); } #[test] fn non_line_feeds() { new_ucmd!() .arg("non-line_feeds_1.txt") .arg("non-line_feeds_2.txt") .succeeds() .stdout_only_fixture("non-line_feeds.expected"); } #[test] fn non_unicode() { new_ucmd!() .arg("non-unicode_1.bin") .arg("non-unicode_2.bin") .succeeds() .stdout_only_fixture("non-unicode.expected"); #[cfg(unix)] { let invalid_utf8: u8 = 167; new_ucmd!() .arg("-t") .arg(OsStr::from_bytes(&[invalid_utf8])) .arg("non-unicode_1.bin") .arg("non-unicode_2.bin") .succeeds() .stdout_only_fixture("non-unicode_sep.expected"); } #[cfg(windows)] { let invalid_utf16: OsString = OsStringExt::from_wide(&[0xD800]); new_ucmd!() .arg("-t") .arg(&invalid_utf16) .arg("non-unicode_1.bin") .arg("non-unicode_2.bin") .fails() .stderr_is( "join: unprintable field separators are only supported on unix-like platforms\n", ); } } #[test] fn null_field_separators() { new_ucmd!() .arg("-t") .arg("\\0") .arg("non-unicode_1.bin") .arg("non-unicode_2.bin") .succeeds() .stdout_only_fixture("null-sep.expected"); } #[test] fn null_line_endings() { new_ucmd!() .arg("-z") .arg("non-unicode_1.bin") .arg("non-unicode_2.bin") .succeeds() .stdout_only_fixture("z.expected"); } #[test] #[cfg(any(target_os = "linux", target_os = "freebsd", target_os = "netbsd"))] fn test_full() { let dev_full = OpenOptions::new().write(true).open("/dev/full").unwrap(); new_ucmd!() .arg("fields_1.txt") .arg("fields_2.txt") .set_stdout(dev_full) .fails() .stderr_contains("No space left on device"); }