cp: --preserve should keep xattr (#5834)

* cp: --preserve should keep xattr

Should help with tests/cp/acl.sh

* Update tests/by-util/test_cp.rs

Co-authored-by: Daniel Hofstetter <daniel.hofstetter@42dh.com>

* Update tests/by-util/test_cp.rs

Co-authored-by: Daniel Hofstetter <daniel.hofstetter@42dh.com>

* Update tests/by-util/test_cp.rs

Co-authored-by: Daniel Hofstetter <daniel.hofstetter@42dh.com>

---------

Co-authored-by: Daniel Hofstetter <daniel.hofstetter@42dh.com>
This commit is contained in:
Sylvestre Ledru 2024-01-15 10:59:04 +01:00 committed by GitHub
parent e340d8177e
commit fff83995fb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 62 additions and 3 deletions

1
Cargo.lock generated
View file

@ -507,6 +507,7 @@ dependencies = [
"uucore", "uucore",
"uuhelp_parser", "uuhelp_parser",
"walkdir", "walkdir",
"xattr",
"zip", "zip",
] ]

View file

@ -500,6 +500,7 @@ rlimit = "0.10.1"
[target.'cfg(unix)'.dev-dependencies] [target.'cfg(unix)'.dev-dependencies]
nix = { workspace = true, features = ["process", "signal", "user"] } nix = { workspace = true, features = ["process", "signal", "user"] }
rand_pcg = "0.3" rand_pcg = "0.3"
xattr = { workspace = true }
[build-dependencies] [build-dependencies]
phf_codegen = { workspace = true } phf_codegen = { workspace = true }

View file

@ -826,6 +826,7 @@ impl Attributes {
ownership: Preserve::Yes { required: true }, ownership: Preserve::Yes { required: true },
mode: Preserve::Yes { required: true }, mode: Preserve::Yes { required: true },
timestamps: Preserve::Yes { required: true }, timestamps: Preserve::Yes { required: true },
xattr: Preserve::Yes { required: true },
..Self::NONE ..Self::NONE
}; };
@ -1441,7 +1442,7 @@ pub(crate) fn copy_attributes(
})?; })?;
handle_preserve(&attributes.xattr, || -> CopyResult<()> { handle_preserve(&attributes.xattr, || -> CopyResult<()> {
#[cfg(unix)] #[cfg(all(unix, not(target_os = "android")))]
{ {
let xattrs = xattr::list(source)?; let xattrs = xattr::list(source)?;
for attr in xattrs { for attr in xattrs {
@ -1450,7 +1451,7 @@ pub(crate) fn copy_attributes(
} }
} }
} }
#[cfg(not(unix))] #[cfg(not(all(unix, not(target_os = "android"))))]
{ {
// The documentation for GNU cp states: // The documentation for GNU cp states:
// //

View file

@ -2,7 +2,7 @@
// //
// For the full copyright and license information, please view the LICENSE // For the full copyright and license information, please view the LICENSE
// file that was distributed with this source code. // file that was distributed with this source code.
// spell-checker:ignore (flags) reflink (fs) tmpfs (linux) rlimit Rlim NOFILE clob btrfs ROOTDIR USERDIR procfs outfile uufs // spell-checker:ignore (flags) reflink (fs) tmpfs (linux) rlimit Rlim NOFILE clob btrfs ROOTDIR USERDIR procfs outfile uufs xattrs
use crate::common::util::TestScenario; use crate::common::util::TestScenario;
#[cfg(not(windows))] #[cfg(not(windows))]
@ -56,6 +56,8 @@ static TEST_MOUNT_MOUNTPOINT: &str = "mount";
static TEST_MOUNT_OTHER_FILESYSTEM_FILE: &str = "mount/DO_NOT_copy_me.txt"; static TEST_MOUNT_OTHER_FILESYSTEM_FILE: &str = "mount/DO_NOT_copy_me.txt";
#[cfg(unix)] #[cfg(unix)]
static TEST_NONEXISTENT_FILE: &str = "nonexistent_file.txt"; static TEST_NONEXISTENT_FILE: &str = "nonexistent_file.txt";
#[cfg(all(unix, not(any(target_os = "android", target_os = "macos"))))]
use xattr;
/// Assert that mode, ownership, and permissions of two metadata objects match. /// Assert that mode, ownership, and permissions of two metadata objects match.
#[cfg(all(not(windows), not(target_os = "freebsd")))] #[cfg(all(not(windows), not(target_os = "freebsd")))]
@ -3736,3 +3738,57 @@ fn test_cp_no_such() {
.fails() .fails()
.stderr_is("cp: 'no-such/' is not a directory\n"); .stderr_is("cp: 'no-such/' is not a directory\n");
} }
#[cfg(all(unix, not(any(target_os = "android", target_os = "macos"))))]
fn compare_xattrs<P: AsRef<Path>>(path1: P, path2: P) -> bool {
let get_sorted_xattrs = |path: P| {
xattr::list(path)
.map(|attrs| {
let mut attrs = attrs.collect::<Vec<_>>();
attrs.sort();
attrs
})
.unwrap_or_else(|_| Vec::new())
};
get_sorted_xattrs(path1) == get_sorted_xattrs(path2)
}
#[cfg(all(unix, not(any(target_os = "android", target_os = "macos"))))]
#[test]
fn test_acl_preserve() {
use std::process::Command;
let scene = TestScenario::new(util_name!());
let at = &scene.fixtures;
let path1 = "a";
let path2 = "b";
let file = "a/file";
let file_target = "b/file";
at.mkdir(path1);
at.mkdir(path2);
at.touch(file);
let path = at.plus_as_string(file);
// calling the command directly. xattr requires some dev packages to be installed
// and it adds a complex dependency just for a test
match Command::new("setfacl")
.args(["-m", "group::rwx", &path1])
.status()
.map(|status| status.code())
{
Ok(Some(0)) => {}
Ok(_) => {
println!("test skipped: setfacl failed");
return;
}
Err(e) => {
println!("test skipped: setfacl failed with {}", e);
return;
}
}
scene.ucmd().args(&["-p", &path, path2]).succeeds();
assert!(compare_xattrs(&file, &file_target));
}