use core::pin::Pin; use futures_core::future::{Future, FusedFuture}; use futures_core::stream::{Stream, FusedStream}; use futures_core::task::{Context, Poll}; use pin_utils::{unsafe_pinned, unsafe_unpinned}; /// Future for the [`concat`](super::StreamExt::concat) method. #[derive(Debug)] #[must_use = "futures do nothing unless you `.await` or poll them"] pub struct Concat { stream: St, accum: Option, } impl Unpin for Concat {} impl Concat where St: Stream, St::Item: Extend<::Item> + IntoIterator + Default, { unsafe_pinned!(stream: St); unsafe_unpinned!(accum: Option); pub(super) fn new(stream: St) -> Concat { Concat { stream, accum: None, } } } impl Future for Concat where St: Stream, St::Item: Extend<::Item> + IntoIterator + Default, { type Output = St::Item; fn poll( mut self: Pin<&mut Self>, cx: &mut Context<'_> ) -> Poll { loop { match ready!(self.as_mut().stream().poll_next(cx)) { None => { return Poll::Ready(self.as_mut().accum().take().unwrap_or_default()) } Some(e) => { let accum = self.as_mut().accum(); if let Some(a) = accum { a.extend(e) } else { *accum = Some(e) } } } } } } impl FusedFuture for Concat where St: FusedStream, St::Item: Extend<::Item> + IntoIterator + Default, { fn is_terminated(&self) -> bool { self.accum.is_none() && self.stream.is_terminated() } }