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
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
use std::process::{Command, Stdio};
use crate::error::{Error, CommandFailedOutput};
pub(crate) trait SystemHTTPClient: Sized + Send + Sync {
const COMMAND: &'static str;
fn installed_spawn() -> Command {
Command::new(Self::COMMAND)
}
fn installed() -> bool {
!matches!(
Self::installed_spawn()
.stdin(Stdio::null())
.stdout(Stdio::null())
.stderr(Stdio::null())
.status(),
Err(err) if err.kind() == std::io::ErrorKind::NotFound
)
}
fn get(&self, uri: &str) -> Result<Vec<u8>, Error>;
}
macro_rules! system_http_clients {
{$($(#[$cfg:meta])? mod $mod:ident::$client:ident;)*} => {
$($(#[$cfg])? mod $mod;)*
#[allow(non_camel_case_types)]
#[derive(Clone, Copy, Debug)]
pub(super) enum ResolvedSystemHTTPClient {
$($(#[$cfg])? $client,)*
}
impl SystemHTTPClient for ResolvedSystemHTTPClient {
const COMMAND: &'static str = "";
fn installed() -> bool {
unimplemented!()
}
fn get(&self, uri: &str) -> Result<Vec<u8>, Error> {
match self {
$($(#[$cfg])? Self::$client => $mod::$client.get(uri)),*
}
}
}
lazy_static::lazy_static! {
static ref HTTP_CLIENT: Option<ResolvedSystemHTTPClient> = {
#[allow(clippy::never_loop)]
loop {
$($(#[$cfg])? {
if <$mod::$client as SystemHTTPClient>::installed() {
break Some(ResolvedSystemHTTPClient::$client);
}
})*
break None;
}
};
}
pub const fn supported_http_clients() -> &'static [&'static str] {
&[
$($(#[$cfg])? { stringify!($client) }),*
]
}
#[cfg(test)]
pub(crate) fn all_http_clients() -> impl Iterator<Item = ResolvedSystemHTTPClient> {
[
$($(#[$cfg])? {
if <$mod::$client>::installed() {
Some(ResolvedSystemHTTPClient::$client)
} else {
None
}
}),*
].into_iter().flatten()
}
};
}
system_http_clients! {
mod wget::wget;
#[cfg(windows)]
mod powershell::PowerShell;
#[cfg(windows)]
mod curl::cURL;
#[cfg(not(windows))]
mod curl::cURL;
#[cfg(not(windows))]
mod powershell::PowerShell;
}
pub(super) fn resolve() -> Result<ResolvedSystemHTTPClient, Error> {
match *HTTP_CLIENT {
Some(client) => Ok(client),
None => Err(Error::SystemHTTPClientNotFound),
}
}
#[inline]
pub fn installed() -> bool {
HTTP_CLIENT.is_some()
}