mirror of
https://github.com/RGBCube/uutils-coreutils
synced 2025-07-28 19:47:45 +00:00
Merge pull request #4859 from shinhs0506/cp-perm
cp: preserve permissions on -r -p --parents
This commit is contained in:
commit
24e0bef781
3 changed files with 61 additions and 29 deletions
|
@ -404,8 +404,18 @@ pub(crate) fn copy_directory(
|
||||||
Err(e) => show_error!("{}", e),
|
Err(e) => show_error!("{}", e),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy the attributes from the root directory to the target directory.
|
// Copy the attributes from the root directory to the target directory.
|
||||||
copy_attributes(root, target, &options.attributes)?;
|
if options.parents {
|
||||||
|
let dest = target.join(root.file_name().unwrap());
|
||||||
|
copy_attributes(root, dest.as_path(), &options.attributes)?;
|
||||||
|
for (x, y) in aligned_ancestors(root, dest.as_path()) {
|
||||||
|
copy_attributes(x, y, &options.attributes)?;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
copy_attributes(root, target, &options.attributes)?;
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1151,14 +1151,20 @@ fn copy_source(
|
||||||
} else {
|
} else {
|
||||||
// Copy as file
|
// Copy as file
|
||||||
let dest = construct_dest_path(source_path, target, target_type, options)?;
|
let dest = construct_dest_path(source_path, target, target_type, options)?;
|
||||||
copy_file(
|
let res = copy_file(
|
||||||
progress_bar,
|
progress_bar,
|
||||||
source_path,
|
source_path,
|
||||||
dest.as_path(),
|
dest.as_path(),
|
||||||
options,
|
options,
|
||||||
symlinked_files,
|
symlinked_files,
|
||||||
true,
|
true,
|
||||||
)
|
);
|
||||||
|
if options.parents {
|
||||||
|
for (x, y) in aligned_ancestors(source, dest.as_path()) {
|
||||||
|
copy_attributes(x, y, &options.attributes)?;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
res
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1704,11 +1710,6 @@ fn copy_file(
|
||||||
}
|
}
|
||||||
|
|
||||||
copy_attributes(source, dest, &options.attributes)?;
|
copy_attributes(source, dest, &options.attributes)?;
|
||||||
if options.parents && should_preserve_attribute(options) {
|
|
||||||
for (x, y) in aligned_ancestors(source, dest) {
|
|
||||||
copy_attributes(x, y, &options.attributes)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(progress_bar) = progress_bar {
|
if let Some(progress_bar) = progress_bar {
|
||||||
progress_bar.inc(fs::metadata(source)?.len());
|
progress_bar.inc(fs::metadata(source)?.len());
|
||||||
|
@ -1758,27 +1759,6 @@ fn copy_helper(
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn should_preserve_attribute(options: &Options) -> bool {
|
|
||||||
let checks = [
|
|
||||||
&options.attributes.mode,
|
|
||||||
&options.attributes.timestamps,
|
|
||||||
&options.attributes.links,
|
|
||||||
&options.attributes.context,
|
|
||||||
&options.attributes.xattr,
|
|
||||||
];
|
|
||||||
|
|
||||||
#[cfg(unix)]
|
|
||||||
let checks = [
|
|
||||||
checks.as_slice(),
|
|
||||||
[&options.attributes.ownership].as_slice(),
|
|
||||||
]
|
|
||||||
.concat();
|
|
||||||
|
|
||||||
checks
|
|
||||||
.iter()
|
|
||||||
.any(|attr| matches!(attr, Preserve::Yes { .. }))
|
|
||||||
}
|
|
||||||
|
|
||||||
// "Copies" a FIFO by creating a new one. This workaround is because Rust's
|
// "Copies" a FIFO by creating a new one. This workaround is because Rust's
|
||||||
// built-in fs::copy does not handle FIFOs (see rust-lang/rust/issues/79390).
|
// built-in fs::copy does not handle FIFOs (see rust-lang/rust/issues/79390).
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
|
|
|
@ -1115,6 +1115,48 @@ fn test_cp_parents_with_permissions_copy_file() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_cp_parents_with_permissions_copy_dir() {
|
||||||
|
let (at, mut ucmd) = at_and_ucmd!();
|
||||||
|
|
||||||
|
let dir1 = "dir";
|
||||||
|
let dir2 = "p1/p2";
|
||||||
|
let file = "p1/p2/file";
|
||||||
|
|
||||||
|
at.mkdir(dir1);
|
||||||
|
at.mkdir_all(dir2);
|
||||||
|
at.touch(file);
|
||||||
|
|
||||||
|
let p1_mode = 0o0777;
|
||||||
|
let p2_mode = 0o0711;
|
||||||
|
let file_mode = 0o0702;
|
||||||
|
|
||||||
|
#[cfg(unix)]
|
||||||
|
{
|
||||||
|
at.set_mode("p1", p1_mode);
|
||||||
|
at.set_mode("p1/p2", p2_mode);
|
||||||
|
at.set_mode(file, file_mode);
|
||||||
|
}
|
||||||
|
|
||||||
|
ucmd.arg("-p")
|
||||||
|
.arg("--parents")
|
||||||
|
.arg("-r")
|
||||||
|
.arg(dir2)
|
||||||
|
.arg(dir1)
|
||||||
|
.succeeds();
|
||||||
|
|
||||||
|
#[cfg(all(unix, not(target_os = "freebsd")))]
|
||||||
|
{
|
||||||
|
let p1_metadata = at.metadata("p1");
|
||||||
|
let p2_metadata = at.metadata("p1/p2");
|
||||||
|
let file_metadata = at.metadata(file);
|
||||||
|
|
||||||
|
assert_metadata_eq!(p1_metadata, at.metadata("dir/p1"));
|
||||||
|
assert_metadata_eq!(p2_metadata, at.metadata("dir/p1/p2"));
|
||||||
|
assert_metadata_eq!(file_metadata, at.metadata("dir/p1/p2/file"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[cfg(unix)]
|
#[cfg(unix)]
|
||||||
fn test_cp_writable_special_file_permissions() {
|
fn test_cp_writable_special_file_permissions() {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue