-
Notifications
You must be signed in to change notification settings - Fork 634
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
StreamExt::All / StreamExt::Any (#2460)
Closes #2458 Co-authored-by: Marek Barvir <barvir@cadwork.cz>
- Loading branch information
Showing
3 changed files
with
236 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
use core::fmt; | ||
use core::pin::Pin; | ||
use futures_core::future::{FusedFuture, Future}; | ||
use futures_core::ready; | ||
use futures_core::stream::Stream; | ||
use futures_core::task::{Context, Poll}; | ||
use pin_project_lite::pin_project; | ||
|
||
pin_project! { | ||
/// Future for the [`all`](super::StreamExt::all) method. | ||
#[must_use = "futures do nothing unless you `.await` or poll them"] | ||
pub struct All<St, Fut, F> { | ||
#[pin] | ||
stream: St, | ||
f: F, | ||
accum: Option<bool>, | ||
#[pin] | ||
future: Option<Fut>, | ||
} | ||
} | ||
|
||
impl<St, Fut, F> fmt::Debug for All<St, Fut, F> | ||
where | ||
St: fmt::Debug, | ||
Fut: fmt::Debug, | ||
{ | ||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||
f.debug_struct("All") | ||
.field("stream", &self.stream) | ||
.field("accum", &self.accum) | ||
.field("future", &self.future) | ||
.finish() | ||
} | ||
} | ||
|
||
impl<St, Fut, F> All<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
pub(super) fn new(stream: St, f: F) -> Self { | ||
Self { stream, f, accum: Some(true), future: None } | ||
} | ||
} | ||
|
||
impl<St, Fut, F> FusedFuture for All<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
fn is_terminated(&self) -> bool { | ||
self.accum.is_none() && self.future.is_none() | ||
} | ||
} | ||
|
||
impl<St, Fut, F> Future for All<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
type Output = bool; | ||
|
||
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<bool> { | ||
let mut this = self.project(); | ||
Poll::Ready(loop { | ||
if let Some(fut) = this.future.as_mut().as_pin_mut() { | ||
// we're currently processing a future to produce a new accum value | ||
let acc = this.accum.unwrap() && ready!(fut.poll(cx)); | ||
if !acc { | ||
break false; | ||
} // early exit | ||
*this.accum = Some(acc); | ||
this.future.set(None); | ||
} else if this.accum.is_some() { | ||
// we're waiting on a new item from the stream | ||
match ready!(this.stream.as_mut().poll_next(cx)) { | ||
Some(item) => { | ||
this.future.set(Some((this.f)(item))); | ||
} | ||
None => { | ||
break this.accum.take().unwrap(); | ||
} | ||
} | ||
} else { | ||
panic!("All polled after completion") | ||
} | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,92 @@ | ||
use core::fmt; | ||
use core::pin::Pin; | ||
use futures_core::future::{FusedFuture, Future}; | ||
use futures_core::ready; | ||
use futures_core::stream::Stream; | ||
use futures_core::task::{Context, Poll}; | ||
use pin_project_lite::pin_project; | ||
|
||
pin_project! { | ||
/// Future for the [`any`](super::StreamExt::any) method. | ||
#[must_use = "futures do nothing unless you `.await` or poll them"] | ||
pub struct Any<St, Fut, F> { | ||
#[pin] | ||
stream: St, | ||
f: F, | ||
accum: Option<bool>, | ||
#[pin] | ||
future: Option<Fut>, | ||
} | ||
} | ||
|
||
impl<St, Fut, F> fmt::Debug for Any<St, Fut, F> | ||
where | ||
St: fmt::Debug, | ||
Fut: fmt::Debug, | ||
{ | ||
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { | ||
f.debug_struct("Any") | ||
.field("stream", &self.stream) | ||
.field("accum", &self.accum) | ||
.field("future", &self.future) | ||
.finish() | ||
} | ||
} | ||
|
||
impl<St, Fut, F> Any<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
pub(super) fn new(stream: St, f: F) -> Self { | ||
Self { stream, f, accum: Some(false), future: None } | ||
} | ||
} | ||
|
||
impl<St, Fut, F> FusedFuture for Any<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
fn is_terminated(&self) -> bool { | ||
self.accum.is_none() && self.future.is_none() | ||
} | ||
} | ||
|
||
impl<St, Fut, F> Future for Any<St, Fut, F> | ||
where | ||
St: Stream, | ||
F: FnMut(St::Item) -> Fut, | ||
Fut: Future<Output = bool>, | ||
{ | ||
type Output = bool; | ||
|
||
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<bool> { | ||
let mut this = self.project(); | ||
Poll::Ready(loop { | ||
if let Some(fut) = this.future.as_mut().as_pin_mut() { | ||
// we're currently processing a future to produce a new accum value | ||
let acc = this.accum.unwrap() || ready!(fut.poll(cx)); | ||
if acc { | ||
break true; | ||
} // early exit | ||
*this.accum = Some(acc); | ||
this.future.set(None); | ||
} else if this.accum.is_some() { | ||
// we're waiting on a new item from the stream | ||
match ready!(this.stream.as_mut().poll_next(cx)) { | ||
Some(item) => { | ||
this.future.set(Some((this.f)(item))); | ||
} | ||
None => { | ||
break this.accum.take().unwrap(); | ||
} | ||
} | ||
} else { | ||
panic!("Any polled after completion") | ||
} | ||
}) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters