From 3badf1d9098138c880117d3952e6bbbf7358da9b Mon Sep 17 00:00:00 2001 From: Matheus Consoli Date: Mon, 24 Apr 2023 16:53:39 -0300 Subject: [PATCH 1/2] Fix doc links pointing to the `Race` trait --- src/future/race/array.rs | 3 +-- src/future/race/tuple.rs | 3 +-- src/future/race/vec.rs | 3 +-- 3 files changed, 3 insertions(+), 6 deletions(-) diff --git a/src/future/race/array.rs b/src/future/race/array.rs index b7ce4f1..66f263f 100644 --- a/src/future/race/array.rs +++ b/src/future/race/array.rs @@ -11,11 +11,10 @@ use pin_project::pin_project; /// A future which waits for the first future to complete. /// -/// This `struct` is created by the [`race`] method on the [`Race`] trait. See +/// This `struct` is created by the [`race`] method on the [`Race`][RaceTrait] trait. See /// its documentation for more. /// /// [`race`]: crate::future::Race::race -/// [`Race`]: crate::future::Race #[must_use = "futures do nothing unless you `.await` or poll them"] #[pin_project] pub struct Race diff --git a/src/future/race/tuple.rs b/src/future/race/tuple.rs index c0fb804..f6c81a4 100644 --- a/src/future/race/tuple.rs +++ b/src/future/race/tuple.rs @@ -12,11 +12,10 @@ macro_rules! impl_race_tuple { ($StructName:ident $($F:ident)+) => { /// A future which waits for the first future to complete. /// - /// This `struct` is created by the [`race`] method on the [`Race`] trait. See + /// This `struct` is created by the [`race`] method on the [`Race`][RaceTrait] trait. See /// its documentation for more. /// /// [`race`]: crate::future::Race::race - /// [`Race`]: crate::future::Race #[pin_project] #[must_use = "futures do nothing unless you `.await` or poll them"] #[allow(non_snake_case)] diff --git a/src/future/race/vec.rs b/src/future/race/vec.rs index 28e75f2..cebc4ae 100644 --- a/src/future/race/vec.rs +++ b/src/future/race/vec.rs @@ -11,11 +11,10 @@ use pin_project::pin_project; /// A future which waits for the first future to complete. /// -/// This `struct` is created by the [`race`] method on the [`Race`] trait. See +/// This `struct` is created by the [`race`] method on the [`Race`][RaceTrait] trait. See /// its documentation for more. /// /// [`race`]: crate::future::Race::race -/// [`Race`]: crate::future::Race #[must_use = "futures do nothing unless you `.await` or poll them"] #[pin_project] pub struct Race From 08876a81e1b65e999f34b3e7cbb798d36ff8dea9 Mon Sep 17 00:00:00 2001 From: Matheus Consoli Date: Mon, 24 Apr 2023 17:05:37 -0300 Subject: [PATCH 2/2] Improve `Race` documentation --- src/future/race/mod.rs | 83 +++++++++++++++++++++++++++++++++++++++--- 1 file changed, 78 insertions(+), 5 deletions(-) diff --git a/src/future/race/mod.rs b/src/future/race/mod.rs index 19ca85c..443c621 100644 --- a/src/future/race/mod.rs +++ b/src/future/race/mod.rs @@ -6,20 +6,93 @@ pub(crate) mod vec; /// Wait for the first future to complete. /// -/// Awaits multiple future at once, returning as soon as one completes. The -/// other futures are cancelled. +/// Awaits multiple futures simultaneously, returning as soon as one completes, even if it +/// completes with an error. All remaining futures are cancelled. pub trait Race { /// The resulting output type. type Output; - /// Which kind of future are we turning this into? + /// The [`Future`] implementation returned by this method. type Future: Future; /// Wait for the first future to complete. /// - /// Awaits multiple futures at once, returning as soon as one completes. The - /// other futures are cancelled. + /// Awaits multiple futures simultaneously, returning as soon as one completes, even if it + /// completes with an error. All remaining futures are cancelled. /// + /// All futures must resolve to the same type. + /// + /// Use [`race_ok`] if you only care about the first successful future to complete. + /// + /// # Examples + /// + /// Await multiple futures until the first resolve. + /// + /// ```rust + /// # futures::executor::block_on(async { + /// use futures_concurrency::prelude::*; + /// + /// async fn fast_and_err(id: u8) -> Result { + /// futures_lite::future::yield_now().await; + /// Err(id) + /// } + /// async fn slow_and_ok(id: u8) -> Result { + /// futures_lite::future::yield_now().await; + /// Ok(id) + /// } + /// + /// let id = (slow_and_ok(0), fast_and_err(1)).race().await; + /// + /// assert_eq!(id, Err(1)); + /// # }); + /// ``` + /// + /// ## Translating from `select!` + /// + /// The following example is taken from the [`futures::select!`] documentation: + /// + /// ```rust + /// # futures::executor::block_on(async { + /// use futures::future; + /// use futures::select; + /// let mut a = future::ready(4); + /// let mut b = future::pending::<()>(); + /// + /// let res = select! { + /// a_res = a => a_res + 1, + /// _ = b => 0, + /// }; + /// assert_eq!(res, 5); + /// # }); + /// ``` + /// + /// and can be translated using `race` as: + /// + /// ```rust + /// # futures::executor::block_on(async { + /// use futures::future; + /// use futures::select; + /// use futures_concurrency::prelude::*; + /// + /// let mut a = async { + /// let a_res = future::ready(4).await; + /// a_res + 1 + /// }; + /// let mut b = async { + /// let _ = future::pending::<()>().await; + /// 0 + /// }; + /// + /// let res = (a, b).race().await; + /// + /// assert_eq!(res, 5); + /// # }); + /// ``` + /// + ///

/// This function returns a new future which polls all futures concurrently. + /// + /// [`race_ok`]: crate::future::RaceOk::race_ok + /// [`futures::select!`]: https://docs.rs/futures/0.3.28/futures/macro.select.html#examples fn race(self) -> Self::Future; }