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
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
use futures::{
    ready,
    stream::Stream,
    task::{Context, Poll},
};
use std::{pin::Pin, time::Duration};
use tokio::time::{self, Interval};

/// Creates new `NumStream` that yields values at an interval of `period` that
/// continuously increment by `increment` where the first value yielded is `initial`.
///
/// A `NumStream` will yield values indefinitely. At any time, the `NumStream` value
/// can be dropped, canceling the stream.
///
/// # Panics
///
/// This function panics if `period` is zero.
///
/// # Examples
///
/// ```
/// use futures::stream::StreamExt;
/// use num_stream::num_stream;
/// use std::time::Duration;
/// use tokio;
///
/// #[tokio::main]
/// async fn main() {
///     let mut nums = num_stream(0, 3, Duration::from_millis(500));
///     loop {
///         println!("Got: {:?}", nums.next().await);
///     }
/// }
/// ```
pub fn num_stream(initial: u64, increment: u64, period: Duration) -> NumStream {
    assert!(period > Duration::new(0, 0), "`period` must be non-zero.");
    NumStream {
        num: initial,
        increment,
        timer: time::interval(period),
    }
}

/// Stream returned by [`num_stream`](fn.num_stream.html)
#[derive(Debug)]
pub struct NumStream {
    /// The next value that will be yielded by `NumStream`
    num: u64,

    /// How much `num` is incremented by after yielding
    increment: u64,

    /// The interval between values yielded by `NumStream`
    timer: Interval,
}

impl Stream for NumStream {
    type Item = u64;

    fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
        match ready!(Pin::new(&mut self.timer).poll_next(cx)) {
            Some(_) => {
                let retval = self.num;
                self.num = self.num.wrapping_add(self.increment);
                return Poll::Ready(Some(retval));
            }
            None => return Poll::Ready(None),
        }
    }
}