mod async_body; pub mod github; pub use anyhow::{Result, anyhow}; pub use async_body::{AsyncBody, Inner}; use derive_more::Deref; use http::HeaderValue; pub use http::{self, Method, Request, Response, StatusCode, Uri}; use futures::{ FutureExt as _, future::{self, BoxFuture}, }; use http::request::Builder; use parking_lot::Mutex; #[cfg(feature = "test-support")] use std::fmt; use std::{any::type_name, sync::Arc}; pub use url::Url; #[derive(Default, Debug, Clone, PartialEq, Eq, Hash)] pub enum RedirectPolicy { #[default] NoFollow, FollowLimit(u32), FollowAll, } pub struct FollowRedirects(pub bool); pub trait HttpRequestExt { /// Whether or not to follow redirects fn follow_redirects(self, follow: RedirectPolicy) -> Self; } impl HttpRequestExt for http::request::Builder { fn follow_redirects(self, follow: RedirectPolicy) -> Self { self.extension(follow) } } pub trait HttpClient: 'static + Send + Sync { fn type_name(&self) -> &'static str; fn user_agent(&self) -> Option<&HeaderValue>; fn send( &self, req: http::Request, ) -> BoxFuture<'static, anyhow::Result>>; fn get<'a>( &'a self, uri: &str, body: AsyncBody, follow_redirects: bool, ) -> BoxFuture<'a, anyhow::Result>> { let request = Builder::new() .uri(uri) .follow_redirects(if follow_redirects { RedirectPolicy::FollowAll } else { RedirectPolicy::NoFollow }) .body(body); match request { Ok(request) => Box::pin(async move { self.send(request).await }), Err(e) => Box::pin(async move { Err(e.into()) }), } } fn post_json<'a>( &'a self, uri: &str, body: AsyncBody, ) -> BoxFuture<'a, anyhow::Result>> { let request = Builder::new() .uri(uri) .method(Method::POST) .header("Content-Type", "application/json") .body(body); match request { Ok(request) => Box::pin(async move { self.send(request).await }), Err(e) => Box::pin(async move { Err(e.into()) }), } } fn proxy(&self) -> Option<&Url>; #[cfg(feature = "test-support")] fn as_fake(&self) -> &FakeHttpClient { panic!("called as_fake on {}", type_name::()) } fn send_multipart_form<'a>( &'a self, _url: &str, _request: reqwest::multipart::Form, ) -> BoxFuture<'a, anyhow::Result>> { future::ready(Err(anyhow!("not implemented"))).boxed() } } /// An [`HttpClient`] that may have a proxy. #[derive(Deref)] pub struct HttpClientWithProxy { #[deref] client: Arc, proxy: Option, } impl HttpClientWithProxy { /// Returns a new [`HttpClientWithProxy`] with the given proxy URL. pub fn new(client: Arc, proxy_url: Option) -> Self { let proxy_url = proxy_url .and_then(|proxy| proxy.parse().ok()) .or_else(read_proxy_from_env); Self::new_url(client, proxy_url) } pub fn new_url(client: Arc, proxy_url: Option) -> Self { Self { client, proxy: proxy_url, } } } impl HttpClient for HttpClientWithProxy { fn send( &self, req: Request, ) -> BoxFuture<'static, anyhow::Result>> { self.client.send(req) } fn user_agent(&self) -> Option<&HeaderValue> { self.client.user_agent() } fn proxy(&self) -> Option<&Url> { self.proxy.as_ref() } fn type_name(&self) -> &'static str { self.client.type_name() } #[cfg(feature = "test-support")] fn as_fake(&self) -> &FakeHttpClient { self.client.as_fake() } fn send_multipart_form<'a>( &'a self, url: &str, form: reqwest::multipart::Form, ) -> BoxFuture<'a, anyhow::Result>> { self.client.send_multipart_form(url, form) } } /// An [`HttpClient`] that has a base URL. pub struct HttpClientWithUrl { base_url: Mutex, client: HttpClientWithProxy, } impl std::ops::Deref for HttpClientWithUrl { type Target = HttpClientWithProxy; fn deref(&self) -> &Self::Target { &self.client } } impl HttpClientWithUrl { /// Returns a new [`HttpClientWithUrl`] with the given base URL. pub fn new( client: Arc, base_url: impl Into, proxy_url: Option, ) -> Self { let client = HttpClientWithProxy::new(client, proxy_url); Self { base_url: Mutex::new(base_url.into()), client, } } pub fn new_url( client: Arc, base_url: impl Into, proxy_url: Option, ) -> Self { let client = HttpClientWithProxy::new_url(client, proxy_url); Self { base_url: Mutex::new(base_url.into()), client, } } /// Returns the base URL. pub fn base_url(&self) -> String { self.base_url.lock().clone() } /// Sets the base URL. pub fn set_base_url(&self, base_url: impl Into) { let base_url = base_url.into(); *self.base_url.lock() = base_url; } /// Builds a URL using the given path. pub fn build_url(&self, path: &str) -> String { format!("{}{}", self.base_url(), path) } /// Builds a Zed API URL using the given path. pub fn build_zed_api_url(&self, path: &str, query: &[(&str, &str)]) -> Result { let base_url = self.base_url(); let base_api_url = match base_url.as_ref() { "https://zed.dev" => "https://api.zed.dev", "https://staging.zed.dev" => "https://api-staging.zed.dev", "http://localhost:3000" => "http://localhost:8080", other => other, }; Ok(Url::parse_with_params( &format!("{}{}", base_api_url, path), query, )?) } /// Builds a Zed Cloud URL using the given path. pub fn build_zed_cloud_url(&self, path: &str, query: &[(&str, &str)]) -> Result { let base_url = self.base_url(); let base_api_url = match base_url.as_ref() { "https://zed.dev" => "https://cloud.zed.dev", "https://staging.zed.dev" => "https://cloud.zed.dev", "http://localhost:3000" => "http://localhost:8787", other => other, }; Ok(Url::parse_with_params( &format!("{}{}", base_api_url, path), query, )?) } /// Builds a Zed LLM URL using the given path. pub fn build_zed_llm_url(&self, path: &str, query: &[(&str, &str)]) -> Result { let base_url = self.base_url(); let base_api_url = match base_url.as_ref() { "https://zed.dev" => "https://cloud.zed.dev", "https://staging.zed.dev" => "https://llm-staging.zed.dev", "http://localhost:3000" => "http://localhost:8787", other => other, }; Ok(Url::parse_with_params( &format!("{}{}", base_api_url, path), query, )?) } } impl HttpClient for HttpClientWithUrl { fn send( &self, req: Request, ) -> BoxFuture<'static, anyhow::Result>> { self.client.send(req) } fn user_agent(&self) -> Option<&HeaderValue> { self.client.user_agent() } fn proxy(&self) -> Option<&Url> { self.client.proxy.as_ref() } fn type_name(&self) -> &'static str { self.client.type_name() } #[cfg(feature = "test-support")] fn as_fake(&self) -> &FakeHttpClient { self.client.as_fake() } fn send_multipart_form<'a>( &'a self, url: &str, request: reqwest::multipart::Form, ) -> BoxFuture<'a, anyhow::Result>> { self.client.send_multipart_form(url, request) } } pub fn read_proxy_from_env() -> Option { const ENV_VARS: &[&str] = &[ "ALL_PROXY", "all_proxy", "HTTPS_PROXY", "https_proxy", "HTTP_PROXY", "http_proxy", ]; ENV_VARS .iter() .find_map(|var| std::env::var(var).ok()) .and_then(|env| env.parse().ok()) } pub struct BlockedHttpClient; impl BlockedHttpClient { pub fn new() -> Self { BlockedHttpClient } } impl HttpClient for BlockedHttpClient { fn send( &self, _req: Request, ) -> BoxFuture<'static, anyhow::Result>> { Box::pin(async { Err(std::io::Error::new( std::io::ErrorKind::PermissionDenied, "BlockedHttpClient disallowed request", ) .into()) }) } fn user_agent(&self) -> Option<&HeaderValue> { None } fn proxy(&self) -> Option<&Url> { None } fn type_name(&self) -> &'static str { type_name::() } #[cfg(feature = "test-support")] fn as_fake(&self) -> &FakeHttpClient { panic!("called as_fake on {}", type_name::()) } } #[cfg(feature = "test-support")] type FakeHttpHandler = Arc< dyn Fn(Request) -> BoxFuture<'static, anyhow::Result>> + Send + Sync + 'static, >; #[cfg(feature = "test-support")] pub struct FakeHttpClient { handler: Mutex>, user_agent: HeaderValue, } #[cfg(feature = "test-support")] impl FakeHttpClient { pub fn create(handler: F) -> Arc where Fut: futures::Future>> + Send + 'static, F: Fn(Request) -> Fut + Send + Sync + 'static, { Arc::new(HttpClientWithUrl { base_url: Mutex::new("http://test.example".into()), client: HttpClientWithProxy { client: Arc::new(Self { handler: Mutex::new(Some(Arc::new(move |req| Box::pin(handler(req))))), user_agent: HeaderValue::from_static(type_name::()), }), proxy: None, }, }) } pub fn with_404_response() -> Arc { Self::create(|_| async move { Ok(Response::builder() .status(404) .body(Default::default()) .unwrap()) }) } pub fn with_200_response() -> Arc { Self::create(|_| async move { Ok(Response::builder() .status(200) .body(Default::default()) .unwrap()) }) } pub fn replace_handler(&self, new_handler: F) where Fut: futures::Future>> + Send + 'static, F: Fn(FakeHttpHandler, Request) -> Fut + Send + Sync + 'static, { let mut handler = self.handler.lock(); let old_handler = handler.take().unwrap(); *handler = Some(Arc::new(move |req| { Box::pin(new_handler(old_handler.clone(), req)) })); } } #[cfg(feature = "test-support")] impl fmt::Debug for FakeHttpClient { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { f.debug_struct("FakeHttpClient").finish() } } #[cfg(feature = "test-support")] impl HttpClient for FakeHttpClient { fn send( &self, req: Request, ) -> BoxFuture<'static, anyhow::Result>> { let future = (self.handler.lock().as_ref().unwrap())(req); future } fn user_agent(&self) -> Option<&HeaderValue> { Some(&self.user_agent) } fn proxy(&self) -> Option<&Url> { None } fn type_name(&self) -> &'static str { type_name::() } fn as_fake(&self) -> &FakeHttpClient { self } }