diff --git a/juniper_axum/CHANGELOG.md b/juniper_axum/CHANGELOG.md
index 60232e595..0e2b7a563 100644
--- a/juniper_axum/CHANGELOG.md
+++ b/juniper_axum/CHANGELOG.md
@@ -6,6 +6,18 @@ All user visible changes to `juniper_axum` crate will be documented in this file
+## master
+
+### Fixed
+
+- `Content-Type` header reading full value instead of just media type. ([#1289], [#1288])
+
+[#1288]: /../../issues/1288
+[#1289]: /../../pull/1289
+
+
+
+
## [0.1.0] ยท 2024-03-20
[0.1.0]: /../../tree/juniper_axum-v0.1.0/juniper_axum
diff --git a/juniper_axum/src/extract.rs b/juniper_axum/src/extract.rs
index fe087f672..1029d05e0 100644
--- a/juniper_axum/src/extract.rs
+++ b/juniper_axum/src/extract.rs
@@ -6,7 +6,7 @@ use axum::{
async_trait,
body::Body,
extract::{FromRequest, FromRequestParts, Query},
- http::{HeaderValue, Method, Request, StatusCode},
+ http::{header, HeaderValue, Method, Request, StatusCode},
response::{IntoResponse as _, Response},
Json, RequestExt as _,
};
@@ -85,7 +85,7 @@ where
async fn from_request(mut req: Request
, state: &State) -> Result {
let content_type = req
.headers()
- .get("content-type")
+ .get(header::CONTENT_TYPE)
.map(HeaderValue::to_str)
.transpose()
.map_err(|_| {
@@ -122,7 +122,7 @@ where
.into_response()
})
}),
- (&Method::POST, Some("application/json")) => {
+ (&Method::POST, Some(x)) if x.starts_with("application/json") => {
Json::>::from_request(req, state)
.await
.map(|req| Self(req.0))
@@ -130,14 +130,16 @@ where
(StatusCode::BAD_REQUEST, format!("Invalid JSON body: {e}")).into_response()
})
}
- (&Method::POST, Some("application/graphql")) => String::from_request(req, state)
- .await
- .map(|body| {
- Self(GraphQLBatchRequest::Single(GraphQLRequest::new(
- body, None, None,
- )))
- })
- .map_err(|_| (StatusCode::BAD_REQUEST, "Not valid UTF-8 body").into_response()),
+ (&Method::POST, Some(x)) if x.starts_with("application/graphql") => {
+ String::from_request(req, state)
+ .await
+ .map(|body| {
+ Self(GraphQLBatchRequest::Single(GraphQLRequest::new(
+ body, None, None,
+ )))
+ })
+ .map_err(|_| (StatusCode::BAD_REQUEST, "Not valid UTF-8 body").into_response())
+ }
(&Method::POST, _) => Err((
StatusCode::UNSUPPORTED_MEDIA_TYPE,
"`Content-Type` header is expected to be either `application/json` or \
@@ -246,6 +248,22 @@ mod juniper_request_tests {
assert_eq!(do_from_request(req).await, expected);
}
+ #[tokio::test]
+ async fn from_json_post_request_with_charset() {
+ let req = Request::post("/")
+ .header("content-type", "application/json; charset=utf-8")
+ .body(Body::from(r#"{"query": "{ add(a: 2, b: 3) }"}"#))
+ .unwrap_or_else(|e| panic!("cannot build `Request`: {e}"));
+
+ let expected = JuniperRequest(GraphQLBatchRequest::Single(GraphQLRequest::new(
+ "{ add(a: 2, b: 3) }".to_string(),
+ None,
+ None,
+ )));
+
+ assert_eq!(do_from_request(req).await, expected);
+ }
+
#[tokio::test]
async fn from_graphql_post_request() {
let req = Request::post("/")
@@ -262,6 +280,22 @@ mod juniper_request_tests {
assert_eq!(do_from_request(req).await, expected);
}
+ #[tokio::test]
+ async fn from_graphql_post_request_with_charset() {
+ let req = Request::post("/")
+ .header("content-type", "application/graphql; charset=utf-8")
+ .body(Body::from(r#"{ add(a: 2, b: 3) }"#))
+ .unwrap_or_else(|e| panic!("cannot build `Request`: {e}"));
+
+ let expected = JuniperRequest(GraphQLBatchRequest::Single(GraphQLRequest::new(
+ "{ add(a: 2, b: 3) }".to_string(),
+ None,
+ None,
+ )));
+
+ assert_eq!(do_from_request(req).await, expected);
+ }
+
/// Performs [`JuniperRequest::from_request()`].
async fn do_from_request(req: Request) -> JuniperRequest {
match JuniperRequest::from_request(req, &()).await {