diff --git a/libimagentrylist/Cargo.toml b/libimagentrylist/Cargo.toml index d254bf0d..97c099c9 100644 --- a/libimagentrylist/Cargo.toml +++ b/libimagentrylist/Cargo.toml @@ -14,3 +14,6 @@ path = "../libimagstore" [dependencies.libimagerror] path = "../libimagerror" +[dependencies.libimagutil] +path = "../libimagutil" + diff --git a/libimagentrylist/src/lib.rs b/libimagentrylist/src/lib.rs index 082b6fa3..14bf910c 100644 --- a/libimagentrylist/src/lib.rs +++ b/libimagentrylist/src/lib.rs @@ -19,6 +19,7 @@ extern crate clap; extern crate toml; extern crate libimagstore; +extern crate libimagutil; #[macro_use] extern crate libimagerror; pub mod cli; diff --git a/libimagentrylist/src/listers/line.rs b/libimagentrylist/src/listers/line.rs index 1b233dc0..a5af428b 100644 --- a/libimagentrylist/src/listers/line.rs +++ b/libimagentrylist/src/listers/line.rs @@ -5,6 +5,7 @@ use lister::Lister; use result::Result; use libimagstore::store::FileLockEntry; +use libimagutil::iter::FoldResult; pub struct LineLister<'a> { unknown_output: &'a str, @@ -26,13 +27,10 @@ impl<'a> Lister for LineLister<'a> { use error::ListError as LE; use error::ListErrorKind as LEK; - entries.fold(Ok(()), |accu, entry| { - accu.and_then(|_| { - write!(stdout(), "{:?}\n", - entry.get_location().to_str().unwrap_or(self.unknown_output)) - .map_err(|e| LE::new(LEK::FormatError, Some(Box::new(e)))) - }) - }) + entries.fold_defresult(|entry| { + write!(stdout(), "{:?}\n", entry.get_location().to_str().unwrap_or(self.unknown_output)) + .map_err(|e| LE::new(LEK::FormatError, Some(Box::new(e)))) + }) } } diff --git a/libimagentrylist/src/listers/path.rs b/libimagentrylist/src/listers/path.rs index e3602770..503b2478 100644 --- a/libimagentrylist/src/listers/path.rs +++ b/libimagentrylist/src/listers/path.rs @@ -5,6 +5,7 @@ use lister::Lister; use result::Result; use libimagstore::store::FileLockEntry; +use libimagutil::iter::FoldResult; pub struct PathLister { absolute: bool, @@ -26,8 +27,8 @@ impl Lister for PathLister { use error::ListError as LE; use error::ListErrorKind as LEK; - entries.fold(Ok(()), |accu, entry| { - accu.and_then(|_| Ok(entry.get_location().clone())) + entries.fold_defresult(|entry| { + Ok(entry.get_location().clone()) .and_then(|pb| { if self.absolute { pb.canonicalize().map_err(|e| LE::new(LEK::FormatError, Some(Box::new(e)))) diff --git a/libimagstore/src/hook/aspect.rs b/libimagstore/src/hook/aspect.rs index 38b5cd07..bfbd5187 100644 --- a/libimagstore/src/hook/aspect.rs +++ b/libimagstore/src/hook/aspect.rs @@ -1,4 +1,5 @@ use libimagerror::trace::trace_error; +use libimagutil::iter::FoldResult; use store::FileLockEntry; use storeid::StoreId; @@ -45,29 +46,25 @@ impl StoreIdAccessor for Aspect { return Err(HE::new(HEK::AccessTypeViolation, None)); } - accessors - .iter() - .fold(Ok(()), |acc, accessor| { - acc.and_then(|_| { - let res = match accessor { - &HDA::StoreIdAccess(accessor) => accessor.access(id), - _ => unreachable!(), - }; + accessors.iter().fold_defresult(|accessor| { + let res = match accessor { + &HDA::StoreIdAccess(accessor) => accessor.access(id), + _ => unreachable!(), + }; - match res { - Ok(res) => Ok(res), - Err(e) => { - if !e.is_aborting() { - trace_error(&e); - // ignore error if it is not aborting, as we printed it already - Ok(()) - } else { - Err(e) - } - } + match res { + Ok(res) => Ok(res), + Err(e) => { + if !e.is_aborting() { + trace_error(&e); + // ignore error if it is not aborting, as we printed it already + Ok(()) + } else { + Err(e) } - }) - }) + } + } + }) } } @@ -91,27 +88,25 @@ impl MutableHookDataAccessor for Aspect { // More sophisticated version would check whether there are _chunks_ of // NonMutableAccess accessors and execute these chunks in parallel. We do not have // performance concerns yet, so this is okay. - accessors.iter().fold(Ok(()), |acc, accessor| { - acc.and_then(|_| { - let res = match accessor { - &HDA::MutableAccess(ref accessor) => accessor.access_mut(fle), - &HDA::NonMutableAccess(ref accessor) => accessor.access(fle), - _ => unreachable!(), - }; + accessors.iter().fold_defresult(|accessor| { + let res = match accessor { + &HDA::MutableAccess(ref accessor) => accessor.access_mut(fle), + &HDA::NonMutableAccess(ref accessor) => accessor.access(fle), + _ => unreachable!(), + }; - match res { - Ok(res) => Ok(res), - Err(e) => { - if !e.is_aborting() { - trace_error(&e); - // ignore error if it is not aborting, as we printed it already - Ok(()) - } else { - Err(e) - } + match res { + Ok(res) => Ok(res), + Err(e) => { + if !e.is_aborting() { + trace_error(&e); + // ignore error if it is not aborting, as we printed it already + Ok(()) + } else { + Err(e) } } - }) + } }) } } @@ -123,29 +118,25 @@ impl NonMutableHookDataAccessor for Aspect { return Err(HE::new(HEK::AccessTypeViolation, None)); } - accessors - .iter() - .fold(Ok(()), |acc, accessor| { - acc.and_then(|_| { - let res = match accessor { - &HDA::NonMutableAccess(accessor) => accessor.access(fle), - _ => unreachable!(), - }; + accessors.iter().fold_defresult(|accessor| { + let res = match accessor { + &HDA::NonMutableAccess(accessor) => accessor.access(fle), + _ => unreachable!(), + }; - match res { - Ok(res) => Ok(res), - Err(e) => { - if !e.is_aborting() { - trace_error(&e); - // ignore error if it is not aborting, as we printed it already - Ok(()) - } else { - Err(e) - } - } + match res { + Ok(res) => Ok(res), + Err(e) => { + if !e.is_aborting() { + trace_error(&e); + // ignore error if it is not aborting, as we printed it already + Ok(()) + } else { + Err(e) } - }) - }) + } + } + }) } } diff --git a/libimagstore/src/store.rs b/libimagstore/src/store.rs index 4238eef6..2ddae130 100644 --- a/libimagstore/src/store.rs +++ b/libimagstore/src/store.rs @@ -37,6 +37,7 @@ use hook::position::HookPosition; use hook::Hook; use libimagerror::into::IntoError; +use libimagutil::iter::FoldResult; use self::glob_store_iter::*; @@ -695,10 +696,11 @@ impl Store { match aspects.lock() { Err(_) => return Err(HookErrorKind::HookExecutionError.into()), Ok(g) => g - }.iter().fold(Ok(()), |acc, aspect| { + }.iter().fold_defresult(|aspect| { debug!("[Aspect][exec]: {:?}", aspect); - acc.and_then(|_| (aspect as &StoreIdAccessor).access(id)) - }).map_err(Box::new).map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e)) + (aspect as &StoreIdAccessor).access(id) + }).map_err(Box::new) + .map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e)) } fn execute_hooks_for_mut_file(&self, @@ -709,10 +711,11 @@ impl Store { match aspects.lock() { Err(_) => return Err(HookErrorKind::HookExecutionError.into()), Ok(g) => g - }.iter().fold(Ok(()), |acc, aspect| { + }.iter().fold_defresult(|aspect| { debug!("[Aspect][exec]: {:?}", aspect); - acc.and_then(|_| aspect.access_mut(fle)) - }).map_err(Box::new).map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e)) + aspect.access_mut(fle) + }).map_err(Box::new) + .map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e)) } } diff --git a/libimagutil/src/iter.rs b/libimagutil/src/iter.rs new file mode 100644 index 00000000..52d723bf --- /dev/null +++ b/libimagutil/src/iter.rs @@ -0,0 +1,30 @@ +/// Folds its contents to a result. +pub trait FoldResult: Sized { + type Item; + + /// Processes all contained items returning the last successful result or the first error. + /// If there are no items, returns `Ok(R::default())`. + fn fold_defresult(self, func: F) -> Result + where R: Default, + F: FnMut(Self::Item) + -> Result + { + self.fold_result(R::default(), func) + } + + /// Processes all contained items returning the last successful result or the first error. + /// If there are no items, returns `Ok(default)`. + fn fold_result(self, default: R, mut func: F) -> Result + where F: FnMut(Self::Item) -> Result; +} + +impl> FoldResult for I { + type Item = X; + + fn fold_result(self, default: R, mut func: F) -> Result + where F: FnMut(Self::Item) -> Result + { + self.fold(Ok(default), |acc, item| acc.and_then(|_| func(item))) + } +} + diff --git a/libimagutil/src/lib.rs b/libimagutil/src/lib.rs index bde49b00..730970e8 100644 --- a/libimagutil/src/lib.rs +++ b/libimagutil/src/lib.rs @@ -18,5 +18,6 @@ extern crate regex; pub mod ismatch; +pub mod iter; pub mod key_value_split; pub mod variants;