250 lines
6.4 KiB
Rust
250 lines
6.4 KiB
Rust
|
|
use std::collections::BTreeSet;
|
||
|
|
use std::ffi::OsStr;
|
||
|
|
use xattr::FileExt;
|
||
|
|
|
||
|
|
use tempfile::{tempfile, tempfile_in, NamedTempFile};
|
||
|
|
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_fd() {
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
let tmp = tempfile().unwrap();
|
||
|
|
assert!(tmp.get_xattr("user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
tmp.list_xattr()
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
|
||
|
|
tmp.set_xattr("user.test", b"my test").unwrap();
|
||
|
|
assert_eq!(tmp.get_xattr("user.test").unwrap().unwrap(), b"my test");
|
||
|
|
assert_eq!(
|
||
|
|
tmp.list_xattr()
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.collect::<Vec<_>>(),
|
||
|
|
vec![OsStr::new("user.test")]
|
||
|
|
);
|
||
|
|
|
||
|
|
tmp.remove_xattr("user.test").unwrap();
|
||
|
|
assert!(tmp.get_xattr("user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
tmp.list_xattr()
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
}
|
||
|
|
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_path() {
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
let tmp = NamedTempFile::new().unwrap();
|
||
|
|
assert!(xattr::get(tmp.path(), "user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
|
||
|
|
xattr::set(tmp.path(), "user.test", b"my test").unwrap();
|
||
|
|
assert_eq!(
|
||
|
|
xattr::get(tmp.path(), "user.test").unwrap().unwrap(),
|
||
|
|
b"my test"
|
||
|
|
);
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.collect::<Vec<_>>(),
|
||
|
|
vec![OsStr::new("user.test")]
|
||
|
|
);
|
||
|
|
|
||
|
|
xattr::remove(tmp.path(), "user.test").unwrap();
|
||
|
|
assert!(xattr::get(tmp.path(), "user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
}
|
||
|
|
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_missing() {
|
||
|
|
assert!(xattr::get("/var/empty/does-not-exist", "user.test").is_err());
|
||
|
|
}
|
||
|
|
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_debug() {
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
|
||
|
|
// Only works on "real" filesystems.
|
||
|
|
let tmp = tempfile().unwrap();
|
||
|
|
|
||
|
|
tmp.set_xattr("user.myattr", b"value").unwrap();
|
||
|
|
let mut attrs = tmp.list_xattr().unwrap();
|
||
|
|
|
||
|
|
let debugstr = format!("{:?}", attrs);
|
||
|
|
|
||
|
|
// Debug is idempotent
|
||
|
|
assert_eq!(debugstr, format!("{:?}", attrs));
|
||
|
|
|
||
|
|
// It produces the right value. We can't just assert that it's equal to the expected value as
|
||
|
|
// the system may have added additional attributes (selinux, etc.). See #68.
|
||
|
|
assert!(debugstr.starts_with("XAttrs(["));
|
||
|
|
assert!(debugstr.ends_with("])"));
|
||
|
|
assert!(debugstr.contains(r#""user.myattr""#));
|
||
|
|
|
||
|
|
// It doesn't affect the underlying iterator.
|
||
|
|
assert_eq!(
|
||
|
|
"user.myattr",
|
||
|
|
attrs.find(|x| x.as_bytes().starts_with(b"user.")).unwrap()
|
||
|
|
);
|
||
|
|
|
||
|
|
// drain it.
|
||
|
|
let _ = attrs.by_ref().count();
|
||
|
|
|
||
|
|
// An empty iterator produces the right value.
|
||
|
|
assert_eq!(r#"XAttrs([])"#, format!("{:?}", attrs));
|
||
|
|
}
|
||
|
|
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_multi() {
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
// Only works on "real" filesystems.
|
||
|
|
let tmp = tempfile().unwrap();
|
||
|
|
let mut items: BTreeSet<_> = [
|
||
|
|
OsStr::new("user.test1"),
|
||
|
|
OsStr::new("user.test2"),
|
||
|
|
OsStr::new("user.test3"),
|
||
|
|
]
|
||
|
|
.iter()
|
||
|
|
.cloned()
|
||
|
|
.collect();
|
||
|
|
|
||
|
|
for it in &items {
|
||
|
|
tmp.set_xattr(it, b"value").unwrap();
|
||
|
|
}
|
||
|
|
for it in tmp
|
||
|
|
.list_xattr()
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
{
|
||
|
|
assert!(items.remove(&*it));
|
||
|
|
}
|
||
|
|
assert!(items.is_empty());
|
||
|
|
}
|
||
|
|
|
||
|
|
// This test is skipped on android because the /data/tmp filesystem doesn't support >4kib of
|
||
|
|
// extended attributes.
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(target_os = "linux", target_os = "freebsd", target_os = "macos"))]
|
||
|
|
fn test_large() {
|
||
|
|
use std::ffi::OsString;
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
// On Linux, this only works in tmpfs!
|
||
|
|
let tmp = if cfg!(target_os = "linux") {
|
||
|
|
tempfile_in("/dev/shm")
|
||
|
|
} else {
|
||
|
|
tempfile()
|
||
|
|
}
|
||
|
|
.unwrap();
|
||
|
|
let mut items: BTreeSet<OsString> = (0..100)
|
||
|
|
.map(|i| format!("user.test{i:0100}").into())
|
||
|
|
.collect();
|
||
|
|
|
||
|
|
for it in &items {
|
||
|
|
tmp.set_xattr(it, b"value").unwrap();
|
||
|
|
}
|
||
|
|
for it in tmp
|
||
|
|
.list_xattr()
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
{
|
||
|
|
assert!(items.remove(&*it));
|
||
|
|
}
|
||
|
|
assert!(items.is_empty());
|
||
|
|
}
|
||
|
|
|
||
|
|
// Tests the deref API variants - regression test for
|
||
|
|
// https://github.com/Stebalien/xattr/issues/57
|
||
|
|
#[test]
|
||
|
|
#[cfg(any(
|
||
|
|
target_os = "linux",
|
||
|
|
target_os = "freebsd",
|
||
|
|
target_os = "macos",
|
||
|
|
target_os = "android"
|
||
|
|
))]
|
||
|
|
fn test_path_deref() {
|
||
|
|
use std::os::unix::ffi::OsStrExt;
|
||
|
|
// Only works on "real" filesystems.
|
||
|
|
let tmp = NamedTempFile::new().unwrap();
|
||
|
|
assert!(xattr::get_deref(tmp.path(), "user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list_deref(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
|
||
|
|
xattr::set_deref(tmp.path(), "user.test", b"my test").unwrap();
|
||
|
|
assert_eq!(
|
||
|
|
xattr::get_deref(tmp.path(), "user.test").unwrap().unwrap(),
|
||
|
|
b"my test"
|
||
|
|
);
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list_deref(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.collect::<Vec<_>>(),
|
||
|
|
vec![OsStr::new("user.test")]
|
||
|
|
);
|
||
|
|
|
||
|
|
xattr::remove_deref(tmp.path(), "user.test").unwrap();
|
||
|
|
assert!(xattr::get_deref(tmp.path(), "user.test").unwrap().is_none());
|
||
|
|
assert_eq!(
|
||
|
|
xattr::list_deref(tmp.path())
|
||
|
|
.unwrap()
|
||
|
|
.filter(|x| x.as_bytes().starts_with(b"user."))
|
||
|
|
.count(),
|
||
|
|
0
|
||
|
|
);
|
||
|
|
}
|