use bytes::Bytes;
use futures::stream::TryStreamExt;
use futures::{AsyncRead, Stream};
use http::{HeaderMap, HeaderValue, Method, Uri, Version};
use hyper::Body;
use std::io;
pub struct Request {
pub method: Method,
pub uri: Uri,
pub version: Version,
pub headers: HeaderMap<HeaderValue>,
body: Body,
}
impl Request {
#[inline]
pub fn raw_body(&mut self) -> Body {
std::mem::take(&mut self.body)
}
#[inline]
pub fn stream(
&mut self,
) -> impl Stream<Item = io::Result<Bytes>> + Sync + Send + Unpin + 'static {
self.raw_body()
.map_err(|err| io::Error::new(io::ErrorKind::Other, err))
}
#[inline]
pub fn reader(&mut self) -> impl AsyncRead + Sync + Send + Unpin + 'static {
self.stream().into_async_read()
}
}
impl From<http::Request<Body>> for Request {
#[inline]
fn from(req: http::Request<Body>) -> Self {
let (parts, body) = req.into_parts();
Self {
method: parts.method,
uri: parts.uri,
version: parts.version,
headers: parts.headers,
body,
}
}
}
impl Default for Request {
#[inline]
fn default() -> Self {
http::Request::new(Body::empty()).into()
}
}
#[cfg(all(test, feature = "runtime"))]
mod tests {
use crate::{App, Request};
use futures::AsyncReadExt;
use http::StatusCode;
use hyper::Body;
#[async_std::test]
async fn body_read() -> Result<(), Box<dyn std::error::Error>> {
let mut app = App::new(());
app.gate_fn(|mut ctx, _next| async move {
let mut data = String::new();
ctx.req_mut().reader().read_to_string(&mut data).await?;
assert_eq!("Hello, World!", data);
Ok(())
});
let service = app.fake_service();
let req = Request::from(http::Request::new(Body::from("Hello, World!")));
let resp = service.serve(req).await?;
assert_eq!(StatusCode::OK, resp.status);
Ok(())
}
}