use std::iter::FromIterator;
use std::str::FromStr;
use std::{collections::HashMap, fmt};
use futures::AsyncReadExt;
use http_types::convert::DeserializeOwned;
use http_types::headers::{HeaderName, HeaderValue, HeaderValues};
use http_types::{Method, Url};
pub const BODY_PRINT_LIMIT: usize = 10_000;
#[derive(Debug, Copy, Clone)]
pub enum BodyPrintLimit {
Limited(usize),
Unlimited,
}
#[derive(Debug, Clone)]
pub struct Request {
pub url: Url,
pub method: Method,
pub headers: HashMap<HeaderName, HeaderValues>,
pub body: Vec<u8>,
pub body_print_limit: BodyPrintLimit,
}
impl fmt::Display for Request {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
writeln!(f, "{} {}", self.method, self.url)?;
for (name, values) in &self.headers {
let values = values
.iter()
.map(|value| format!("{}", value))
.collect::<Vec<_>>();
let values = values.join(",");
writeln!(f, "{}: {}", name, values)?;
}
match self.body_print_limit {
BodyPrintLimit::Limited(limit) if self.body.len() > limit => {
let mut written = false;
for end_byte in limit..(limit + 4).max(self.body.len()) {
if let Ok(truncated) = std::str::from_utf8(&self.body[..end_byte]) {
written = true;
writeln!(f, "{}", truncated)?;
if end_byte < self.body.len() {
writeln!(
f,
"We truncated the body because it was too large: {} bytes (limit: {} bytes)",
self.body.len(),
limit
)?;
writeln!(f, "Increase this limit by setting `WIREMOCK_BODY_PRINT_LIMIT`, or calling `MockServerBuilder::body_print_limit` when building your MockServer instance")?;
}
break;
}
}
if !written {
writeln!(
f,
"Body is likely binary (invalid utf-8) size is {} bytes",
self.body.len()
)
} else {
Ok(())
}
}
_ => {
if let Ok(body) = std::str::from_utf8(&self.body) {
writeln!(f, "{}", body)
} else {
writeln!(
f,
"Body is likely binary (invalid utf-8) size is {} bytes",
self.body.len()
)
}
}
}
}
}
impl Request {
pub fn body_json<T: DeserializeOwned>(&self) -> Result<T, serde_json::Error> {
serde_json::from_slice(&self.body)
}
pub async fn from(mut request: http_types::Request) -> Request {
let method = request.method();
let url = request.url().to_owned();
let mut headers = HashMap::new();
for (header_name, header_values) in &request {
headers.insert(header_name.to_owned(), header_values.to_owned());
}
let mut body: Vec<u8> = vec![];
request
.take_body()
.into_reader()
.read_to_end(&mut body)
.await
.expect("Failed to read body");
Self {
url,
method,
headers,
body,
body_print_limit: BodyPrintLimit::Limited(BODY_PRINT_LIMIT),
}
}
pub(crate) async fn from_hyper(request: hyper::Request<hyper::Body>) -> Request {
let (parts, body) = request.into_parts();
let method = parts.method.into();
let url = match parts.uri.authority() {
Some(_) => parts.uri.to_string(),
None => format!("http://localhost{}", parts.uri),
}
.parse()
.unwrap();
let mut headers = HashMap::new();
for name in parts.headers.keys() {
let name = name.as_str().as_bytes().to_owned();
let name = HeaderName::from_bytes(name).unwrap();
let values = parts.headers.get_all(name.as_str());
for value in values {
let value = value.as_bytes().to_owned();
let value = HeaderValue::from_bytes(value).unwrap();
let value_parts = value.as_str().split(',');
let value_parts = value_parts
.map(|it| it.trim())
.filter_map(|it| HeaderValue::from_str(it).ok());
headers
.entry(name.clone())
.and_modify(|values: &mut HeaderValues| {
values.append(&mut HeaderValues::from_iter(value_parts.clone()))
})
.or_insert_with(|| value_parts.collect());
}
}
let body = hyper::body::to_bytes(body)
.await
.expect("Failed to read request body.")
.to_vec();
Self {
url,
method,
headers,
body,
body_print_limit: BodyPrintLimit::Limited(BODY_PRINT_LIMIT),
}
}
}