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
87
88
89
90
91
92
93
94
95
#![cfg_attr(all(doc, CHANNEL_NIGHTLY), feature(doc_auto_cfg))]
#![cfg_attr(feature = "client", allow(incomplete_features))]
#![cfg_attr(
    feature = "client",
    feature(associated_type_defaults, const_caller_location, specialization,)
)]

#[macro_use]
extern crate async_backtrace;
#[macro_use]
extern crate cfg_if;
#[macro_use]
extern crate derivative;

mod env;
mod error;

pub use env::*;
pub use error::*;

#[doc(hidden)]
pub use lazy_static::lazy_static as service_util_lazy_static;
#[doc(hidden)]
pub use paste::paste as service_util_paste;

cfg_if! {
    if #[cfg(feature = "client")] {
        mod client;
        pub use client::*;
    }
}
cfg_if! {
    if #[cfg(feature = "db")] {
        mod paginate;
        pub use paginate::*;
    } else if #[cfg(feature = "graphql")] {
        mod paginate;
        pub use paginate::*;
    }
}
cfg_if! {
    if #[cfg(feature = "server")] {
        mod server;
        pub use server::*;
    }
}
cfg_if! {
    if #[cfg(feature = "trace")] {
        mod trace;
        pub use trace::*;

        #[doc(hidden)]
        pub use tracing;
    }
}

cfg_if! {
    if #[cfg(feature = "try-join-safe")] {
        #[doc(hidden)]
        pub use futures as service_util_futures;

        #[macro_export]
        macro_rules! try_join_safe {
            ($($expr:expr),+ $(,)?) => { $crate::try_join_safe!($($expr,)+ ~ @ _a ~ $($expr,)+) };
            ($($orig:expr,)* ~ $($final_ident:ident,)* @ $ident:ident ~ $expr:expr, $($remaining:expr,)*) => {
                $crate::service_util_paste! {
                    $crate::try_join_safe! { $($orig,)* ~ $($final_ident,)* $ident, @ [<_ $ident>] ~ $($remaining,)* }
                }
            };
            ( $($expr:expr,)* ~ $($final_ident:ident,)* @ $ident:ident ~) => {
                {
                    let ($($final_ident),*) = $crate::service_util_futures::join!($($expr),+);
                    $crate::try_join_safe!( @ @ $($final_ident)* )
                }
            };
            (@ $($used:ident)* @ $ident:ident $($remaining:ident)*) => {
                $ident.and_then(move |$ident| $crate::try_join_safe!( @ $($used)* $ident @ $($remaining)* ))
            };
            (@ $($used:ident)* @) => {
                Ok(($($used),*))
            };
        }

        #[allow(clippy::manual_async_fn)]
        pub fn try_join_all_safe<I, T: Send, E: Send>(iter: I) -> impl futures::Future<Output = Result<Vec<T>, E>> + Send
        where
            I: IntoIterator + Send,
            <I as IntoIterator>::Item: futures::Future<Output = Result<T, E>> + Send,
        {
            async move {
                futures::future::join_all(iter).await.into_iter().collect()
            }
        }
    }
}