use futures_core::stream::{FusedStream, Stream}; use futures_core::task::{Context, Poll}; use pin_project_lite::pin_project; use std::any::Any; use std::panic::{catch_unwind, AssertUnwindSafe, UnwindSafe}; use std::pin::Pin; pin_project! { /// Stream for the [`catch_unwind`](super::StreamExt::catch_unwind) method. #[derive(Debug)] #[must_use = "streams do nothing unless polled"] pub struct CatchUnwind { #[pin] stream: St, caught_unwind: bool, } } impl CatchUnwind { pub(super) fn new(stream: St) -> Self { Self { stream, caught_unwind: false } } delegate_access_inner!(stream, St, ()); } impl Stream for CatchUnwind { type Item = Result>; fn poll_next(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll> { let mut this = self.project(); if *this.caught_unwind { Poll::Ready(None) } else { let res = catch_unwind(AssertUnwindSafe(|| this.stream.as_mut().poll_next(cx))); match res { Ok(poll) => poll.map(|opt| opt.map(Ok)), Err(e) => { *this.caught_unwind = true; Poll::Ready(Some(Err(e))) } } } } fn size_hint(&self) -> (usize, Option) { if self.caught_unwind { (0, Some(0)) } else { self.stream.size_hint() } } } impl FusedStream for CatchUnwind { fn is_terminated(&self) -> bool { self.caught_unwind || self.stream.is_terminated() } }