2018-06-30 14:55:33 +02:00
|
|
|
use anyhow::{anyhow, Result};
|
2018-06-30 21:57:20 +02:00
|
|
|
use std::fs;
|
2018-06-30 14:55:33 +02:00
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq)]
|
|
|
|
pub struct OwnerFilter {
|
|
|
|
uid: Option<u32>,
|
|
|
|
gid: Option<u32>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl OwnerFilter {
|
|
|
|
pub fn from_string(input: &str) -> Result<Self> {
|
|
|
|
let mut it = input.split(':');
|
|
|
|
let (fst, snd) = (it.next(), it.next());
|
|
|
|
|
|
|
|
let uid = match fst {
|
|
|
|
Some("") | None => None,
|
2018-06-30 21:57:20 +02:00
|
|
|
Some(s) => {
|
|
|
|
let maybe_uid = s
|
|
|
|
.parse()
|
|
|
|
.ok()
|
|
|
|
.or_else(|| users::get_user_by_name(s).map(|user| user.uid()));
|
|
|
|
match maybe_uid {
|
|
|
|
Some(uid) => Some(uid),
|
|
|
|
_ => return Err(anyhow!("'{}' is not a recognized user name", s)),
|
|
|
|
}
|
|
|
|
}
|
2018-06-30 14:55:33 +02:00
|
|
|
};
|
|
|
|
let gid = match snd {
|
|
|
|
Some("") | None => None,
|
2018-06-30 21:57:20 +02:00
|
|
|
Some(s) => {
|
|
|
|
let maybe_gid = s
|
|
|
|
.parse()
|
|
|
|
.ok()
|
|
|
|
.or_else(|| users::get_group_by_name(s).map(|group| group.gid()));
|
|
|
|
match maybe_gid {
|
|
|
|
Some(gid) => Some(gid),
|
|
|
|
_ => return Err(anyhow!("'{}' is not a recognized group name", s)),
|
|
|
|
}
|
|
|
|
}
|
2018-06-30 14:55:33 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
if uid.is_none() && gid.is_none() {
|
|
|
|
Err(anyhow!(
|
|
|
|
"'{}' is not a valid user/group specifier. See 'fd --help'.",
|
|
|
|
input
|
|
|
|
))
|
|
|
|
} else {
|
|
|
|
Ok(OwnerFilter { uid, gid })
|
|
|
|
}
|
|
|
|
}
|
2018-06-30 21:57:20 +02:00
|
|
|
|
|
|
|
pub fn matches(&self, md: &fs::Metadata) -> bool {
|
|
|
|
use std::os::unix::fs::MetadataExt;
|
|
|
|
|
|
|
|
let uid_ok = self.uid.map(|u| u == md.uid()).unwrap_or(true);
|
|
|
|
let gid_ok = self.gid.map(|g| g == md.gid()).unwrap_or(true);
|
|
|
|
|
|
|
|
uid_ok && gid_ok
|
|
|
|
}
|
2018-06-30 14:55:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod owner_parsing {
|
|
|
|
use super::OwnerFilter;
|
|
|
|
|
|
|
|
macro_rules! owner_tests {
|
|
|
|
($($name:ident: $value:expr => $result:pat,)*) => {
|
|
|
|
$(
|
|
|
|
#[test]
|
|
|
|
fn $name() {
|
|
|
|
let o = OwnerFilter::from_string($value);
|
|
|
|
match o {
|
|
|
|
$result => {},
|
|
|
|
_ => panic!("{:?} does not match {}", o, stringify!($result)),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)*
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
owner_tests! {
|
|
|
|
empty: "" => Err(_),
|
|
|
|
uid_only: "5" => Ok(OwnerFilter { uid: Some(5), gid: None }),
|
|
|
|
uid_gid: "9:3" => Ok(OwnerFilter { uid: Some(9), gid: Some(3)}),
|
|
|
|
gid_only: ":8" => Ok(OwnerFilter { uid: None, gid: Some(8)}),
|
|
|
|
colon_only: ":" => Err(_),
|
|
|
|
trailing: "5:" => Ok(OwnerFilter { uid: Some(5), gid: None }),
|
|
|
|
}
|
|
|
|
}
|