//! Test helpers for actix http client to use during testing. use std::fmt::Write as FmtWrite; use actix_http::cookie::{Cookie, CookieJar}; use actix_http::http::header::{self, Header, HeaderValue, IntoHeaderValue}; use actix_http::http::{HeaderName, HttpTryFrom, StatusCode, Version}; use actix_http::{h1, Payload, ResponseHead}; use bytes::Bytes; #[cfg(test)] use futures::Future; use percent_encoding::{percent_encode, USERINFO_ENCODE_SET}; use crate::ClientResponse; #[cfg(test)] thread_local! { static RT: std::cell::RefCell = { std::cell::RefCell::new(actix_rt::Runtime::new().unwrap()) }; } #[cfg(test)] pub(crate) fn run_on(f: F) -> R where F: Fn() -> R, { RT.with(move |rt| { rt.borrow_mut() .block_on(futures::future::lazy(|| Ok::<_, ()>(f()))) }) .unwrap() } #[cfg(test)] pub(crate) fn block_on(f: F) -> Result where F: Future, { RT.with(move |rt| rt.borrow_mut().block_on(f)) } /// Test `ClientResponse` builder pub struct TestResponse { head: ResponseHead, cookies: CookieJar, payload: Option, } impl Default for TestResponse { fn default() -> TestResponse { TestResponse { head: ResponseHead::new(StatusCode::OK), cookies: CookieJar::new(), payload: None, } } } impl TestResponse { /// Create TestResponse and set header pub fn with_header(key: K, value: V) -> Self where HeaderName: HttpTryFrom, V: IntoHeaderValue, { Self::default().header(key, value) } /// Set HTTP version of this response pub fn version(mut self, ver: Version) -> Self { self.head.version = ver; self } /// Set a header pub fn set(mut self, hdr: H) -> Self { if let Ok(value) = hdr.try_into() { self.head.headers.append(H::name(), value); return self; } panic!("Can not set header"); } /// Append a header pub fn header(mut self, key: K, value: V) -> Self where HeaderName: HttpTryFrom, V: IntoHeaderValue, { if let Ok(key) = HeaderName::try_from(key) { if let Ok(value) = value.try_into() { self.head.headers.append(key, value); return self; } } panic!("Can not create header"); } /// Set cookie for this response pub fn cookie<'a>(mut self, cookie: Cookie<'a>) -> Self { self.cookies.add(cookie.into_owned()); self } /// Set response's payload pub fn set_payload>(mut self, data: B) -> Self { let mut payload = h1::Payload::empty(); payload.unread_data(data.into()); self.payload = Some(payload.into()); self } /// Complete response creation and generate `ClientResponse` instance pub fn finish(self) -> ClientResponse { let mut head = self.head; let mut cookie = String::new(); for c in self.cookies.delta() { let name = percent_encode(c.name().as_bytes(), USERINFO_ENCODE_SET); let value = percent_encode(c.value().as_bytes(), USERINFO_ENCODE_SET); let _ = write!(&mut cookie, "; {}={}", name, value); } if !cookie.is_empty() { head.headers.insert( header::SET_COOKIE, HeaderValue::from_str(&cookie.as_str()[2..]).unwrap(), ); } if let Some(pl) = self.payload { ClientResponse::new(head, pl) } else { ClientResponse::new(head, h1::Payload::empty().into()) } } }