Add iterator extension for iterators over Result<T, E>
This commit is contained in:
parent
67410b3ad2
commit
0870665668
1 changed files with 159 additions and 31 deletions
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue