futures_time/stream/interval.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
use std::future::Future;
use std::pin::Pin;
use std::task::{Context, Poll};
use async_io::Timer;
use futures_core::stream::Stream;
use crate::time::{Duration, Instant};
/// Creates a new stream that yields at a set interval.
///
/// The stream first yields after `dur`, and continues to yield every
/// `dur` after that. The stream accounts for time elapsed between calls, and
/// will adjust accordingly to prevent time skews.
///
/// Each interval may be slightly longer than the specified duration, but never
/// less.
///
/// Note that intervals are not intended for high resolution timers, but rather
/// they will likely fire some granularity after the exact instant that they're
/// otherwise indicated to fire at.
pub fn interval(dur: Duration) -> Interval {
Interval {
timer: Timer::after(dur.into()),
interval: dur,
}
}
/// A stream representing notifications at fixed interval
///
/// This stream is created by the [`interval`] function. See its
/// documentation for more.
///
/// [`interval`]: fn.interval.html
#[must_use = "streams do nothing unless polled or .awaited"]
#[derive(Debug)]
pub struct Interval {
timer: Timer,
interval: Duration,
}
impl Stream for Interval {
type Item = Instant;
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
let instant = match Pin::new(&mut self.timer).poll(cx) {
Poll::Ready(instant) => instant,
Poll::Pending => return Poll::Pending,
};
let interval = self.interval;
let _ = std::mem::replace(&mut self.timer, Timer::after(interval.into()));
Poll::Ready(Some(instant.into()))
}
}