mirror of
https://github.com/instructkr/claw-code.git
synced 2026-06-07 02:12:45 -04:00
253 lines
7.5 KiB
Rust
253 lines
7.5 KiB
Rust
use crate::error::ApiError;
|
|
|
|
const HTTP_PROXY_KEYS: [&str; 2] = ["HTTP_PROXY", "http_proxy"];
|
|
const HTTPS_PROXY_KEYS: [&str; 2] = ["HTTPS_PROXY", "https_proxy"];
|
|
const NO_PROXY_KEYS: [&str; 2] = ["NO_PROXY", "no_proxy"];
|
|
|
|
/// Snapshot of the proxy-related environment variables that influence the
|
|
/// outbound HTTP client. Captured up front so callers can inspect, log, and
|
|
/// test the resolved configuration without re-reading the process environment.
|
|
#[derive(Debug, Clone, Default, PartialEq, Eq)]
|
|
pub struct ProxyConfig {
|
|
pub http_proxy: Option<String>,
|
|
pub https_proxy: Option<String>,
|
|
pub no_proxy: Option<String>,
|
|
}
|
|
|
|
impl ProxyConfig {
|
|
/// Read proxy settings from the live process environment, honouring both
|
|
/// the upper- and lower-case spellings used by curl, git, and friends.
|
|
#[must_use]
|
|
pub fn from_env() -> Self {
|
|
Self::from_lookup(|key| std::env::var(key).ok())
|
|
}
|
|
|
|
fn from_lookup<F>(mut lookup: F) -> Self
|
|
where
|
|
F: FnMut(&str) -> Option<String>,
|
|
{
|
|
Self {
|
|
http_proxy: first_non_empty(&HTTP_PROXY_KEYS, &mut lookup),
|
|
https_proxy: first_non_empty(&HTTPS_PROXY_KEYS, &mut lookup),
|
|
no_proxy: first_non_empty(&NO_PROXY_KEYS, &mut lookup),
|
|
}
|
|
}
|
|
|
|
#[must_use]
|
|
pub fn is_empty(&self) -> bool {
|
|
self.http_proxy.is_none() && self.https_proxy.is_none()
|
|
}
|
|
}
|
|
|
|
/// Build a `reqwest::Client` that honours the standard `HTTP_PROXY`,
|
|
/// `HTTPS_PROXY`, and `NO_PROXY` environment variables. When no proxy is
|
|
/// configured the client behaves identically to `reqwest::Client::new()`.
|
|
pub fn build_http_client() -> Result<reqwest::Client, ApiError> {
|
|
build_http_client_with(&ProxyConfig::from_env())
|
|
}
|
|
|
|
/// Infallible counterpart to [`build_http_client`] for constructors that
|
|
/// historically returned `Self` rather than `Result<Self, _>`. When the proxy
|
|
/// configuration is malformed we fall back to a default client so that
|
|
/// callers retain the previous behaviour and the failure surfaces on the
|
|
/// first outbound request instead of at construction time.
|
|
#[must_use]
|
|
pub fn build_http_client_or_default() -> reqwest::Client {
|
|
build_http_client().unwrap_or_else(|_| reqwest::Client::new())
|
|
}
|
|
|
|
/// Build a `reqwest::Client` from an explicit [`ProxyConfig`]. Used by tests
|
|
/// and by callers that want to override process-level environment lookups.
|
|
pub fn build_http_client_with(config: &ProxyConfig) -> Result<reqwest::Client, ApiError> {
|
|
let mut builder = reqwest::Client::builder().no_proxy();
|
|
|
|
let no_proxy = config
|
|
.no_proxy
|
|
.as_deref()
|
|
.and_then(reqwest::NoProxy::from_string);
|
|
|
|
if let Some(url) = config.https_proxy.as_deref() {
|
|
let mut proxy = reqwest::Proxy::https(url)?;
|
|
if let Some(filter) = no_proxy.clone() {
|
|
proxy = proxy.no_proxy(Some(filter));
|
|
}
|
|
builder = builder.proxy(proxy);
|
|
}
|
|
|
|
if let Some(url) = config.http_proxy.as_deref() {
|
|
let mut proxy = reqwest::Proxy::http(url)?;
|
|
if let Some(filter) = no_proxy.clone() {
|
|
proxy = proxy.no_proxy(Some(filter));
|
|
}
|
|
builder = builder.proxy(proxy);
|
|
}
|
|
|
|
Ok(builder.build()?)
|
|
}
|
|
|
|
fn first_non_empty<F>(keys: &[&str], lookup: &mut F) -> Option<String>
|
|
where
|
|
F: FnMut(&str) -> Option<String>,
|
|
{
|
|
keys.iter()
|
|
.find_map(|key| lookup(key).filter(|value| !value.is_empty()))
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use std::collections::HashMap;
|
|
|
|
use super::{build_http_client_with, ProxyConfig};
|
|
|
|
fn config_from_map(pairs: &[(&str, &str)]) -> ProxyConfig {
|
|
let map: HashMap<String, String> = pairs
|
|
.iter()
|
|
.map(|(key, value)| ((*key).to_string(), (*value).to_string()))
|
|
.collect();
|
|
ProxyConfig::from_lookup(|key| map.get(key).cloned())
|
|
}
|
|
|
|
#[test]
|
|
fn proxy_config_is_empty_when_no_env_vars_are_set() {
|
|
// given
|
|
let config = config_from_map(&[]);
|
|
|
|
// when
|
|
let empty = config.is_empty();
|
|
|
|
// then
|
|
assert!(empty);
|
|
assert_eq!(config, ProxyConfig::default());
|
|
}
|
|
|
|
#[test]
|
|
fn proxy_config_reads_uppercase_http_https_and_no_proxy() {
|
|
// given
|
|
let pairs = [
|
|
("HTTP_PROXY", "http://proxy.internal:3128"),
|
|
("HTTPS_PROXY", "http://secure.internal:3129"),
|
|
("NO_PROXY", "localhost,127.0.0.1,.corp"),
|
|
];
|
|
|
|
// when
|
|
let config = config_from_map(&pairs);
|
|
|
|
// then
|
|
assert_eq!(
|
|
config.http_proxy.as_deref(),
|
|
Some("http://proxy.internal:3128")
|
|
);
|
|
assert_eq!(
|
|
config.https_proxy.as_deref(),
|
|
Some("http://secure.internal:3129")
|
|
);
|
|
assert_eq!(
|
|
config.no_proxy.as_deref(),
|
|
Some("localhost,127.0.0.1,.corp")
|
|
);
|
|
assert!(!config.is_empty());
|
|
}
|
|
|
|
#[test]
|
|
fn proxy_config_falls_back_to_lowercase_keys() {
|
|
// given
|
|
let pairs = [
|
|
("http_proxy", "http://lower.internal:3128"),
|
|
("https_proxy", "http://lower-secure.internal:3129"),
|
|
("no_proxy", ".lower"),
|
|
];
|
|
|
|
// when
|
|
let config = config_from_map(&pairs);
|
|
|
|
// then
|
|
assert_eq!(
|
|
config.http_proxy.as_deref(),
|
|
Some("http://lower.internal:3128")
|
|
);
|
|
assert_eq!(
|
|
config.https_proxy.as_deref(),
|
|
Some("http://lower-secure.internal:3129")
|
|
);
|
|
assert_eq!(config.no_proxy.as_deref(), Some(".lower"));
|
|
}
|
|
|
|
#[test]
|
|
fn proxy_config_prefers_uppercase_over_lowercase_when_both_set() {
|
|
// given
|
|
let pairs = [
|
|
("HTTP_PROXY", "http://upper.internal:3128"),
|
|
("http_proxy", "http://lower.internal:3128"),
|
|
];
|
|
|
|
// when
|
|
let config = config_from_map(&pairs);
|
|
|
|
// then
|
|
assert_eq!(
|
|
config.http_proxy.as_deref(),
|
|
Some("http://upper.internal:3128")
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn proxy_config_treats_empty_strings_as_unset() {
|
|
// given
|
|
let pairs = [("HTTP_PROXY", ""), ("http_proxy", "")];
|
|
|
|
// when
|
|
let config = config_from_map(&pairs);
|
|
|
|
// then
|
|
assert!(config.http_proxy.is_none());
|
|
}
|
|
|
|
#[test]
|
|
fn build_http_client_succeeds_when_no_proxy_is_configured() {
|
|
// given
|
|
let config = ProxyConfig::default();
|
|
|
|
// when
|
|
let result = build_http_client_with(&config);
|
|
|
|
// then
|
|
assert!(result.is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn build_http_client_succeeds_with_valid_http_and_https_proxies() {
|
|
// given
|
|
let config = ProxyConfig {
|
|
http_proxy: Some("http://proxy.internal:3128".to_string()),
|
|
https_proxy: Some("http://secure.internal:3129".to_string()),
|
|
no_proxy: Some("localhost,127.0.0.1".to_string()),
|
|
};
|
|
|
|
// when
|
|
let result = build_http_client_with(&config);
|
|
|
|
// then
|
|
assert!(result.is_ok());
|
|
}
|
|
|
|
#[test]
|
|
fn build_http_client_returns_http_error_for_invalid_proxy_url() {
|
|
// given
|
|
let config = ProxyConfig {
|
|
http_proxy: None,
|
|
https_proxy: Some("not a url".to_string()),
|
|
no_proxy: None,
|
|
};
|
|
|
|
// when
|
|
let result = build_http_client_with(&config);
|
|
|
|
// then
|
|
let error = result.expect_err("invalid proxy URL must be reported as a build failure");
|
|
assert!(
|
|
matches!(error, crate::error::ApiError::Http(_)),
|
|
"expected ApiError::Http for invalid proxy URL, got: {error:?}"
|
|
);
|
|
}
|
|
}
|