1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
use std::{
  future::Future,
  ops::{Deref, DerefMut},
};

use async_scoped::{Scope, TokioScope};

/// Run futures in parallel.
///
///
/// # Panic
///
/// Panics if any task panics.
///
/// A rough demo of how this works:
///
/// # Example
///
/// ```rust,ignore
/// use rspack_futures::FuturesResults;
///
/// #[tokio::main]
/// fn main() {
///   let futures = vec![async { 1 }, async { 2 }];
///   let results = futures.into_iter().collect::<FuturesResults>();
///
///   assert_eq!(results, vec![Ok(1), Ok(2)]);
/// }
/// ```
#[derive(Default)]
pub struct FuturesResults<T> {
  inner: Vec<T>,
}

impl<T> FuturesResults<T> {
  pub fn into_inner(self) -> Vec<T> {
    self.inner
  }
}

impl<'f, Fut> FromIterator<Fut> for FuturesResults<Fut::Output>
where
  Fut: Future + Send + 'f,
  Fut::Output: Send + 'static,
{
  fn from_iter<I>(iter: I) -> Self
  where
    I: IntoIterator<Item = Fut>,
  {
    let (_, inner) = Scope::scope_and_block(|s: &mut TokioScope<'_, _>| {
      iter.into_iter().for_each(|fut| {
        s.spawn(fut);
      });
    });

    Self {
      inner: inner
        .into_iter()
        .map(|i| match i {
          Ok(i) => i,
          Err(err) => {
            if err.is_panic() {
              std::panic::resume_unwind(err.into_panic())
            } else {
              unreachable!("Error should be a panic {err}")
            }
          }
        })
        .collect(),
    }
  }
}

impl<T> Deref for FuturesResults<T> {
  type Target = Vec<T>;

  fn deref(&self) -> &Self::Target {
    &self.inner
  }
}

impl<T> DerefMut for FuturesResults<T> {
  fn deref_mut(&mut self) -> &mut Self::Target {
    &mut self.inner
  }
}