Skip to content

Commit

Permalink
Merge pull request #264 from montekki/fs-stream-for-each
Browse files Browse the repository at this point in the history
Adds for_each stream combinator
  • Loading branch information
yoshuawuyts committed Oct 1, 2019
2 parents 0b57100 + 6da7efc commit a97a1ff
Show file tree
Hide file tree
Showing 2 changed files with 83 additions and 0 deletions.
46 changes: 46 additions & 0 deletions src/stream/stream/for_each.rs
@@ -0,0 +1,46 @@
use std::marker::PhantomData;
use std::pin::Pin;

use crate::future::Future;
use crate::stream::Stream;
use crate::task::{Context, Poll};

#[doc(hidden)]
#[allow(missing_debug_implementations)]
pub struct ForEachFuture<S, F, T> {
stream: S,
f: F,
__t: PhantomData<T>,
}

impl<S, F, T> ForEachFuture<S, F, T> {
pin_utils::unsafe_pinned!(stream: S);
pin_utils::unsafe_unpinned!(f: F);

pub(super) fn new(stream: S, f: F) -> Self {
ForEachFuture {
stream,
f,
__t: PhantomData,
}
}
}

impl<S, F> Future for ForEachFuture<S, F, S::Item>
where
S: Stream + Sized,
F: FnMut(S::Item),
{
type Output = ();

fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
loop {
let next = futures_core::ready!(self.as_mut().stream().poll_next(cx));

match next {
Some(v) => (self.as_mut().f())(v),
None => return Poll::Ready(()),
}
}
}
}
37 changes: 37 additions & 0 deletions src/stream/stream/mod.rs
Expand Up @@ -30,6 +30,7 @@ mod filter_map;
mod find;
mod find_map;
mod fold;
mod for_each;
mod fuse;
mod inspect;
mod min_by;
Expand All @@ -49,6 +50,7 @@ use filter_map::FilterMap;
use find::FindFuture;
use find_map::FindMapFuture;
use fold::FoldFuture;
use for_each::ForEachFuture;
use min_by::MinByFuture;
use next::NextFuture;
use nth::NthFuture;
Expand Down Expand Up @@ -750,6 +752,41 @@ extension_trait! {
FoldFuture::new(self, init, f)
}

#[doc = r#"
Call a closure on each element of the stream.
# Examples
```
# fn main() { async_std::task::block_on(async {
#
use async_std::prelude::*;
use std::collections::VecDeque;
use std::sync::mpsc::channel;
let (tx, rx) = channel();
let s: VecDeque<usize> = vec![1, 2, 3].into_iter().collect();
let sum = s.for_each(move |x| tx.clone().send(x).unwrap()).await;
let v: Vec<_> = rx.iter().collect();
assert_eq!(v, vec![1, 2, 3]);
#
# }) }
```
"#]
fn for_each<F>(
self,
f: F,
) -> impl Future<Output = ()> [ForEachFuture<Self, F, Self::Item>]
where
Self: Sized,
F: FnMut(Self::Item),
{
ForEachFuture::new(self, f)
}

#[doc = r#"
Tests if any element of the stream matches a predicate.
Expand Down

0 comments on commit a97a1ff

Please sign in to comment.