|
|
@ -7,24 +7,53 @@ use crate::task::{Context, Poll};
|
|
|
|
|
|
|
|
|
|
|
|
pin_project! {
|
|
|
|
pin_project! {
|
|
|
|
/// A stream that will repeatedly yield the same list of elements
|
|
|
|
/// A stream that will repeatedly yield the same list of elements
|
|
|
|
pub struct Cycle<T> {
|
|
|
|
pub struct Cycle<S, T> {
|
|
|
|
source: Vec<T>,
|
|
|
|
#[pin]
|
|
|
|
|
|
|
|
source: S,
|
|
|
|
index: usize,
|
|
|
|
index: usize,
|
|
|
|
len: usize,
|
|
|
|
buffer: Vec<T>,
|
|
|
|
|
|
|
|
state: CycleState,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
impl<T: Copy> Stream for Cycle<T> {
|
|
|
|
#[derive(Eq, PartialEq)]
|
|
|
|
type Item = T;
|
|
|
|
enum CycleState {
|
|
|
|
|
|
|
|
FromStream,
|
|
|
|
|
|
|
|
FromBuffer,
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
impl<S, T> Stream for Cycle<S,T>
|
|
|
|
|
|
|
|
where
|
|
|
|
|
|
|
|
S: Stream<Item = T>,
|
|
|
|
|
|
|
|
T: Copy,
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
fn poll_next(mut self: Pin<&mut Self>, _cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
|
|
|
type Item = S::Item;
|
|
|
|
let value = self.source[self.index];
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
let next = self.index + 1;
|
|
|
|
fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
|
|
|
|
|
|
|
|
let this = self.project();
|
|
|
|
|
|
|
|
|
|
|
|
self.as_mut().index = next % self.len;
|
|
|
|
let mut next;
|
|
|
|
|
|
|
|
if CycleState::FromStream == *this.state {
|
|
|
|
|
|
|
|
next = futures_core::ready!(this.source.poll_next(cx));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if let Some(val) = next {
|
|
|
|
|
|
|
|
this.buffer.push(val);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
*this.state = CycleState::FromBuffer;
|
|
|
|
|
|
|
|
next = Some(this.buffer[*this.index]);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
let mut index = *this.index;
|
|
|
|
|
|
|
|
if index == this.buffer.len() {
|
|
|
|
|
|
|
|
index = 0
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
next = Some(this.buffer[index]);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*this.index = index + 1;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Poll::Ready(Some(value))
|
|
|
|
Poll::Ready(next)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -40,21 +69,21 @@ impl<T: Copy> Stream for Cycle<T> {
|
|
|
|
/// use async_std::prelude::*;
|
|
|
|
/// use async_std::prelude::*;
|
|
|
|
/// use async_std::stream;
|
|
|
|
/// use async_std::stream;
|
|
|
|
///
|
|
|
|
///
|
|
|
|
/// let mut s = stream::cycle(vec![1,2,3]);
|
|
|
|
/// let mut s = stream::cycle(stream::once(7));
|
|
|
|
///
|
|
|
|
///
|
|
|
|
/// assert_eq!(s.next().await, Some(1));
|
|
|
|
/// assert_eq!(s.next().await, Some(7));
|
|
|
|
/// assert_eq!(s.next().await, Some(2));
|
|
|
|
/// assert_eq!(s.next().await, Some(7));
|
|
|
|
/// assert_eq!(s.next().await, Some(3));
|
|
|
|
/// assert_eq!(s.next().await, Some(7));
|
|
|
|
/// assert_eq!(s.next().await, Some(1));
|
|
|
|
/// assert_eq!(s.next().await, Some(7));
|
|
|
|
/// assert_eq!(s.next().await, Some(2));
|
|
|
|
/// assert_eq!(s.next().await, Some(7));
|
|
|
|
/// #
|
|
|
|
/// #
|
|
|
|
/// # })
|
|
|
|
/// # })
|
|
|
|
/// ```
|
|
|
|
/// ```
|
|
|
|
pub fn cycle<T: Copy>(source: Vec<T>) -> impl Stream<Item = T> {
|
|
|
|
pub fn cycle<S: Stream<Item = T>, T: Copy>(source: S) -> impl Stream<Item = S::Item> {
|
|
|
|
let len = source.len();
|
|
|
|
|
|
|
|
Cycle {
|
|
|
|
Cycle {
|
|
|
|
source,
|
|
|
|
source,
|
|
|
|
index: 0,
|
|
|
|
index: 0,
|
|
|
|
len,
|
|
|
|
buffer: Vec::new(),
|
|
|
|
|
|
|
|
state: CycleState::FromStream,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|