Add `AsyncReadBody` (#1072)

* Add `AsyncReadBody`

* changelog

* sort cargo.toml
This commit is contained in:
David Pedersen 2022-06-08 11:02:42 +02:00 committed by GitHub
parent 115a47b191
commit 73b1bafbf8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 116 additions and 6 deletions

View File

@ -8,8 +8,10 @@ and this project adheres to [Semantic Versioning].
# Unreleased
- **fixed:** Use `impl IntoResponse` less in docs ([#1049])
- **added:** Add `AsyncReadBody` for creating a body from a `tokio::io::AsyncRead` ([#1072])
[#1049]: https://github.com/tokio-rs/axum/pull/1049
[#1072]: https://github.com/tokio-rs/axum/pull/1072
# 0.3.3 (18. May, 2022)

View File

@ -12,14 +12,16 @@ version = "0.3.3"
[features]
default = []
erased-json = ["serde_json", "serde"]
typed-routing = ["axum-macros", "serde", "percent-encoding"]
async-read-body = ["tokio-util/io"]
cookie = ["cookie-lib"]
cookie-signed = ["cookie", "cookie-lib/signed"]
cookie-private = ["cookie", "cookie-lib/private"]
cookie-signed = ["cookie", "cookie-lib/signed"]
erased-json = ["serde_json", "serde"]
form = ["serde", "serde_html_form"]
query = ["serde", "serde_html_form"]
spa = ["tower-http/fs"]
typed-routing = ["axum-macros", "serde", "percent-encoding"]
[dependencies]
axum = { path = "../axum", version = "0.5", default-features = false }
@ -27,6 +29,7 @@ bytes = "1.1.0"
http = "0.2"
mime = "0.3"
pin-project-lite = "0.2"
tokio = "1.19"
tower = { version = "0.4", default_features = false, features = ["util"] }
tower-http = { version = "0.3", features = ["map-response-body"] }
tower-layer = "0.3"
@ -34,11 +37,12 @@ tower-service = "0.3"
# optional dependencies
axum-macros = { path = "../axum-macros", version = "0.2.2", optional = true }
serde = { version = "1.0", optional = true }
serde_json = { version = "1.0.71", optional = true }
percent-encoding = { version = "2.1", optional = true }
cookie-lib = { package = "cookie", version = "0.16", features = ["percent-encode"], optional = true }
percent-encoding = { version = "2.1", optional = true }
serde = { version = "1.0", optional = true }
serde_html_form = { version = "0.1", optional = true }
serde_json = { version = "1.0.71", optional = true }
tokio-util = { version = "0.7", optional = true }
[dev-dependencies]
axum = { path = "../axum", version = "0.5", features = ["headers"] }

View File

@ -0,0 +1,96 @@
use axum::{
body::{self, Bytes, HttpBody, StreamBody},
http::HeaderMap,
response::{IntoResponse, Response},
Error,
};
use pin_project_lite::pin_project;
use std::{
pin::Pin,
task::{Context, Poll},
};
use tokio::io::AsyncRead;
use tokio_util::io::ReaderStream;
pin_project! {
/// An [`HttpBody`] created from an [`AsyncRead`].
///
/// # Example
///
/// `AsyncReadBody` can be used to stream the contents of a file:
///
/// ```rust
/// use axum::{
/// Router,
/// routing::get,
/// http::{StatusCode, header::CONTENT_TYPE},
/// response::{Response, IntoResponse},
/// };
/// use axum_extra::body::AsyncReadBody;
/// use tokio::fs::File;
///
/// async fn cargo_toml() -> Result<Response, (StatusCode, String)> {
/// let file = File::open("Cargo.toml")
/// .await
/// .map_err(|err| {
/// (StatusCode::NOT_FOUND, format!("File not found: {}", err))
/// })?;
///
/// let headers = [(CONTENT_TYPE, "text/x-toml")];
/// let body = AsyncReadBody::new(file);
/// Ok((headers, body).into_response())
/// }
///
/// let app = Router::new().route("/Cargo.toml", get(cargo_toml));
/// # let _: Router = app;
/// ```
#[cfg(feature = "async-read-body")]
#[derive(Debug)]
pub struct AsyncReadBody<R> {
#[pin]
read: StreamBody<ReaderStream<R>>,
}
}
impl<R> AsyncReadBody<R> {
/// Create a new `AsyncReadBody`.
pub fn new(read: R) -> Self
where
R: AsyncRead + Send + 'static,
{
Self {
read: StreamBody::new(ReaderStream::new(read)),
}
}
}
impl<R> HttpBody for AsyncReadBody<R>
where
R: AsyncRead + Send + 'static,
{
type Data = Bytes;
type Error = Error;
fn poll_data(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Option<Result<Self::Data, Self::Error>>> {
self.project().read.poll_data(cx)
}
fn poll_trailers(
self: Pin<&mut Self>,
_cx: &mut Context<'_>,
) -> Poll<Result<Option<HeaderMap>, Self::Error>> {
Poll::Ready(Ok(None))
}
}
impl<R> IntoResponse for AsyncReadBody<R>
where
R: AsyncRead + Send + 'static,
{
fn into_response(self) -> Response {
Response::new(body::boxed(self))
}
}

View File

@ -0,0 +1,7 @@
//! Additional bodies.
#[cfg(feature = "async-read-body")]
mod async_read_body;
#[cfg(feature = "async-read-body")]
pub use self::async_read_body::AsyncReadBody;

View File

@ -43,6 +43,7 @@
#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))]
#![cfg_attr(test, allow(clippy::float_cmp))]
pub mod body;
pub mod extract;
pub mod response;
pub mod routing;