use crate::common::util::TestScenario; #[cfg(not(windows))] use libc::{mode_t, umask}; use once_cell::sync::Lazy; #[cfg(not(windows))] use std::os::unix::fs::PermissionsExt; use std::sync::Mutex; // tests in `test_mkdir.rs` cannot run in parallel since some tests alter the umask. This may cause // other tests to run under a wrong set of permissions // // when writing a test case, acquire this mutex before proceeding with the main logic of the test static TEST_MUTEX: Lazy> = Lazy::new(|| Mutex::new(())); static TEST_DIR1: &str = "mkdir_test1"; static TEST_DIR2: &str = "mkdir_test2"; static TEST_DIR3: &str = "mkdir_test3"; static TEST_DIR4: &str = "mkdir_test4/mkdir_test4_1"; static TEST_DIR5: &str = "mkdir_test5/mkdir_test5_1"; static TEST_DIR6: &str = "mkdir_test6"; static TEST_FILE7: &str = "mkdir_test7"; static TEST_DIR8: &str = "mkdir_test8/mkdir_test8_1/mkdir_test8_2"; static TEST_DIR9: &str = "mkdir_test9/../mkdir_test9_1/../mkdir_test9_2"; static TEST_DIR10: &str = "mkdir_test10/."; static TEST_DIR11: &str = "mkdir_test11/.."; #[cfg(not(windows))] static TEST_DIR12: &str = "mkdir_test12"; #[test] fn test_invalid_arg() { let _guard = TEST_MUTEX.lock(); new_ucmd!().arg("--definitely-invalid").fails().code_is(1); } #[test] fn test_mkdir_mkdir() { let _guard = TEST_MUTEX.lock(); new_ucmd!().arg(TEST_DIR1).succeeds(); } #[test] fn test_mkdir_verbose() { let _guard = TEST_MUTEX.lock(); let expected = "mkdir: created directory 'mkdir_test1'\n"; new_ucmd!() .arg(TEST_DIR1) .arg("-v") .run() .stdout_is(expected); } #[test] fn test_mkdir_dup_dir() { let _guard = TEST_MUTEX.lock(); let scene = TestScenario::new(util_name!()); scene.ucmd().arg(TEST_DIR2).succeeds(); scene.ucmd().arg(TEST_DIR2).fails(); } #[test] fn test_mkdir_mode() { let _guard = TEST_MUTEX.lock(); new_ucmd!().arg("-m").arg("755").arg(TEST_DIR3).succeeds(); } #[test] fn test_mkdir_parent() { let _guard = TEST_MUTEX.lock(); let scene = TestScenario::new(util_name!()); scene.ucmd().arg("-p").arg(TEST_DIR4).succeeds(); scene.ucmd().arg("-p").arg(TEST_DIR4).succeeds(); scene.ucmd().arg("--parent").arg(TEST_DIR4).succeeds(); scene.ucmd().arg("--parents").arg(TEST_DIR4).succeeds(); } #[test] fn test_mkdir_no_parent() { let _guard = TEST_MUTEX.lock(); new_ucmd!().arg(TEST_DIR5).fails(); } #[test] fn test_mkdir_dup_dir_parent() { let _guard = TEST_MUTEX.lock(); let scene = TestScenario::new(util_name!()); scene.ucmd().arg(TEST_DIR6).succeeds(); scene.ucmd().arg("-p").arg(TEST_DIR6).succeeds(); } #[cfg(not(windows))] #[test] fn test_mkdir_parent_mode() { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); let default_umask: mode_t = 0o160; let original_umask = unsafe { umask(default_umask) }; ucmd.arg("-p").arg("a/b").succeeds().no_stderr().no_stdout(); assert!(at.dir_exists("a")); // parents created by -p have permissions set to "=rwx,u+wx" assert_eq!( at.metadata("a").permissions().mode() as mode_t, ((!default_umask & 0o777) | 0o300) + 0o40000 ); assert!(at.dir_exists("a/b")); // sub directory's permission is determined only by the umask assert_eq!( at.metadata("a/b").permissions().mode() as mode_t, (!default_umask & 0o777) + 0o40000 ); unsafe { umask(original_umask); } } #[cfg(not(windows))] #[test] fn test_mkdir_parent_mode_check_existing_parent() { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); at.mkdir("a"); let default_umask: mode_t = 0o160; let original_umask = unsafe { umask(default_umask) }; ucmd.arg("-p") .arg("a/b/c") .succeeds() .no_stderr() .no_stdout(); assert!(at.dir_exists("a")); // parent dirs that already exist do not get their permissions modified assert_eq!( at.metadata("a").permissions().mode() as mode_t, (!original_umask & 0o777) + 0o40000 ); assert!(at.dir_exists("a/b")); assert_eq!( at.metadata("a/b").permissions().mode() as mode_t, ((!default_umask & 0o777) | 0o300) + 0o40000 ); assert!(at.dir_exists("a/b/c")); assert_eq!( at.metadata("a/b/c").permissions().mode() as mode_t, (!default_umask & 0o777) + 0o40000 ); unsafe { umask(original_umask); } } #[test] fn test_mkdir_dup_file() { let _guard = TEST_MUTEX.lock(); let scene = TestScenario::new(util_name!()); scene.fixtures.touch(TEST_FILE7); scene.ucmd().arg(TEST_FILE7).fails(); // mkdir should fail for a file even if -p is specified. scene.ucmd().arg("-p").arg(TEST_FILE7).fails(); } #[test] #[cfg(not(windows))] fn test_symbolic_mode() { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); ucmd.arg("-m").arg("a=rwx").arg(TEST_DIR1).succeeds(); let perms = at.metadata(TEST_DIR1).permissions().mode(); assert_eq!(perms, 0o40777); } #[test] #[cfg(not(windows))] fn test_symbolic_alteration() { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); let default_umask = 0o022; let original_umask = unsafe { umask(default_umask) }; ucmd.arg("-m").arg("-w").arg(TEST_DIR1).succeeds(); let perms = at.metadata(TEST_DIR1).permissions().mode(); assert_eq!(perms, 0o40577); unsafe { umask(original_umask) }; } #[test] #[cfg(not(windows))] fn test_multi_symbolic() { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); ucmd.arg("-m") .arg("u=rwx,g=rx,o=") .arg(TEST_DIR1) .succeeds(); let perms = at.metadata(TEST_DIR1).permissions().mode(); assert_eq!(perms, 0o40750); } #[test] fn test_recursive_reporting() { let _guard = TEST_MUTEX.lock(); new_ucmd!() .arg("-p") .arg("-v") .arg(TEST_DIR8) .succeeds() .stdout_contains("created directory 'mkdir_test8'") .stdout_contains("created directory 'mkdir_test8/mkdir_test8_1'") .stdout_contains("created directory 'mkdir_test8/mkdir_test8_1/mkdir_test8_2'"); new_ucmd!().arg("-v").arg(TEST_DIR8).fails().no_stdout(); new_ucmd!() .arg("-p") .arg("-v") .arg(TEST_DIR9) .succeeds() .stdout_contains("created directory 'mkdir_test9'") .stdout_contains("created directory 'mkdir_test9/../mkdir_test9_1'") .stdout_contains("created directory 'mkdir_test9/../mkdir_test9_1/../mkdir_test9_2'"); } #[test] fn test_mkdir_trailing_dot() { let _guard = TEST_MUTEX.lock(); let scene2 = TestScenario::new("ls"); new_ucmd!() .arg("-p") .arg("-v") .arg("mkdir_test10-2") .succeeds(); new_ucmd!() .arg("-p") .arg("-v") .arg(TEST_DIR10) .succeeds() .stdout_contains("created directory 'mkdir_test10'"); new_ucmd!() .arg("-p") .arg("-v") .arg(TEST_DIR11) .succeeds() .stdout_contains("created directory 'mkdir_test11'"); let result = scene2.ucmd().arg("-al").run(); println!("ls dest {}", result.stdout_str()); } #[test] #[cfg(not(windows))] fn test_umask_compliance() { fn test_single_case(umask_set: mode_t) { let _guard = TEST_MUTEX.lock(); let (at, mut ucmd) = at_and_ucmd!(); let original_umask = unsafe { umask(umask_set) }; ucmd.arg(TEST_DIR12).succeeds(); let perms = at.metadata(TEST_DIR12).permissions().mode() as mode_t; assert_eq!(perms, (!umask_set & 0o0777) + 0o40000); // before compare, add the set GUID, UID bits unsafe { umask(original_umask); } // set umask back to original } for i in 0o0..0o027 { // tests all permission combinations test_single_case(i as mode_t); } }