2024-02-24 20:50:00 -08:00
|
|
|
#![allow(dead_code, unused_imports)]
|
2021-06-30 16:34:34 -07:00
|
|
|
|
2024-02-24 20:50:00 -08:00
|
|
|
#[cfg(not(target_family = "wasm"))]
|
|
|
|
use std::env::var as env_var;
|
2023-11-15 11:00:19 -08:00
|
|
|
use std::future::Future;
|
2024-02-24 09:46:11 -08:00
|
|
|
use std::sync::OnceLock;
|
2023-11-15 11:00:19 -08:00
|
|
|
|
2024-02-24 20:50:00 -08:00
|
|
|
use futures::try_join;
|
2023-03-20 21:35:53 -07:00
|
|
|
use riven::consts::{PlatformRoute, QueueType, RegionalRoute};
|
2021-12-29 09:35:09 -08:00
|
|
|
use riven::{RiotApi, RiotApiConfig};
|
2024-02-24 20:50:00 -08:00
|
|
|
#[cfg(not(target_family = "wasm"))]
|
|
|
|
pub use tokio_shared_rt::test as riven_test;
|
|
|
|
#[cfg(target_family = "wasm")]
|
|
|
|
pub use wasm_bindgen_test::wasm_bindgen_test as riven_test;
|
|
|
|
#[cfg(target_family = "wasm")]
|
|
|
|
#[allow(non_upper_case_globals)]
|
|
|
|
pub fn env_var<K: AsRef<str>>(key: K) -> Result<String, std::env::VarError> {
|
|
|
|
use wasm_bindgen::prelude::wasm_bindgen;
|
|
|
|
|
|
|
|
#[wasm_bindgen]
|
|
|
|
extern "C" {
|
|
|
|
type Process;
|
|
|
|
static process: Process;
|
|
|
|
|
|
|
|
type Env;
|
|
|
|
|
|
|
|
#[wasm_bindgen(method, getter)]
|
|
|
|
fn env(this: &Process) -> Env;
|
|
|
|
|
|
|
|
#[wasm_bindgen(method, structural, indexing_getter)]
|
|
|
|
fn get(this: &Env, field: &str) -> Option<String>;
|
|
|
|
}
|
|
|
|
|
|
|
|
process
|
|
|
|
.env()
|
|
|
|
.get(key.as_ref())
|
|
|
|
.ok_or(std::env::VarError::NotPresent)
|
|
|
|
}
|
2021-06-30 16:34:34 -07:00
|
|
|
|
2024-02-24 09:28:33 -08:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! rassert {
|
|
|
|
( $x:expr ) => {
|
|
|
|
{
|
|
|
|
if $x { Ok(()) } else { Err(stringify!($x)) }?
|
|
|
|
}
|
|
|
|
};
|
|
|
|
( $x:expr, $format:expr $(, $arg:expr)* ) => {
|
|
|
|
{
|
|
|
|
if $x { Ok(()) } else { Err( format!($format $(, $arg )* ) ) }?
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! rassert_eq {
|
|
|
|
( $a:expr, $b:expr ) => { rassert!($a == $b) };
|
|
|
|
( $a:expr, $b:expr, $format:expr $(, $arg:expr)* ) => {
|
|
|
|
rassert!($a == $b, $format $(, $arg )* )
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! rassert_ne {
|
|
|
|
( $a:expr, $b:expr ) => { rassert!($a != $b) };
|
|
|
|
( $a:expr, $b:expr, $format:expr $(, $arg:expr)* ) => {
|
|
|
|
rassert!($a != $b, $format $(, $arg )* )
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-02-24 09:46:11 -08:00
|
|
|
static RIOT_API: OnceLock<RiotApi> = OnceLock::new();
|
|
|
|
pub fn riot_api() -> &'static RiotApi {
|
|
|
|
RIOT_API.get_or_init(|| {
|
2024-02-24 09:28:33 -08:00
|
|
|
// Initialize logger here, as a convenient trigger spot.
|
2024-02-24 20:50:00 -08:00
|
|
|
#[cfg(not(target_family = "wasm"))]
|
2024-02-24 09:28:33 -08:00
|
|
|
env_logger::init();
|
2024-02-24 20:50:00 -08:00
|
|
|
#[cfg(target_family = "wasm")]
|
|
|
|
console_log::init_with_level(log::Level::Info).unwrap();
|
2024-02-24 09:28:33 -08:00
|
|
|
|
2024-02-24 20:50:00 -08:00
|
|
|
let api_key = env_var("RGAPI_KEY")
|
2021-12-29 09:35:09 -08:00
|
|
|
.ok()
|
2024-02-24 09:46:11 -08:00
|
|
|
.or_else(|| {
|
|
|
|
use std::iter::FromIterator;
|
|
|
|
|
|
|
|
let path =
|
|
|
|
std::path::PathBuf::from_iter([env!("CARGO_MANIFEST_DIR"), "../apikey.txt"]);
|
|
|
|
std::fs::read_to_string(path).ok()
|
|
|
|
})
|
2021-06-30 16:34:34 -07:00
|
|
|
.expect("Failed to find RGAPI_KEY env var or apikey.txt.");
|
2021-09-19 11:22:09 -07:00
|
|
|
RiotApi::new(RiotApiConfig::with_key(api_key.trim()).preconfig_burst())
|
2024-02-24 09:46:11 -08:00
|
|
|
})
|
2021-06-30 16:34:34 -07:00
|
|
|
}
|
|
|
|
|
2023-03-20 21:35:53 -07:00
|
|
|
pub async fn league_v4_match_v5_latest_combo(route: PlatformRoute) -> Result<(), String> {
|
|
|
|
const NUM_MATCHES: usize = 10;
|
|
|
|
|
2024-02-24 09:46:11 -08:00
|
|
|
let challenger_future = riot_api()
|
2023-03-20 21:35:53 -07:00
|
|
|
.league_v4()
|
|
|
|
.get_challenger_league(route, QueueType::RANKED_SOLO_5x5);
|
2023-07-19 13:15:16 -07:00
|
|
|
let challenger_league = challenger_future
|
|
|
|
.await
|
|
|
|
.map_err(|e| format!("Failed to get challenger league: {}", e))?;
|
2023-03-20 21:35:53 -07:00
|
|
|
|
2023-07-19 13:51:58 -07:00
|
|
|
let Some(queue) = challenger_league.queue else {
|
|
|
|
assert!(challenger_league.entries.is_empty());
|
|
|
|
eprintln!("Off-season, challenger league is empty.");
|
|
|
|
return Ok(());
|
|
|
|
};
|
|
|
|
|
|
|
|
if QueueType::RANKED_SOLO_5x5 != queue {
|
|
|
|
return Err(format!("Unexpected `queue`: {:?}", queue));
|
2023-03-20 21:35:53 -07:00
|
|
|
}
|
|
|
|
if challenger_league.entries.is_empty() {
|
|
|
|
return Err("Challenger league is unexpectedly empty!".to_owned());
|
|
|
|
}
|
|
|
|
|
|
|
|
let match_ids_futures = challenger_league
|
|
|
|
.entries
|
|
|
|
.iter()
|
|
|
|
.take(5)
|
|
|
|
.map(|entry| async move {
|
2024-02-24 09:46:11 -08:00
|
|
|
let summoner_future = riot_api()
|
2023-03-20 21:35:53 -07:00
|
|
|
.summoner_v4()
|
|
|
|
.get_by_summoner_id(route, &entry.summoner_id);
|
2023-07-19 13:15:16 -07:00
|
|
|
let summoner_info = summoner_future
|
|
|
|
.await
|
2024-01-04 09:26:16 -08:00
|
|
|
.map_err(|e| format!("Failed to find summoner info: {}", e))?;
|
2023-03-20 21:35:53 -07:00
|
|
|
|
2024-02-24 09:46:11 -08:00
|
|
|
let match_ids_future = riot_api().match_v5().get_match_ids_by_puuid(
|
2023-03-20 21:35:53 -07:00
|
|
|
route.to_regional(),
|
2023-05-10 11:27:57 -07:00
|
|
|
&summoner_info.puuid,
|
2023-03-20 21:35:53 -07:00
|
|
|
Some(5),
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
None,
|
|
|
|
);
|
2023-07-19 13:15:16 -07:00
|
|
|
let match_ids = match_ids_future
|
|
|
|
.await
|
2024-01-04 09:26:16 -08:00
|
|
|
.map_err(|e| format!("Failed to find summoner match IDs: {}", e))?;
|
2023-03-20 21:35:53 -07:00
|
|
|
Ok(match_ids) as Result<_, String>
|
|
|
|
});
|
|
|
|
|
|
|
|
let match_ids = futures::future::try_join_all(match_ids_futures).await?;
|
|
|
|
|
|
|
|
let mut match_ids: Vec<String> = match_ids.into_iter().flatten().collect();
|
|
|
|
match_ids.sort_unstable_by(|a, b| a.cmp(b).reverse()); // Sort descending, so latest are first.
|
|
|
|
|
2024-02-24 20:50:00 -08:00
|
|
|
let _ = try_join!(
|
2023-03-20 21:35:53 -07:00
|
|
|
match_v5_get(route.to_regional(), match_ids.iter().take(NUM_MATCHES)),
|
|
|
|
match_v5_get_timeline(route.to_regional(), match_ids.iter().take(NUM_MATCHES)),
|
|
|
|
)?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn tft_match_v1_get(
|
|
|
|
route: RegionalRoute,
|
|
|
|
matches: impl IntoIterator<Item = impl AsRef<str>>,
|
|
|
|
) -> Result<(), String> {
|
|
|
|
let futures = matches.into_iter().map(|matche| async move {
|
2023-03-08 13:49:53 -08:00
|
|
|
let matche = matche.as_ref();
|
2024-02-24 09:46:11 -08:00
|
|
|
let p = riot_api().tft_match_v1().get_match(route, matche);
|
2023-03-08 13:49:53 -08:00
|
|
|
let m = p
|
|
|
|
.await
|
|
|
|
.map_err(|e| format!("Failed to get match {}: {:?}", matche, e))?
|
|
|
|
.ok_or(format!("Match {} not found.", matche))?;
|
|
|
|
|
|
|
|
if matche != &*m.metadata.match_id {
|
|
|
|
return Err(format!(
|
|
|
|
"Bad match id? Sent {}, received {}.",
|
|
|
|
matche, m.metadata.match_id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
if m.metadata.participants.is_empty() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!(
|
|
|
|
"Match {} should have participants (metadata).",
|
|
|
|
matche
|
|
|
|
));
|
2023-03-08 13:49:53 -08:00
|
|
|
}
|
|
|
|
if m.metadata.participants.len() != m.info.participants.len() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!(
|
|
|
|
"Match {} participants do not line up with participant UUIDs.",
|
|
|
|
matche
|
|
|
|
));
|
2023-03-08 13:49:53 -08:00
|
|
|
}
|
|
|
|
if m.info.participants.is_empty() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!("Match {} should have participants (info).", matche));
|
2023-03-08 13:49:53 -08:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
});
|
|
|
|
futures::future::try_join_all(futures).await?;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-03-20 21:35:53 -07:00
|
|
|
pub async fn match_v5_get(
|
|
|
|
route: RegionalRoute,
|
|
|
|
matches: impl IntoIterator<Item = impl AsRef<str>>,
|
|
|
|
) -> Result<(), String> {
|
|
|
|
let futures = matches.into_iter().map(|matche| async move {
|
2023-03-08 13:49:53 -08:00
|
|
|
let matche = matche.as_ref();
|
2024-02-24 09:46:11 -08:00
|
|
|
let p = riot_api().match_v5().get_match(route, matche);
|
2021-12-29 09:35:09 -08:00
|
|
|
let m = p
|
|
|
|
.await
|
|
|
|
.map_err(|e| format!("Failed to get match {}: {:?}", matche, e))?
|
|
|
|
.ok_or(format!("Match {} not found.", matche))?;
|
|
|
|
|
|
|
|
if matche != &*m.metadata.match_id {
|
|
|
|
return Err(format!(
|
|
|
|
"Bad match id? Sent {}, received {}.",
|
|
|
|
matche, m.metadata.match_id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
if m.metadata.participants.is_empty() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!("Match {} should have participants.", matche));
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
2021-12-29 11:50:45 -08:00
|
|
|
if m.metadata.participants.len() != m.info.participants.len() {
|
2023-04-01 22:37:56 -07:00
|
|
|
// Sometimes only returns match IDs for one team? JP1_391732436
|
|
|
|
// Do not return error.
|
|
|
|
eprintln!(
|
2023-04-01 22:06:38 -07:00
|
|
|
"Match {} participants do not line up with participant UUIDs.",
|
|
|
|
matche
|
2023-04-01 22:37:56 -07:00
|
|
|
);
|
2021-12-29 11:50:45 -08:00
|
|
|
}
|
|
|
|
for participant in &m.info.participants {
|
2023-03-20 21:35:53 -07:00
|
|
|
participant
|
|
|
|
.champion()
|
2023-04-01 22:06:38 -07:00
|
|
|
.map_err(|e| format!("Failed to determine match {} champion: {}", matche, e))?;
|
2021-12-29 11:50:45 -08:00
|
|
|
}
|
2024-03-05 11:27:30 -08:00
|
|
|
|
|
|
|
{
|
|
|
|
let game_id = m.info.game_id;
|
|
|
|
if 0 == game_id {
|
|
|
|
eprintln!(
|
|
|
|
"Match {} `game_id` is zero, skipping remaining tests (see https://github.com/RiotGames/developer-relations/issues/898).",
|
|
|
|
matche
|
|
|
|
);
|
|
|
|
return Ok(());
|
|
|
|
} else if matche[(matche.find('_').unwrap() + 1)..] != game_id.to_string() {
|
|
|
|
return Err(format!(
|
|
|
|
"Match {} timeline number ID should match `game_id` {}.",
|
|
|
|
matche, game_id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-29 09:35:09 -08:00
|
|
|
if m.info.teams.is_empty() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!("Match {} should have teams.", matche));
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
2023-03-08 13:49:53 -08:00
|
|
|
Ok(())
|
|
|
|
});
|
2023-11-15 11:00:19 -08:00
|
|
|
join_all_future_errs(futures).await
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn match_v5_get_timeline(
|
|
|
|
route: RegionalRoute,
|
2023-03-20 21:35:53 -07:00
|
|
|
matches: impl IntoIterator<Item = impl AsRef<str>>,
|
2021-12-29 09:35:09 -08:00
|
|
|
) -> Result<(), String> {
|
2023-03-20 21:35:53 -07:00
|
|
|
let futures = matches.into_iter().map(|matche| async move {
|
2023-03-08 13:49:53 -08:00
|
|
|
let matche = matche.as_ref();
|
2024-02-24 09:46:11 -08:00
|
|
|
let p = riot_api().match_v5().get_timeline(route, matche);
|
2021-12-29 09:35:09 -08:00
|
|
|
let m = p
|
|
|
|
.await
|
|
|
|
.map_err(|e| format!("Failed to get match {}: {:?}", matche, e))?
|
|
|
|
.ok_or(format!("Match {} not found.", matche))?;
|
|
|
|
if matche != &*m.metadata.match_id {
|
|
|
|
return Err(format!(
|
|
|
|
"Bad match id? Sent {}, received {}.",
|
|
|
|
matche, m.metadata.match_id
|
|
|
|
));
|
|
|
|
}
|
|
|
|
if m.metadata.participants.is_empty() {
|
2023-04-01 22:06:38 -07:00
|
|
|
return Err(format!("Match {} should have participants.", matche));
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
|
|
|
if let Some(game_id) = m.info.game_id {
|
2024-03-05 11:27:30 -08:00
|
|
|
if 0 == game_id {
|
|
|
|
eprintln!("Match {} timeline `game_id` is zero (see https://github.com/RiotGames/developer-relations/issues/898).", matche);
|
|
|
|
} else if matche[(matche.find('_').unwrap() + 1)..] != game_id.to_string() {
|
|
|
|
return Err(format!(
|
|
|
|
"Match {} timeline number ID should match `game_id` {}.",
|
|
|
|
matche, game_id
|
|
|
|
));
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if m.info.frames.is_empty() {
|
2024-03-05 11:27:30 -08:00
|
|
|
return Err(format!("Match {} timeline should have frames.", matche));
|
2021-12-29 09:35:09 -08:00
|
|
|
}
|
2023-03-08 13:49:53 -08:00
|
|
|
Ok(())
|
|
|
|
});
|
2023-11-15 11:00:19 -08:00
|
|
|
join_all_future_errs(futures).await
|
|
|
|
}
|
|
|
|
|
2024-03-14 23:26:05 -07:00
|
|
|
pub async fn spectator_v4_combo(route: PlatformRoute) -> Result<(), String> {
|
|
|
|
let featured_p = riot_api().spectator_v4().get_featured_games(route);
|
|
|
|
let featured = featured_p.await.map_err(|e| e.to_string())?;
|
|
|
|
|
|
|
|
if featured.game_list.is_empty() {
|
|
|
|
eprintln!("Featured game list is empty!");
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let featured_game = &featured.game_list[0];
|
|
|
|
let participant = &featured_game.participants[0];
|
|
|
|
let summoner_id = participant.summoner_id.as_ref().ok_or_else(|| {
|
|
|
|
format!(
|
2024-03-14 23:47:22 -07:00
|
|
|
"Summoner in spectator featured game {} missing summoner ID: {}",
|
|
|
|
featured_game.game_id, &participant.summoner_name,
|
2024-03-14 23:26:05 -07:00
|
|
|
)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
let livegame_p = riot_api()
|
|
|
|
.spectator_v4()
|
|
|
|
.get_current_game_info_by_summoner(route, &summoner_id);
|
2024-03-14 23:47:22 -07:00
|
|
|
let livegame_o = livegame_p.await.map_err(|e| {
|
|
|
|
format!(
|
|
|
|
"Failed to get live game {} for summoner ID {}: {}",
|
|
|
|
featured_game.game_id, summoner_id, e,
|
|
|
|
)
|
|
|
|
})?;
|
2024-03-14 23:26:05 -07:00
|
|
|
if let Some(livegame) = livegame_o {
|
|
|
|
let participant_match = livegame
|
|
|
|
.participants
|
|
|
|
.iter()
|
|
|
|
.find(|p| p.summoner_name == participant.summoner_name);
|
|
|
|
rassert!(
|
|
|
|
participant_match.is_some(),
|
|
|
|
"Failed to find summoner in match: {}.",
|
|
|
|
&participant.summoner_name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn spectator_v5_combo(route: PlatformRoute) -> Result<(), String> {
|
|
|
|
let featured_p = riot_api().spectator_v5().get_featured_games(route);
|
|
|
|
let featured = featured_p.await.map_err(|e| e.to_string())?;
|
|
|
|
|
|
|
|
if featured.game_list.is_empty() {
|
|
|
|
eprintln!("Featured game list is empty!");
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let featured_game = &featured.game_list[0];
|
|
|
|
let participant = &featured_game.participants[0];
|
|
|
|
let puuid = participant.puuid.as_ref().ok_or_else(|| {
|
|
|
|
format!(
|
2024-03-14 23:47:22 -07:00
|
|
|
"Summoner in spectator featured game {} missing summoner ID: {}",
|
|
|
|
featured_game.game_id, &participant.summoner_name,
|
2024-03-14 23:26:05 -07:00
|
|
|
)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
let livegame_p = riot_api()
|
|
|
|
.spectator_v5()
|
|
|
|
.get_current_game_info_by_puuid(route, &puuid);
|
2024-03-14 23:47:22 -07:00
|
|
|
let livegame_o = livegame_p.await.map_err(|e| {
|
|
|
|
format!(
|
|
|
|
"Failed to get live game {} for summoner PUUID {}: {}",
|
|
|
|
featured_game.game_id, puuid, e,
|
|
|
|
)
|
|
|
|
})?;
|
2024-03-14 23:26:05 -07:00
|
|
|
if let Some(livegame) = livegame_o {
|
|
|
|
let participant_match = livegame
|
|
|
|
.participants
|
|
|
|
.iter()
|
|
|
|
.find(|p| p.summoner_name == participant.summoner_name);
|
|
|
|
rassert!(
|
|
|
|
participant_match.is_some(),
|
|
|
|
"Failed to find summoner in match: {}.",
|
|
|
|
&participant.summoner_name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
pub async fn spectator_tft_v5_combo(route: PlatformRoute) -> Result<(), String> {
|
|
|
|
let featured_p = riot_api().spectator_tft_v5().get_featured_games(route);
|
|
|
|
let featured = featured_p.await.map_err(|e| e.to_string())?;
|
|
|
|
|
|
|
|
if featured.game_list.is_empty() {
|
|
|
|
eprintln!("Featured game list is empty!");
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let featured_game = &featured.game_list[0];
|
|
|
|
let participant = &featured_game.participants[0];
|
|
|
|
let puuid = participant.puuid.as_ref().ok_or_else(|| {
|
|
|
|
format!(
|
2024-03-14 23:47:22 -07:00
|
|
|
"Summoner in spectator featured game {} missing summoner ID: {}",
|
|
|
|
featured_game.game_id, &participant.summoner_name,
|
2024-03-14 23:26:05 -07:00
|
|
|
)
|
|
|
|
})?;
|
|
|
|
|
|
|
|
let livegame_p = riot_api()
|
|
|
|
.spectator_tft_v5()
|
|
|
|
.get_current_game_info_by_puuid(route, &puuid);
|
2024-03-14 23:47:22 -07:00
|
|
|
let livegame_o = livegame_p.await.map_err(|e| {
|
|
|
|
format!(
|
|
|
|
"Failed to get live game {} for summoner PUUID {}: {}",
|
|
|
|
featured_game.game_id, puuid, e,
|
|
|
|
)
|
|
|
|
})?;
|
2024-03-14 23:26:05 -07:00
|
|
|
if let Some(livegame) = livegame_o {
|
|
|
|
let participant_match = livegame
|
|
|
|
.participants
|
|
|
|
.iter()
|
|
|
|
.find(|p| p.summoner_name == participant.summoner_name);
|
|
|
|
rassert!(
|
|
|
|
participant_match.is_some(),
|
|
|
|
"Failed to find summoner in match: {}.",
|
|
|
|
&participant.summoner_name
|
|
|
|
);
|
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-11-15 11:00:19 -08:00
|
|
|
/// Joins all futures and keeps ALL error messages, separated by newlines.
|
|
|
|
async fn join_all_future_errs<T>(
|
|
|
|
result_tasks: impl Iterator<Item = impl Future<Output = Result<T, String>>>,
|
|
|
|
) -> Result<(), String> {
|
|
|
|
futures::future::join_all(result_tasks)
|
|
|
|
.await
|
|
|
|
.into_iter()
|
|
|
|
.filter_map(Result::err)
|
|
|
|
.reduce(|a, b| a + "\n" + &b)
|
|
|
|
.map(Err)
|
|
|
|
.unwrap_or(Ok(()))
|
2021-06-30 16:34:34 -07:00
|
|
|
}
|