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
115
116
117
118
119
120
121
122
123
124
125
126
127
128
|
//! ref: composer/src/Composer/Util/Http/ProxyManager.php
use std::sync::{Mutex, OnceLock};
use crate::downloader::transport_exception::TransportException;
use crate::util::http::proxy_item::ProxyItem;
use crate::util::http::request_proxy::RequestProxy;
use crate::util::no_proxy_pattern::NoProxyPattern;
static INSTANCE: OnceLock<Mutex<Option<ProxyManager>>> = OnceLock::new();
#[derive(Debug)]
pub struct ProxyManager {
error: Option<String>,
http_proxy: Option<ProxyItem>,
https_proxy: Option<ProxyItem>,
no_proxy_handler: Option<NoProxyPattern>,
}
impl ProxyManager {
fn new() -> Self {
let mut instance = Self {
error: None,
http_proxy: None,
https_proxy: None,
no_proxy_handler: None,
};
if let Err(e) = instance.get_proxy_data() {
instance.error = Some(e.to_string());
}
instance
}
pub fn get_instance() -> &'static Mutex<Option<ProxyManager>> {
INSTANCE.get_or_init(|| Mutex::new(Some(ProxyManager::new())))
}
pub fn reset() {
if let Some(mutex) = INSTANCE.get() {
*mutex.lock().unwrap() = Some(ProxyManager::new());
}
}
pub fn has_proxy(&self) -> bool {
self.http_proxy.is_some() || self.https_proxy.is_some()
}
pub fn get_proxy_for_request(
&self,
request_url: &str,
) -> Result<RequestProxy, TransportException> {
if let Some(ref error) = self.error {
return Err(TransportException::new(format!(
"Unable to use a proxy: {}",
error
)));
}
let scheme = request_url.split("://").next().unwrap_or("").to_string();
let proxy = self.get_proxy_for_scheme(&scheme);
if proxy.is_none() {
return Ok(RequestProxy::none());
}
if self.no_proxy(request_url) {
return Ok(RequestProxy::no_proxy());
}
Ok(proxy.unwrap().to_request_proxy(scheme))
}
fn get_proxy_for_scheme(&self, scheme: &str) -> Option<&ProxyItem> {
if scheme == "http" {
return self.http_proxy.as_ref();
}
if scheme == "https" {
return self.https_proxy.as_ref();
}
None
}
fn get_proxy_data(&mut self) -> anyhow::Result<()> {
// handle http_proxy/HTTP_PROXY on CLI only for security reasons
// PHP_SAPI is always 'cli' for this application
let (env, name) = Self::get_proxy_env("http_proxy");
if let Some(env) = env {
self.http_proxy = Some(ProxyItem::new(env, name));
}
if self.http_proxy.is_none() {
let (env, name) = Self::get_proxy_env("cgi_http_proxy");
if let Some(env) = env {
self.http_proxy = Some(ProxyItem::new(env, name));
}
}
let (env, name) = Self::get_proxy_env("https_proxy");
if let Some(env) = env {
self.https_proxy = Some(ProxyItem::new(env, name));
}
let (env, _name) = Self::get_proxy_env("no_proxy");
if let Some(env) = env {
self.no_proxy_handler = Some(NoProxyPattern::new(&env));
}
Ok(())
}
fn get_proxy_env(env_name: &str) -> (Option<String>, String) {
for name in [env_name.to_lowercase(), env_name.to_uppercase()] {
if let Ok(val) = std::env::var(&name) {
if !val.is_empty() {
return (Some(val), name);
}
}
}
(None, String::new())
}
fn no_proxy(&self, request_url: &str) -> bool {
match &self.no_proxy_handler {
None => false,
Some(handler) => handler.test(request_url),
}
}
}
|