Add iterator extension for iterators over Result<T, E>

This commit is contained in:
Matthias Beyer 2017-11-25 16:50:39 +01:00
parent 67410b3ad2
commit 0870665668

View file

@ -17,7 +17,105 @@
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA // Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
// //
macro_rules! mk_iterator {
{
modname = $modname:ident,
itername = $itername:ident,
iteryield = $yield:ty,
extname = $extname:ident,
extfnname = $extfnname:ident,
fun = $fun:expr
} => {
use storeid::StoreId;
#[allow(unused_imports)]
use store::FileLockEntry;
use store::Store;
use error::Result;
pub struct $itername<'a>(Box<Iterator<Item = StoreId>>, &'a Store);
impl<'a> Iterator for $itername<'a> {
type Item = Result<$yield>;
fn next(&mut self) -> Option<Self::Item> {
self.0.next().map(|id| $fun(id, self.1))
}
}
pub trait $extname<'a> {
fn $extfnname(self, store: &'a Store) -> $itername<'a>;
}
impl<'a, I> $extname<'a> for I
where I: Iterator<Item = StoreId> + 'static
{
fn $extfnname(self, store: &'a Store) -> $itername<'a> {
$itername(Box::new(self), store)
}
}
}
}
use error::StoreError;
pub enum ExtensionError<E> {
Forwarded(E),
StoreError(StoreError)
}
macro_rules! mk_iterator_mod { macro_rules! mk_iterator_mod {
{
modname = $modname:ident,
itername = $itername:ident,
iteryield = $yield:ty,
extname = $extname:ident,
extfnname = $extfnname:ident,
fun = $fun:expr,
resultitername = $resultitername:ident,
resultextname = $resultextname:ident
} => {
pub mod $modname {
mk_iterator! {
modname = $modname,
itername = $itername,
iteryield = $yield,
extname = $extname,
extfnname = $extfnname,
fun = $fun
}
use std::result::Result as RResult;
pub struct $resultitername<'a, I>(I, &'a Store);
impl<'a, I, E> Iterator for $resultitername<'a, I>
where I: Iterator<Item = RResult<StoreId, E>>
{
type Item = RResult<$yield, $crate::iter::ExtensionError<E>>;
fn next(&mut self) -> Option<Self::Item> {
match self.0.next() {
Some(Ok(sid)) => Some($fun(sid, self.1).map_err($crate::iter::ExtensionError::StoreError)),
Some(Err(e)) => Some(Err($crate::iter::ExtensionError::Forwarded(e))),
None => None,
}
}
}
pub trait $resultextname<'a> : Iterator {
fn $extfnname(self, store: &'a Store) -> $resultitername<'a, Self>
where Self: Sized
{
$resultitername(self, store)
}
}
impl<'a, I> $resultextname<'a> for I
where I: Iterator
{ /* empty */ }
}
};
{ {
modname = $modname:ident, modname = $modname:ident,
itername = $itername:ident, itername = $itername:ident,
@ -27,34 +125,14 @@ macro_rules! mk_iterator_mod {
fun = $fun:expr fun = $fun:expr
} => { } => {
pub mod $modname { pub mod $modname {
use storeid::StoreId; mk_iterator! {
#[allow(unused_imports)] modname = $modname,
use store::FileLockEntry; itername = $itername,
use store::Store; iteryield = $yield,
use error::Result; extname = $extname,
extfnname = $extfnname,
pub struct $itername<'a>(Box<Iterator<Item = StoreId>>, &'a Store); fun = $fun
impl<'a> Iterator for $itername<'a> {
type Item = Result<$yield>;
fn next(&mut self) -> Option<Self::Item> {
self.0.next().map(|id| $fun(id, self.1))
}
} }
pub trait $extname<'a> {
fn $extfnname(self, store: &'a Store) -> $itername<'a>;
}
impl<'a, I> $extname<'a> for I
where I: Iterator<Item = StoreId> + 'static
{
fn $extfnname(self, store: &'a Store) -> $itername<'a> {
$itername(Box::new(self), store)
}
}
} }
} }
} }
@ -65,7 +143,9 @@ mk_iterator_mod! {
iteryield = FileLockEntry<'a>, iteryield = FileLockEntry<'a>,
extname = StoreIdCreateIteratorExtension, extname = StoreIdCreateIteratorExtension,
extfnname = into_create_iter, extfnname = into_create_iter,
fun = |id: StoreId, store: &'a Store| store.create(id) fun = |id: StoreId, store: &'a Store| store.create(id),
resultitername = StoreCreateResultIterator,
resultextname = StoreIdCreateResultIteratorExtension
} }
mk_iterator_mod! { mk_iterator_mod! {
@ -74,7 +154,9 @@ mk_iterator_mod! {
iteryield = (), iteryield = (),
extname = StoreIdDeleteIteratorExtension, extname = StoreIdDeleteIteratorExtension,
extfnname = into_delete_iter, extfnname = into_delete_iter,
fun = |id: StoreId, store: &'a Store| store.delete(id) fun = |id: StoreId, store: &'a Store| store.delete(id),
resultitername = StoreDeleteResultIterator,
resultextname = StoreIdDeleteResultIteratorExtension
} }
mk_iterator_mod! { mk_iterator_mod! {
@ -83,7 +165,9 @@ mk_iterator_mod! {
iteryield = Option<FileLockEntry<'a>>, iteryield = Option<FileLockEntry<'a>>,
extname = StoreIdGetIteratorExtension, extname = StoreIdGetIteratorExtension,
extfnname = into_get_iter, extfnname = into_get_iter,
fun = |id: StoreId, store: &'a Store| store.get(id) fun = |id: StoreId, store: &'a Store| store.get(id),
resultitername = StoreGetResultIterator,
resultextname = StoreIdGetResultIteratorExtension
} }
mk_iterator_mod! { mk_iterator_mod! {
@ -92,6 +176,50 @@ mk_iterator_mod! {
iteryield = FileLockEntry<'a>, iteryield = FileLockEntry<'a>,
extname = StoreIdRetrieveIteratorExtension, extname = StoreIdRetrieveIteratorExtension,
extfnname = into_retrieve_iter, extfnname = into_retrieve_iter,
fun = |id: StoreId, store: &'a Store| store.retrieve(id) fun = |id: StoreId, store: &'a Store| store.retrieve(id),
resultitername = StoreRetrieveResultIterator,
resultextname = StoreIdRetrieveResultIteratorExtension
}
#[cfg(test)]
mod compile_test {
// This module contains code to check whether this actually compiles the way we would like it to
// compile
use store::Store;
use storeid::StoreId;
#[allow(dead_code)]
fn store() -> Store {
unimplemented!()
}
#[allow(dead_code)]
fn test_compile_get() {
use super::get::StoreIdGetIteratorExtension;
let store = store();
let _ = store
.entries()
.unwrap()
.into_get_iter(&store);
}
#[allow(dead_code)]
fn test_compile_get_result() {
use super::get::StoreIdGetResultIteratorExtension;
fn to_result(e: StoreId) -> Result<StoreId, ()> {
Ok(e)
}
let store = store();
let _ = store
.entries()
.unwrap()
.map(to_result)
.into_get_iter(&store);
}
} }