From 27f92d16d8fe9e9a349c448dcedb0367aa2ba0c8 Mon Sep 17 00:00:00 2001 From: tyranron Date: Thu, 26 Sep 2024 17:58:05 +0300 Subject: [PATCH] Improvements and corrections --- juniper_hyper/CHANGELOG.md | 8 +-- juniper_hyper/Cargo.toml | 2 +- juniper_hyper/src/lib.rs | 123 +++++++++++++++++++++---------------- 3 files changed, 76 insertions(+), 57 deletions(-) diff --git a/juniper_hyper/CHANGELOG.md b/juniper_hyper/CHANGELOG.md index 1fd5085ae..225331e52 100644 --- a/juniper_hyper/CHANGELOG.md +++ b/juniper_hyper/CHANGELOG.md @@ -10,9 +10,11 @@ All user visible changes to `juniper_hyper` crate will be documented in this fil ### BC Breaks -- Bumped up [MSRV] to 1.75. ([#1272]) +- Bumped up [MSRV] to 1.79. ([#1263]) +- Made `hyper::Request` in `graphql()` and `graphql_sync()` functions generic over `T: hyper::body::Body`. ([#1263], [#1102]) -[#1272]: /../../pull/1272 +[#1102]: /../../issues/1102 +[#1263]: /../../pull/1263 @@ -25,12 +27,10 @@ All user visible changes to `juniper_hyper` crate will be documented in this fil - Switched to 0.16 version of [`juniper` crate]. - Switched to 1 version of [`hyper` crate]. ([#1217]) - Changed return type of all functions from `Response` to `Response`. ([#1101], [#1096]) -- Add support for `Request` where `T` is a type that implements `hyper::body::Body` trait. ([#1263]) [#1096]: /../../issues/1096 [#1101]: /../../pull/1101 [#1217]: /../../pull/1217 -[#1263]: /../../pull/1263 diff --git a/juniper_hyper/Cargo.toml b/juniper_hyper/Cargo.toml index 0f5f28737..ea8dcecc8 100644 --- a/juniper_hyper/Cargo.toml +++ b/juniper_hyper/Cargo.toml @@ -2,7 +2,7 @@ name = "juniper_hyper" version = "0.9.0" edition = "2021" -rust-version = "1.75" +rust-version = "1.79" description = "`juniper` GraphQL integration with `hyper`." license = "BSD-2-Clause" authors = [ diff --git a/juniper_hyper/src/lib.rs b/juniper_hyper/src/lib.rs index 7105fb042..62b2d848a 100644 --- a/juniper_hyper/src/lib.rs +++ b/juniper_hyper/src/lib.rs @@ -4,7 +4,7 @@ use std::{error::Error, fmt, string::FromUtf8Error, sync::Arc}; use http_body_util::BodyExt as _; use hyper::{ - body, + body::Body, header::{self, HeaderValue}, Method, Request, Response, StatusCode, }; @@ -15,14 +15,7 @@ use juniper::{ use serde_json::error::Error as SerdeError; use url::form_urlencoded; -pub async fn graphql_sync< - CtxT, - QueryT, - MutationT, - SubscriptionT, - S, - T: body::Body, ->( +pub async fn graphql_sync( root_node: Arc>, context: Arc, req: Request, @@ -36,6 +29,7 @@ where SubscriptionT::TypeInfo: Sync, CtxT: Sync, S: ScalarValue + Send + Sync, + T: Body, { match parse_req(req).await { Ok(req) => execute_request_sync(root_node, context, req).await, @@ -43,14 +37,7 @@ where } } -pub async fn graphql< - CtxT, - QueryT, - MutationT, - SubscriptionT, - S, - T: body::Body, ->( +pub async fn graphql( root_node: Arc>, context: Arc, req: Request, @@ -64,6 +51,7 @@ where SubscriptionT::TypeInfo: Sync, CtxT: Sync, S: ScalarValue + Send + Sync, + T: Body, { match parse_req(req).await { Ok(req) => execute_request(root_node, context, req).await, @@ -71,9 +59,11 @@ where } } -async fn parse_req>( - req: Request, -) -> Result, Response> { +async fn parse_req(req: Request) -> Result, Response> +where + S: ScalarValue, + T: Body, +{ match *req.method() { Method::GET => parse_get_req(req), Method::POST => { @@ -92,9 +82,11 @@ async fn parse_req>( .map_err(render_error) } -fn parse_get_req>( - req: Request, -) -> Result, GraphQLRequestError> { +fn parse_get_req(req: Request) -> Result, GraphQLRequestError> +where + S: ScalarValue, + T: Body, +{ req.uri() .query() .map(|q| gql_request_from_get(q).map(GraphQLBatchRequest::Single)) @@ -105,9 +97,13 @@ fn parse_get_req>( }) } -async fn parse_post_json_req>( +async fn parse_post_json_req( body: T, -) -> Result, GraphQLRequestError> { +) -> Result, GraphQLRequestError> +where + S: ScalarValue, + T: Body, +{ let chunk = body .collect() .await @@ -120,9 +116,13 @@ async fn parse_post_json_req( +async fn parse_post_graphql_req( body: T, -) -> Result, GraphQLRequestError> { +) -> Result, GraphQLRequestError> +where + S: ScalarValue, + T: Body, +{ let chunk = body .collect() .await @@ -157,9 +157,10 @@ pub async fn playground( resp } -fn render_error>( - err: GraphQLRequestError, -) -> Response { +fn render_error(err: GraphQLRequestError) -> Response +where + T: Body, +{ let mut resp = new_response(StatusCode::BAD_REQUEST); *resp.body_mut() = err.to_string(); resp @@ -227,11 +228,12 @@ where resp } -fn gql_request_from_get( +fn gql_request_from_get( input: &str, ) -> Result, GraphQLRequestError> where S: ScalarValue, + T: Body, { let mut query = None; let mut operation_name = None; @@ -272,7 +274,7 @@ where } } -fn invalid_err(parameter_name: &str) -> GraphQLRequestError { +fn invalid_err(parameter_name: &str) -> GraphQLRequestError { GraphQLRequestError::Invalid(format!( "`{parameter_name}` parameter is specified multiple times", )) @@ -293,8 +295,7 @@ fn new_html_response(code: StatusCode) -> Response { resp } -#[derive(Debug)] -enum GraphQLRequestError { +enum GraphQLRequestError { BodyHyper(T::Error), BodyUtf8(FromUtf8Error), BodyJSONError(SerdeError), @@ -302,29 +303,49 @@ enum GraphQLRequestError { Invalid(String), } -impl> fmt::Display for GraphQLRequestError { +// NOTE: Manual implementation instead of `#[derive(Debug)]` is used to omit imposing unnecessary +// `T: Debug` bound on the implementation. +impl fmt::Debug for GraphQLRequestError +where + T: Body, +{ + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { + match self { + Self::BodyHyper(e) => fmt::Debug::fmt(e, f), + Self::BodyUtf8(e) => fmt::Debug::fmt(e, f), + Self::BodyJSONError(e) => fmt::Debug::fmt(e, f), + Self::Variables(e) => fmt::Debug::fmt(e, f), + Self::Invalid(e) => fmt::Debug::fmt(e, f), + } + } +} + +impl fmt::Display for GraphQLRequestError +where + T: Body, +{ fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { match self { - GraphQLRequestError::BodyHyper(err) => fmt::Display::fmt(err, f), - GraphQLRequestError::BodyUtf8(err) => fmt::Display::fmt(err, f), - GraphQLRequestError::BodyJSONError(err) => fmt::Display::fmt(err, f), - GraphQLRequestError::Variables(err) => fmt::Display::fmt(err, f), - GraphQLRequestError::Invalid(err) => fmt::Display::fmt(err, f), + Self::BodyHyper(e) => fmt::Display::fmt(e, f), + Self::BodyUtf8(e) => fmt::Display::fmt(e, f), + Self::BodyJSONError(e) => fmt::Display::fmt(e, f), + Self::Variables(e) => fmt::Display::fmt(e, f), + Self::Invalid(e) => fmt::Display::fmt(e, f), } } } -impl + std::fmt::Debug> Error for GraphQLRequestError +impl Error for GraphQLRequestError where - ::Error: std::fmt::Debug, + T: Body, { fn source(&self) -> Option<&(dyn Error + 'static)> { match self { - GraphQLRequestError::BodyHyper(err) => Some(err), - GraphQLRequestError::BodyUtf8(err) => Some(err), - GraphQLRequestError::BodyJSONError(err) => Some(err), - GraphQLRequestError::Variables(err) => Some(err), - GraphQLRequestError::Invalid(_) => None, + Self::BodyHyper(e) => Some(e), + Self::BodyUtf8(e) => Some(e), + Self::BodyJSONError(e) => Some(e), + Self::Variables(e) => Some(e), + Self::Invalid(_) => None, } } } @@ -335,7 +356,7 @@ mod tests { convert::Infallible, error::Error, net::SocketAddr, panic, sync::Arc, time::Duration, }; - use http_body_util::BodyExt; + use http_body_util::BodyExt as _; use hyper::{ body::Incoming, server::conn::http1, service::service_fn, Method, Request, Response, StatusCode, @@ -513,14 +534,12 @@ mod tests { } #[tokio::test] - /// run test for a custom request type - `Request>` - async fn test_custom_hyper_integration() { + async fn test_custom_request_hyper_integration() { run_hyper_integration(3002, false, false).await } #[tokio::test] - /// run test for a custom request type - `Request>` in sync mode - async fn test_custom_sync_hyper_integration() { + async fn test_custom_request_sync_hyper_integration() { run_hyper_integration(3003, true, true).await } }