From ccc38107dd521ccda7f9611353640460c2cd69f7 Mon Sep 17 00:00:00 2001 From: Jeffrey Finkelstein Date: Fri, 23 Sep 2022 15:56:44 -0400 Subject: [PATCH] cp: allow removing symbolic link loop destination Allow `cp --remove-destination` to remove a symbolic link loop (or a symbolic link that initiates a chain of too many symbolic links). Before this commit, if `loop` were a symbolic link to itself, then cp --remove-destination file loop would fail with an error message. After this commit, it succeeds. This matches the behaviotr of GNU cp. --- src/uu/cp/src/cp.rs | 13 ++++++++++++- tests/by-util/test_cp.rs | 13 +++++++++++++ 2 files changed, 25 insertions(+), 1 deletion(-) diff --git a/src/uu/cp/src/cp.rs b/src/uu/cp/src/cp.rs index 2c5222bc966..c71ca720f24 100644 --- a/src/uu/cp/src/cp.rs +++ b/src/uu/cp/src/cp.rs @@ -1375,6 +1375,17 @@ fn handle_existing_dest( Ok(()) } +/// Decide whether the given path exists. +fn file_or_link_exists(path: &Path) -> bool { + // Using `Path.exists()` or `Path.try_exists()` is not sufficient, + // because if `path` is a symbolic link and there are too many + // levels of symbolic link, then those methods will return false + // or an OS error. + std::fs::metadata(path) + .or_else(|_| std::fs::symlink_metadata(path)) + .is_ok() +} + /// Copy the a file from `source` to `dest`. `source` will be dereferenced if /// `options.dereference` is set to true. `dest` will be dereferenced only if /// the source was not a symlink. @@ -1392,7 +1403,7 @@ fn copy_file( symlinked_files: &mut HashSet, source_in_command_line: bool, ) -> CopyResult<()> { - if dest.exists() { + if file_or_link_exists(dest) { handle_existing_dest(source, dest, options, source_in_command_line)?; } diff --git a/tests/by-util/test_cp.rs b/tests/by-util/test_cp.rs index d6b00c3445d..4ceafd4ec38 100644 --- a/tests/by-util/test_cp.rs +++ b/tests/by-util/test_cp.rs @@ -2077,3 +2077,16 @@ fn test_cp_mode_hardlink_no_dereference() { assert!(at.symlink_exists("z")); assert_eq!(at.read_symlink("z"), "slink"); } + +#[test] +fn test_remove_destination_symbolic_link_loop() { + let (at, mut ucmd) = at_and_ucmd!(); + at.symlink_file("loop", "loop"); + at.plus("loop"); + at.touch("f"); + ucmd.args(&["--remove-destination", "f", "loop"]) + .succeeds() + .no_stdout() + .no_stderr(); + assert!(at.file_exists("loop")); +}