use std::fmt;
use std::sync::Arc;
use crate::middleware::{Middleware, Next};
use crate::protocol::{EffectSender, HttpResult, ProtocolRequestBuilder};
use crate::{Config, Request, RequestBuilder, ResponseAsync, Result};
use http_types::{Method, Url};
pub struct Client {
config: Config,
effect_sender: Arc<dyn EffectSender + Send + Sync>,
#[allow(clippy::rc_buffer)]
middleware: Arc<Vec<Arc<dyn Middleware>>>,
}
impl Clone for Client {
fn clone(&self) -> Self {
Self {
config: self.config.clone(),
effect_sender: Arc::clone(&self.effect_sender),
middleware: Arc::new(self.middleware.iter().cloned().collect()),
}
}
}
impl fmt::Debug for Client {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "Client {{}}")
}
}
impl Client {
#[cfg(test)]
pub(crate) fn new<Sender>(sender: Sender) -> Self
where
Sender: EffectSender + Send + Sync + 'static,
{
Self {
config: Config::default(),
effect_sender: Arc::new(sender),
middleware: Arc::new(vec![]),
}
}
#[allow(dead_code)]
pub(crate) fn with(mut self, middleware: impl Middleware) -> Self {
let m = Arc::get_mut(&mut self.middleware)
.expect("Registering middleware is not possible after the Client has been used");
m.push(Arc::new(middleware));
self
}
pub async fn send(&self, request: impl Into<Request>) -> Result<ResponseAsync> {
let mut request: Request = request.into();
let middleware = self.middleware.clone();
let mw_stack = match request.take_middleware() {
Some(req_mw) => {
let mut mw = Vec::with_capacity(middleware.len() + req_mw.len());
mw.extend(middleware.iter().cloned());
mw.extend(req_mw);
Arc::new(mw)
}
None => middleware,
};
let next = Next::new(&mw_stack, &|request, client| {
Box::pin(async move {
let request = request
.into_protocol_request()
.await
.expect("Failed to create request");
match client.effect_sender.send(request).await {
HttpResult::Ok(response) => Ok(response.into()),
HttpResult::Err(e) => Err(e),
}
})
});
let client = Self {
config: self.config.clone(),
effect_sender: Arc::clone(&self.effect_sender),
middleware: Arc::new(vec![]),
};
let response = next.run(request, client).await?;
Ok(ResponseAsync::new(response.into()))
}
pub async fn recv_bytes(&self, request: impl Into<Request>) -> Result<Vec<u8>> {
let mut response = self.send(request.into()).await?;
response.body_bytes().await
}
pub async fn recv_string(&self, request: impl Into<Request>) -> Result<String> {
let mut response = self.send(request.into()).await?;
response.body_string().await
}
pub async fn recv_json<T: serde::de::DeserializeOwned>(
&self,
request: impl Into<Request>,
) -> Result<T> {
let mut response = self.send(request.into()).await?;
response.body_json::<T>().await
}
pub async fn recv_form<T: serde::de::DeserializeOwned>(
&self,
request: impl Into<Request>,
) -> Result<T> {
let mut response = self.send(request.into()).await?;
response.body_form::<T>().await
}
pub fn get(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Get, self.url(uri), self.clone())
}
pub fn head(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Head, self.url(uri), self.clone())
}
pub fn post(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Post, self.url(uri), self.clone())
}
pub fn put(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Put, self.url(uri), self.clone())
}
pub fn delete(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Delete, self.url(uri), self.clone())
}
pub fn connect(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Connect, self.url(uri), self.clone())
}
pub fn options(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Options, self.url(uri), self.clone())
}
pub fn trace(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Trace, self.url(uri), self.clone())
}
pub fn patch(&self, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(Method::Patch, self.url(uri), self.clone())
}
pub fn request(&self, verb: Method, uri: impl AsRef<str>) -> RequestBuilder<()> {
RequestBuilder::new_for_middleware(verb, self.url(uri), self.clone())
}
#[must_use]
pub fn config(&self) -> &Config {
&self.config
}
fn url(&self, uri: impl AsRef<str>) -> Url {
match &self.config.base_url {
None => uri.as_ref().parse().unwrap(),
Some(base) => base.join(uri.as_ref()).unwrap(),
}
}
}
#[cfg(test)]
mod client_tests {
use super::Client;
use crate::protocol::{HttpRequest, HttpResponse};
use crate::testing::FakeShell;
#[futures_test::test]
async fn an_http_get() {
let mut shell = FakeShell::default();
shell.provide_response(HttpResponse::ok().body("Hello World!").build());
let client = Client::new(shell.clone());
let mut response = client.get("https://example.com").await.unwrap();
assert_eq!(response.body_string().await.unwrap(), "Hello World!");
assert_eq!(
shell.take_requests_received(),
vec![HttpRequest::get("https://example.com/").build()]
);
}
}