From 2c6d19de99d9e33ac0163954b20440d0a28c4213 Mon Sep 17 00:00:00 2001 From: Alex Bakon Date: Mon, 15 Sep 2025 10:20:16 -0400 Subject: [PATCH] Add OrPending future that wraps an optional future This provides a different behavior for `Option` than the existing Option future: instead of resolving to `Option` it resolves to `F::Output` if the future is present; otherwise it does not resolve. This can be useful for code paths that `select!` on multiple futures where one of the futures is optional. Using this wrapper allows sharing code paths since if the optional future is not present, that `select!` arm simply won't resolve. --- futures-util/src/future/mod.rs | 3 + futures-util/src/future/or_pending.rs | 88 +++++++++++++++++++++++++++ 2 files changed, 91 insertions(+) create mode 100644 futures-util/src/future/or_pending.rs diff --git a/futures-util/src/future/mod.rs b/futures-util/src/future/mod.rs index 2209ffd56e..6e599f6121 100644 --- a/futures-util/src/future/mod.rs +++ b/futures-util/src/future/mod.rs @@ -65,6 +65,9 @@ pub use self::try_maybe_done::{try_maybe_done, TryMaybeDone}; mod option; pub use self::option::OptionFuture; +mod or_pending; +pub use self::or_pending::OrPending; + mod poll_fn; pub use self::poll_fn::{poll_fn, PollFn}; diff --git a/futures-util/src/future/or_pending.rs b/futures-util/src/future/or_pending.rs new file mode 100644 index 0000000000..f99f993a63 --- /dev/null +++ b/futures-util/src/future/or_pending.rs @@ -0,0 +1,88 @@ +//! Definition of [`OrPending`] future wrapper. + +use core::pin::Pin; +use futures_core::future::{FusedFuture, Future}; +use futures_core::task::{Context, Poll}; +use pin_project_lite::pin_project; + +pin_project! { + /// A wrapper for a [`Future`] which may or may not be present. + /// + /// If the inner future is present, this wrapper will resolve when it does. + /// Otherwise this wrapper will never resolve (`Future::poll` will always + /// return [`Poll::Pending`]). Created by the [`From`] implementation for + /// [`Option`](std::option::Option). + /// + /// # Examples + /// + /// ``` + /// # futures::executor::block_on(async { + /// use futures::future::OptionFuture; + /// + /// let mut a: OptionFuture<_> = Some(async { 123 }).into(); + /// assert_eq!(a.await, Some(123)); + /// + /// a = None.into(); + /// assert_eq!(a.await, None); + /// # }); + /// ``` + #[derive(Debug, Clone)] + #[must_use = "futures do nothing unless you `.await` or poll them"] + pub struct OrPending { + #[pin] + inner: Option, + } +} + +impl OrPending { + /// Gets a mutable reference to the inner future (if any). + pub fn as_pin_mut(self: Pin<&mut Self>) -> Option> { + self.project().inner.as_pin_mut() + } + + /// Drops the inner future. + /// + /// After this, all calls to [`Future::poll`] will return [`Poll::Pending`]. + pub fn reset(self: Pin<&mut Self>) { + self.project().inner.set(None); + } +} + +impl Default for OrPending { + fn default() -> Self { + Self { inner: None } + } +} + +impl Future for OrPending { + type Output = F::Output; + + fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll { + let mut this = self.project(); + match this.inner.as_mut().as_pin_mut() { + None => Poll::Pending, + Some(x) => match x.poll(cx) { + Poll::Pending => Poll::Pending, + Poll::Ready(t) => { + this.inner.set(None); + Poll::Ready(t) + } + }, + } + } +} + +impl FusedFuture for OrPending { + fn is_terminated(&self) -> bool { + match &self.inner { + Some(x) => x.is_terminated(), + None => true, + } + } +} + +impl From> for OrPending { + fn from(option: Option) -> Self { + Self { inner: option } + } +}