Reimplemented throttle to never drop Delay, added boolean flag

pull/356/head
Wouter Geraedts 5 years ago
parent 77a1849303
commit 6990c1403f

@ -318,7 +318,7 @@ extension_trait! {
#[doc = r#" #[doc = r#"
Limit the amount of items yielded per timeslice in a stream. Limit the amount of items yielded per timeslice in a stream.
This stream does not drop any items, but will only limit the rate at which items pass through. This stream does not drop any items, but will only limit the rate at which items pass through.
# Examples # Examples
``` ```
# fn main() { async_std::task::block_on(async { # fn main() { async_std::task::block_on(async {

@ -1,6 +1,6 @@
use std::future::Future; use std::future::Future;
use std::pin::Pin; use std::pin::Pin;
use std::time::Duration; use std::time::{Duration, Instant};
use futures_timer::Delay; use futures_timer::Delay;
use pin_project_lite::pin_project; use pin_project_lite::pin_project;
@ -23,7 +23,9 @@ pin_project! {
stream: S, stream: S,
duration: Duration, duration: Duration,
#[pin] #[pin]
delay: Option<Delay>, blocked: bool,
#[pin]
delay: Delay,
} }
} }
@ -32,7 +34,8 @@ impl<S: Stream> Throttle<S> {
Throttle { Throttle {
stream, stream,
duration, duration,
delay: None, blocked: false,
delay: Delay::new(Duration::default()),
} }
} }
} }
@ -42,9 +45,10 @@ impl<S: Stream> Stream for Throttle<S> {
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<S::Item>> { fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<S::Item>> {
let mut this = self.project(); let mut this = self.project();
if let Some(d) = this.delay.as_mut().as_pin_mut() { if *this.blocked {
let d = this.delay.as_mut();
if d.poll(cx).is_ready() { if d.poll(cx).is_ready() {
this.delay.set(None); *this.blocked = false;
} else { } else {
return Poll::Pending; return Poll::Pending;
} }
@ -57,7 +61,8 @@ impl<S: Stream> Stream for Throttle<S> {
} }
Poll::Ready(None) => Poll::Ready(None), Poll::Ready(None) => Poll::Ready(None),
Poll::Ready(Some(v)) => { Poll::Ready(Some(v)) => {
this.delay.set(Some(Delay::new(*this.duration))); *this.blocked = true;
this.delay.reset(Instant::now() + *this.duration);
Poll::Ready(Some(v)) Poll::Ready(Some(v))
} }
} }

Loading…
Cancel
Save