Move from a helper function to a typeclass.

This introduces the `FoldResut` trait to move from `func(receiver, ...)`
style to `receiver.func(...)` style. Also add a means to pass the
default result explicitly.
This commit is contained in:
John Sirois 2016-07-14 07:28:50 -06:00
parent 2c40b8734e
commit 7f57e5e234
5 changed files with 54 additions and 36 deletions

View file

@ -5,7 +5,7 @@ use lister::Lister;
use result::Result; use result::Result;
use libimagstore::store::FileLockEntry; use libimagstore::store::FileLockEntry;
use libimagutil::iter::fold_ok; use libimagutil::iter::FoldResult;
pub struct LineLister<'a> { pub struct LineLister<'a> {
unknown_output: &'a str, unknown_output: &'a str,
@ -27,7 +27,7 @@ impl<'a> Lister for LineLister<'a> {
use error::ListError as LE; use error::ListError as LE;
use error::ListErrorKind as LEK; use error::ListErrorKind as LEK;
fold_ok(entries, |entry| { entries.fold_defresult(|entry| {
write!(stdout(), "{:?}\n", entry.get_location().to_str().unwrap_or(self.unknown_output)) write!(stdout(), "{:?}\n", entry.get_location().to_str().unwrap_or(self.unknown_output))
.map_err(|e| LE::new(LEK::FormatError, Some(Box::new(e)))) .map_err(|e| LE::new(LEK::FormatError, Some(Box::new(e))))
}) })

View file

@ -5,7 +5,7 @@ use lister::Lister;
use result::Result; use result::Result;
use libimagstore::store::FileLockEntry; use libimagstore::store::FileLockEntry;
use libimagutil::iter::fold_ok; use libimagutil::iter::FoldResult;
pub struct PathLister { pub struct PathLister {
absolute: bool, absolute: bool,
@ -27,7 +27,7 @@ impl Lister for PathLister {
use error::ListError as LE; use error::ListError as LE;
use error::ListErrorKind as LEK; use error::ListErrorKind as LEK;
fold_ok(entries, |entry| { entries.fold_defresult(|entry| {
Ok(entry.get_location().clone()) Ok(entry.get_location().clone())
.and_then(|pb| { .and_then(|pb| {
if self.absolute { if self.absolute {

View file

@ -1,5 +1,5 @@
use libimagerror::trace::trace_error; use libimagerror::trace::trace_error;
use libimagutil::iter::fold_ok; use libimagutil::iter::FoldResult;
use store::FileLockEntry; use store::FileLockEntry;
use storeid::StoreId; use storeid::StoreId;
@ -46,7 +46,7 @@ impl StoreIdAccessor for Aspect {
return Err(HE::new(HEK::AccessTypeViolation, None)); return Err(HE::new(HEK::AccessTypeViolation, None));
} }
fold_ok(accessors.iter(), |accessor| { accessors.iter().fold_defresult(|accessor| {
let res = match accessor { let res = match accessor {
&HDA::StoreIdAccess(accessor) => accessor.access(id), &HDA::StoreIdAccess(accessor) => accessor.access(id),
_ => unreachable!(), _ => unreachable!(),
@ -88,7 +88,7 @@ impl MutableHookDataAccessor for Aspect {
// More sophisticated version would check whether there are _chunks_ of // More sophisticated version would check whether there are _chunks_ of
// NonMutableAccess accessors and execute these chunks in parallel. We do not have // NonMutableAccess accessors and execute these chunks in parallel. We do not have
// performance concerns yet, so this is okay. // performance concerns yet, so this is okay.
fold_ok(accessors.iter(), |accessor| { accessors.iter().fold_defresult(|accessor| {
let res = match accessor { let res = match accessor {
&HDA::MutableAccess(ref accessor) => accessor.access_mut(fle), &HDA::MutableAccess(ref accessor) => accessor.access_mut(fle),
&HDA::NonMutableAccess(ref accessor) => accessor.access(fle), &HDA::NonMutableAccess(ref accessor) => accessor.access(fle),
@ -118,7 +118,7 @@ impl NonMutableHookDataAccessor for Aspect {
return Err(HE::new(HEK::AccessTypeViolation, None)); return Err(HE::new(HEK::AccessTypeViolation, None));
} }
fold_ok(accessors.iter(), |accessor| { accessors.iter().fold_defresult(|accessor| {
let res = match accessor { let res = match accessor {
&HDA::NonMutableAccess(accessor) => accessor.access(fle), &HDA::NonMutableAccess(accessor) => accessor.access(fle),
_ => unreachable!(), _ => unreachable!(),

View file

@ -37,7 +37,7 @@ use hook::position::HookPosition;
use hook::Hook; use hook::Hook;
use libimagerror::into::IntoError; use libimagerror::into::IntoError;
use libimagutil::iter::fold_ok; use libimagutil::iter::FoldResult;
use self::glob_store_iter::*; use self::glob_store_iter::*;
@ -693,16 +693,14 @@ impl Store {
id: &StoreId) id: &StoreId)
-> HookResult<()> -> HookResult<()>
{ {
fold_ok( match aspects.lock() {
match aspects.lock() { Err(_) => return Err(HookErrorKind::HookExecutionError.into()),
Err(_) => return Err(HookErrorKind::HookExecutionError.into()), Ok(g) => g
Ok(g) => g }.iter().fold_defresult(|aspect| {
}.iter(), debug!("[Aspect][exec]: {:?}", aspect);
|aspect| { (aspect as &StoreIdAccessor).access(id)
debug!("[Aspect][exec]: {:?}", aspect); }).map_err(Box::new)
(aspect as &StoreIdAccessor).access(id) .map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e))
}).map_err(Box::new)
.map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e))
} }
fn execute_hooks_for_mut_file(&self, fn execute_hooks_for_mut_file(&self,
@ -710,16 +708,14 @@ impl Store {
fle: &mut FileLockEntry) fle: &mut FileLockEntry)
-> HookResult<()> -> HookResult<()>
{ {
fold_ok( match aspects.lock() {
match aspects.lock() { Err(_) => return Err(HookErrorKind::HookExecutionError.into()),
Err(_) => return Err(HookErrorKind::HookExecutionError.into()), Ok(g) => g
Ok(g) => g }.iter().fold_defresult(|aspect| {
}.iter(), debug!("[Aspect][exec]: {:?}", aspect);
|aspect| { aspect.access_mut(fle)
debug!("[Aspect][exec]: {:?}", aspect); }).map_err(Box::new)
aspect.access_mut(fle) .map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e))
}).map_err(Box::new)
.map_err(|e| HookErrorKind::HookExecutionError.into_error_with_cause(e))
} }
} }

View file

@ -1,8 +1,30 @@
/// Processes `iter` returning the last successful result or the first error. /// Folds its contents to a result.
pub fn fold_ok<X, I, R, E, F>(iter: I, mut func: F) -> Result<R, E> pub trait FoldResult: Sized {
where I: Iterator<Item = X>, type Item;
R: Default,
F: FnMut(X) -> Result<R, E> /// Processes all contained items returning the last successful result or the first error.
{ /// If there are no items, returns `Ok(R::default())`.
iter.fold(Ok(R::default()), |acc, item| acc.and_then(|_| func(item))) fn fold_defresult<R, E, F>(self, func: F) -> Result<R, E>
where R: Default,
F: FnMut(Self::Item)
-> Result<R, E>
{
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<R, E, F>(self, default: R, mut func: F) -> Result<R, E>
where F: FnMut(Self::Item) -> Result<R, E>;
} }
impl<X, I: Iterator<Item = X>> FoldResult for I {
type Item = X;
fn fold_result<R, E, F>(self, default: R, mut func: F) -> Result<R, E>
where F: FnMut(Self::Item) -> Result<R, E>
{
self.fold(Ok(default), |acc, item| acc.and_then(|_| func(item)))
}
}