| 1 | use std::pin::Pin; |
| 2 | use std::task::{Context, Poll}; |
| 3 | |
| 4 | use futures_core::Stream; |
| 5 | use pin_project_lite::pin_project; |
| 6 | |
| 7 | use crate::stream_ext::Fuse; |
| 8 | use crate::StreamExt; |
| 9 | |
| 10 | pin_project! { |
| 11 | /// Stream returned by the [`chain`](super::StreamExt::peekable) method. |
| 12 | pub struct Peekable<T: Stream> { |
| 13 | peek: Option<T::Item>, |
| 14 | #[pin] |
| 15 | stream: Fuse<T>, |
| 16 | } |
| 17 | } |
| 18 | |
| 19 | impl<T: Stream> Peekable<T> { |
| 20 | pub(crate) fn new(stream: T) -> Self { |
| 21 | let stream = stream.fuse(); |
| 22 | Self { peek: None, stream } |
| 23 | } |
| 24 | |
| 25 | /// Peek at the next item in the stream. |
| 26 | pub async fn peek(&mut self) -> Option<&T::Item> |
| 27 | where |
| 28 | T: Unpin, |
| 29 | { |
| 30 | if let Some(ref it) = self.peek { |
| 31 | Some(it) |
| 32 | } else { |
| 33 | self.peek = self.next().await; |
| 34 | self.peek.as_ref() |
| 35 | } |
| 36 | } |
| 37 | } |
| 38 | |
| 39 | impl<T: Stream> Stream for Peekable<T> { |
| 40 | type Item = T::Item; |
| 41 | |
| 42 | fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> { |
| 43 | let this = self.project(); |
| 44 | if let Some(it) = this.peek.take() { |
| 45 | Poll::Ready(Some(it)) |
| 46 | } else { |
| 47 | this.stream.poll_next(cx) |
| 48 | } |
| 49 | } |
| 50 | } |
| 51 | |