mirror of
https://github.com/tokio-rs/axum.git
synced 2025-03-13 19:27:53 +01:00
Add Headers
response (#193)
* Add `Headers` Example usage: ```rust use axum::{ route, routing::RoutingDsl, response::{IntoResponse, Headers}, handler::get, }; use http::header::{HeaderName, HeaderValue}; // It works with any `IntoIterator<Item = (Key, Value)>` where `Key` can be // turned into a `HeaderName` and `Value` can be turned into a `HeaderValue` // // Such as `Vec<(HeaderName, HeaderValue)>` async fn just_headers() -> impl IntoResponse { Headers(vec![ (HeaderName::from_static("X-Foo"), HeaderValue::from_static("foo")), ]) } // Or `[(&str, &str)]` async fn from_strings() -> impl IntoResponse { Headers([("X-Foo", "foo")]) } ``` Fixes https://github.com/tokio-rs/axum/issues/187 * Make work on Rust versions without `IntoIterator` for arrays * format * changelog
This commit is contained in:
parent
baa99e5084
commit
97c140cdf7
4 changed files with 243 additions and 8 deletions
|
@ -1 +1 @@
|
|||
msrv = "1.40"
|
||||
msrv = "1.51"
|
||||
|
|
|
@ -14,7 +14,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||
- Add `NestedUri` for extracting request URI in nested services ([#161](https://github.com/tokio-rs/axum/pull/161))
|
||||
- Implement `FromRequest` for `http::Extensions`
|
||||
- Implement SSE as an `IntoResponse` instead of a service ([#98](https://github.com/tokio-rs/axum/pull/98))
|
||||
- Add `Redirect` response. ([#192](https://github.com/tokio-rs/axum/pull/192))
|
||||
- Add `Headers` for easily customizing headers on a response ([#193](https://github.com/tokio-rs/axum/pull/193))
|
||||
- Add `Redirect` response ([#192](https://github.com/tokio-rs/axum/pull/192))
|
||||
- Make `RequestParts::{new, try_into_request}` public ([#194](https://github.com/tokio-rs/axum/pull/194))
|
||||
|
||||
## Breaking changes
|
||||
|
|
229
src/response/headers.rs
Normal file
229
src/response/headers.rs
Normal file
|
@ -0,0 +1,229 @@
|
|||
use super::IntoResponse;
|
||||
use crate::body::{box_body, BoxBody};
|
||||
use bytes::Bytes;
|
||||
use http::header::{HeaderMap, HeaderName, HeaderValue};
|
||||
use http::{Response, StatusCode};
|
||||
use http_body::{Body, Full};
|
||||
use std::{convert::TryInto, fmt};
|
||||
use tower::{util::Either, BoxError};
|
||||
|
||||
/// A response with headers.
|
||||
///
|
||||
/// # Example
|
||||
///
|
||||
/// ```rust
|
||||
/// use axum::{
|
||||
/// route,
|
||||
/// routing::RoutingDsl,
|
||||
/// response::{IntoResponse, Headers},
|
||||
/// handler::get,
|
||||
/// };
|
||||
/// use http::header::{HeaderName, HeaderValue};
|
||||
///
|
||||
/// // It works with any `IntoIterator<Item = (Key, Value)>` where `Key` can be
|
||||
/// // turned into a `HeaderName` and `Value` can be turned into a `HeaderValue`
|
||||
/// //
|
||||
/// // Such as `Vec<(HeaderName, HeaderValue)>`
|
||||
/// async fn just_headers() -> impl IntoResponse {
|
||||
/// Headers(vec![
|
||||
/// (HeaderName::from_static("X-Foo"), HeaderValue::from_static("foo")),
|
||||
/// ])
|
||||
/// }
|
||||
///
|
||||
/// // Or `Vec<(&str, &str)>`
|
||||
/// async fn from_strings() -> impl IntoResponse {
|
||||
/// Headers(vec![("X-Foo", "foo")])
|
||||
/// }
|
||||
///
|
||||
/// // Or `[(&str, &str)]` if you're on Rust 1.53+
|
||||
///
|
||||
/// let app = route("/just-headers", get(just_headers))
|
||||
/// .route("/from-strings", get(from_strings));
|
||||
/// # async {
|
||||
/// # axum::Server::bind(&"".parse().unwrap()).serve(app.into_make_service()).await.unwrap();
|
||||
/// # };
|
||||
/// ```
|
||||
///
|
||||
/// If a conversion to `HeaderName` or `HeaderValue` fails a `500 Internal
|
||||
/// Server Error` response will be returned.
|
||||
///
|
||||
/// You can also return `(Headers, impl IntoResponse)` to customize the headers
|
||||
/// of a response, or `(StatusCode, Headeres, impl IntoResponse)` to customize
|
||||
/// the status code and headers.
|
||||
#[derive(Clone, Copy, Debug)]
|
||||
pub struct Headers<H>(pub H);
|
||||
|
||||
impl<H> Headers<H> {
|
||||
fn try_into_header_map<K, V>(self) -> Result<HeaderMap, Response<Full<Bytes>>>
|
||||
where
|
||||
H: IntoIterator<Item = (K, V)>,
|
||||
K: TryInto<HeaderName>,
|
||||
K::Error: fmt::Display,
|
||||
V: TryInto<HeaderValue>,
|
||||
V::Error: fmt::Display,
|
||||
{
|
||||
self.0
|
||||
.into_iter()
|
||||
.map(|(key, value)| {
|
||||
let key = key.try_into().map_err(Either::A)?;
|
||||
let value = value.try_into().map_err(Either::B)?;
|
||||
Ok((key, value))
|
||||
})
|
||||
.collect::<Result<_, _>>()
|
||||
.map_err(|err| {
|
||||
let err = match err {
|
||||
Either::A(err) => err.to_string(),
|
||||
Either::B(err) => err.to_string(),
|
||||
};
|
||||
|
||||
let body = Full::new(Bytes::copy_from_slice(err.as_bytes()));
|
||||
let mut res = Response::new(body);
|
||||
*res.status_mut() = StatusCode::INTERNAL_SERVER_ERROR;
|
||||
res
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
impl<H, K, V> IntoResponse for Headers<H>
|
||||
where
|
||||
H: IntoIterator<Item = (K, V)>,
|
||||
K: TryInto<HeaderName>,
|
||||
K::Error: fmt::Display,
|
||||
V: TryInto<HeaderValue>,
|
||||
V::Error: fmt::Display,
|
||||
{
|
||||
type Body = Full<Bytes>;
|
||||
type BodyError = <Self::Body as Body>::Error;
|
||||
|
||||
fn into_response(self) -> http::Response<Self::Body> {
|
||||
let headers = self.try_into_header_map();
|
||||
|
||||
match headers {
|
||||
Ok(headers) => {
|
||||
let mut res = Response::new(Full::new(Bytes::new()));
|
||||
*res.headers_mut() = headers;
|
||||
res
|
||||
}
|
||||
Err(err) => err,
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl<H, T, K, V> IntoResponse for (Headers<H>, T)
|
||||
where
|
||||
T: IntoResponse,
|
||||
T::Body: Body<Data = Bytes> + Send + Sync + 'static,
|
||||
<T::Body as Body>::Error: Into<BoxError>,
|
||||
H: IntoIterator<Item = (K, V)>,
|
||||
K: TryInto<HeaderName>,
|
||||
K::Error: fmt::Display,
|
||||
V: TryInto<HeaderValue>,
|
||||
V::Error: fmt::Display,
|
||||
{
|
||||
type Body = BoxBody;
|
||||
type BodyError = <Self::Body as Body>::Error;
|
||||
|
||||
// this boxing could be improved with a EitherBody but thats
|
||||
// an issue for another time
|
||||
fn into_response(self) -> Response<Self::Body> {
|
||||
let headers = match self.0.try_into_header_map() {
|
||||
Ok(headers) => headers,
|
||||
Err(res) => return res.map(box_body),
|
||||
};
|
||||
|
||||
(headers, self.1).into_response().map(box_body)
|
||||
}
|
||||
}
|
||||
|
||||
impl<H, T, K, V> IntoResponse for (StatusCode, Headers<H>, T)
|
||||
where
|
||||
T: IntoResponse,
|
||||
T::Body: Body<Data = Bytes> + Send + Sync + 'static,
|
||||
<T::Body as Body>::Error: Into<BoxError>,
|
||||
H: IntoIterator<Item = (K, V)>,
|
||||
K: TryInto<HeaderName>,
|
||||
K::Error: fmt::Display,
|
||||
V: TryInto<HeaderValue>,
|
||||
V::Error: fmt::Display,
|
||||
{
|
||||
type Body = BoxBody;
|
||||
type BodyError = <Self::Body as Body>::Error;
|
||||
|
||||
fn into_response(self) -> Response<Self::Body> {
|
||||
let headers = match self.1.try_into_header_map() {
|
||||
Ok(headers) => headers,
|
||||
Err(res) => return res.map(box_body),
|
||||
};
|
||||
|
||||
(self.0, headers, self.2).into_response().map(box_body)
|
||||
}
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use super::*;
|
||||
use futures_util::FutureExt;
|
||||
use http::header::USER_AGENT;
|
||||
|
||||
#[test]
|
||||
fn vec_of_header_name_and_value() {
|
||||
let res = Headers(vec![(USER_AGENT, HeaderValue::from_static("axum"))]).into_response();
|
||||
|
||||
assert_eq!(res.headers()["user-agent"], "axum");
|
||||
assert_eq!(res.status(), StatusCode::OK);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn vec_of_strings() {
|
||||
let res = Headers(vec![("user-agent", "axum")]).into_response();
|
||||
|
||||
assert_eq!(res.headers()["user-agent"], "axum");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn with_body() {
|
||||
let res = (Headers(vec![("user-agent", "axum")]), "foo").into_response();
|
||||
|
||||
assert_eq!(res.headers()["user-agent"], "axum");
|
||||
let body = hyper::body::to_bytes(res.into_body())
|
||||
.now_or_never()
|
||||
.unwrap()
|
||||
.unwrap();
|
||||
assert_eq!(&body[..], b"foo");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn with_status_and_body() {
|
||||
let res = (
|
||||
StatusCode::NOT_FOUND,
|
||||
Headers(vec![("user-agent", "axum")]),
|
||||
"foo",
|
||||
)
|
||||
.into_response();
|
||||
|
||||
assert_eq!(res.headers()["user-agent"], "axum");
|
||||
assert_eq!(res.status(), StatusCode::NOT_FOUND);
|
||||
let body = hyper::body::to_bytes(res.into_body())
|
||||
.now_or_never()
|
||||
.unwrap()
|
||||
.unwrap();
|
||||
assert_eq!(&body[..], b"foo");
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn invalid_header_name() {
|
||||
let bytes: &[u8] = &[0, 159, 146, 150]; // invalid utf-8
|
||||
let res = Headers(vec![(bytes, "axum")]).into_response();
|
||||
|
||||
assert_eq!(res.status(), StatusCode::INTERNAL_SERVER_ERROR);
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn invalid_header_value() {
|
||||
let bytes: &[u8] = &[0, 159, 146, 150]; // invalid utf-8
|
||||
let res = Headers(vec![("user-agent", bytes)]).into_response();
|
||||
|
||||
assert!(res.headers().get("user-agent").is_none());
|
||||
assert_eq!(res.status(), StatusCode::INTERNAL_SERVER_ERROR);
|
||||
}
|
||||
}
|
|
@ -13,15 +13,20 @@ use http_body::{
|
|||
use std::{borrow::Cow, convert::Infallible};
|
||||
use tower::{util::Either, BoxError};
|
||||
|
||||
mod headers;
|
||||
mod redirect;
|
||||
|
||||
pub mod sse;
|
||||
|
||||
#[doc(no_inline)]
|
||||
pub use crate::Json;
|
||||
|
||||
mod redirect;
|
||||
|
||||
pub use self::redirect::Redirect;
|
||||
|
||||
pub mod sse;
|
||||
pub use sse::{sse, Sse};
|
||||
#[doc(inline)]
|
||||
pub use self::{
|
||||
headers::Headers,
|
||||
redirect::Redirect,
|
||||
sse::{sse, Sse},
|
||||
};
|
||||
|
||||
/// Trait for generating responses.
|
||||
///
|
||||
|
|
Loading…
Add table
Reference in a new issue