From bb529cca976272602618ec4bd2b34c196127526a Mon Sep 17 00:00:00 2001 From: Joe Grund Date: Mon, 28 Oct 2024 12:54:21 +0100 Subject: [PATCH 1/2] Backport fix of `Content-Type` header parsing in `juniper_axum` (#1289, #1288) Signed-off-by: Joe Grund --- juniper_axum/CHANGELOG.md | 12 ++++++++ juniper_axum/src/extract.rs | 56 +++++++++++++++++++++++++++++-------- 2 files changed, 57 insertions(+), 11 deletions(-) diff --git a/juniper_axum/CHANGELOG.md b/juniper_axum/CHANGELOG.md index 60232e59..0e2b7a56 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 fe087f67..1029d05e 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 { From ab85a28cf15b43a7240bdec305e4a03daff289a8 Mon Sep 17 00:00:00 2001 From: tyranron Date: Mon, 28 Oct 2024 12:58:35 +0100 Subject: [PATCH 2/2] Prepare 0.1.1 release of `juniper_axum` crate --- juniper_axum/CHANGELOG.md | 3 ++- juniper_axum/Cargo.toml | 2 +- juniper_axum/README.md | 8 ++++---- 3 files changed, 7 insertions(+), 6 deletions(-) diff --git a/juniper_axum/CHANGELOG.md b/juniper_axum/CHANGELOG.md index 0e2b7a56..db290864 100644 --- a/juniper_axum/CHANGELOG.md +++ b/juniper_axum/CHANGELOG.md @@ -6,7 +6,8 @@ All user visible changes to `juniper_axum` crate will be documented in this file -## master +## [0.1.1] · 2024-10-28 +[0.1.1]: /../../tree/juniper_axum-v0.1.1/juniper_axum ### Fixed diff --git a/juniper_axum/Cargo.toml b/juniper_axum/Cargo.toml index 61148a09..2805f57d 100644 --- a/juniper_axum/Cargo.toml +++ b/juniper_axum/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "juniper_axum" -version = "0.1.0" +version = "0.1.1" edition = "2021" rust-version = "1.73" description = "`juniper` GraphQL integration with `axum`." diff --git a/juniper_axum/README.md b/juniper_axum/README.md index 54356c87..4b674443 100644 --- a/juniper_axum/README.md +++ b/juniper_axum/README.md @@ -6,7 +6,7 @@ [![CI](https://github.com/graphql-rust/juniper/workflows/CI/badge.svg?branch=master "CI")](https://github.com/graphql-rust/juniper/actions?query=workflow%3ACI+branch%3Amaster) [![Rust 1.73+](https://img.shields.io/badge/rustc-1.73+-lightgray.svg "Rust 1.73+")](https://blog.rust-lang.org/2023/10/05/Rust-1.73.0.html) -- [Changelog](https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.0/juniper_axum/CHANGELOG.md) +- [Changelog](https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.1/juniper_axum/CHANGELOG.md) [`axum`] web server integration for [`juniper`] ([GraphQL] implementation for [Rust]). @@ -31,7 +31,7 @@ Check [`examples/simple.rs`][1] and [`examples/custom.rs`][1] for example code o ## License -This project is licensed under [BSD 2-Clause License](https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.0/juniper_axum/LICENSE). +This project is licensed under [BSD 2-Clause License](https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.1/juniper_axum/LICENSE). @@ -43,5 +43,5 @@ This project is licensed under [BSD 2-Clause License](https://github.com/graphql [Juniper Book]: https://graphql-rust.github.io/juniper [Rust]: https://www.rust-lang.org -[1]: https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.0/juniper_axum/examples/simple.rs -[2]: https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.0/juniper_axum/examples/custom.rs +[1]: https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.1/juniper_axum/examples/simple.rs +[2]: https://github.com/graphql-rust/juniper/blob/juniper_axum-v0.1.1/juniper_axum/examples/custom.rs