use core::mem; use core::pin::Pin; use futures_core::future::{FusedFuture, Future}; use futures_core::ready; use futures_core::stream::{FusedStream, Stream}; use futures_core::task::{Context, Poll}; use pin_project_lite::pin_project; pin_project! { /// Future for the [`unzip`](super::StreamExt::unzip) method. #[derive(Debug)] #[must_use = "futures do nothing unless you `.await` or poll them"] pub struct Unzip { #[pin] stream: St, left: FromA, right: FromB, } } impl Unzip { fn finish(self: Pin<&mut Self>) -> (FromA, FromB) { let this = self.project(); (mem::take(this.left), mem::take(this.right)) } pub(super) fn new(stream: St) -> Self { Self { stream, left: Default::default(), right: Default::default() } } } impl FusedFuture for Unzip where St: FusedStream, FromA: Default + Extend, FromB: Default + Extend, { fn is_terminated(&self) -> bool { self.stream.is_terminated() } } impl Future for Unzip where St: Stream, FromA: Default + Extend, FromB: Default + Extend, { type Output = (FromA, FromB); fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<(FromA, FromB)> { let mut this = self.as_mut().project(); loop { match ready!(this.stream.as_mut().poll_next(cx)) { Some(e) => { this.left.extend(Some(e.0)); this.right.extend(Some(e.1)); } None => return Poll::Ready(self.finish()), } } } }