#![allow(unused_imports)] #![allow(unused_parens)] //! You might want you stick with certain ecosystem like reqwest or actix-web, //! and don't want to mingle two ecosystem, since it significantly increase //! compile time and binary size. //! //! Arangors offers you flexibility to use any HTTP client you like for both //! async and sync implementation. Thanks to maybe_async, we can now unify async //! and sync API. //! //! Several implementations are provided: async `reqwest`, blocking `reqwest`, //! `surf`(async-std) and later `awc`. use anyhow::Error; use http::{HeaderMap, HeaderValue, Method}; #[cfg(feature = "reqwest_async")] use reqwest::Client; use uclient::{ClientError, ClientExt}; use url::Url; use arangors::GenericConnection; use std::convert::TryInto; /// when use async http client, `blocking` feature MUST be disabled // This cfg is only to make rust compiler happy in Github Action, you can just ignore it #[cfg(feature = "reqwest_async")] #[derive(Debug, Clone)] pub struct ReqwestClient(pub Client, HeaderMap); /// you can also use macro: maybe_async::async_impl, with which the whole code /// block will just vanish when you enabled `blocking` feature. /// Also, the API of reqwest is almost the same for async and sync. You can also /// use maybe_async::maybe_async to remove async/await keyword in need, and just /// import reqwesat::Client and rewest::blocking::Client respectively in async /// and sync implementation. See `uclient::reqwest` source code. // This cfg is only to make rust compiler happy in Github Action, you can just ignore it #[cfg(feature = "reqwest_async")] #[async_trait::async_trait] impl ClientExt for ReqwestClient { fn new>>(headers: U) -> Result { let client = Client::builder().gzip(true); let headers = match headers.into() { Some(h) => h, None => HeaderMap::new(), }; client .build() .map(|c| ReqwestClient(c, headers)) .map_err(|e| ClientError::HttpClient(format!("{:?}", e))) } fn headers(&mut self) -> &mut HeaderMap { &mut self.1 } async fn request( &self, mut request: http::Request, ) -> Result, ClientError> { let headers = request.headers_mut(); for (header, value) in self.1.iter() { if !headers.contains_key(header) { headers.insert(header, value.clone()); } } let req = request.try_into().unwrap(); let resp = self .0 .execute(req) .await .map_err(|e| ClientError::HttpClient(format!("{:?}", e)))?; let status_code = resp.status(); let headers = resp.headers().clone(); let version = resp.version(); let content = resp .text() .await .map_err(|e| ClientError::HttpClient(format!("{:?}", e)))?; let mut build = http::Response::builder(); for header in headers.iter() { build = build.header(header.0, header.1); } http::response::Builder::from(build) .status(status_code) .version(version) .body(content) .map_err(|e| ClientError::HttpClient(format!("{:?}", e))) } } // This cfg is only to make rust compiler happy in Github Action, you can just // ignore it #[cfg(feature = "reqwest_async")] #[tokio::main] async fn main() -> Result<(), Error> { const URL: &str = "http://localhost:8529"; let conn = GenericConnection::::establish_jwt(URL, "username", "password").await?; // from here the API is the same let db = conn.db("test_db").await?; let info = db.info().await?; println!("{:?}", info); Ok(()) } #[cfg(not(feature = "reqwest_async"))] fn main() {}