diff --git a/tonic-types/Cargo.toml b/tonic-types/Cargo.toml index 0e13a66ec..c6f259b37 100644 --- a/tonic-types/Cargo.toml +++ b/tonic-types/Cargo.toml @@ -1,7 +1,7 @@ [package] authors = [ "Lucio Franco ", - "Rafael Lemos " + "Rafael Lemos ", ] categories = ["web-programming", "network-programming", "asynchronous"] description = """ @@ -18,6 +18,7 @@ repository = "https://github.com/hyperium/tonic" version = "0.11.0" [dependencies] +http = "0.2" prost = "0.12" prost-types = "0.12" -tonic = {version = "0.11", path = "../tonic", default-features = false} +tonic = { version = "0.11", path = "../tonic", default-features = false } diff --git a/tonic-types/README.md b/tonic-types/README.md index 1818a1400..a33e8277d 100644 --- a/tonic-types/README.md +++ b/tonic-types/README.md @@ -6,6 +6,8 @@ This crate also introduces the [`StatusExt`] trait and implements it in [`tonic::Status`], allowing the implementation of the [gRPC Richer Error Model] with [`tonic`] in a convenient way. +[`CodeExt`] adds [gRPC Richer Error Model] functionality to [`tonic::Code`]. Notably [`http::status::StatusCode`] mapping as described [here](https://cloud.google.com/apis/design/errors#generating_errors) and [here](https://github.com/googleapis/googleapis/blob/master/google/rpc/code.proto). + ## Usage Useful protobuf types are available through the [`pb`] module. They can be @@ -130,16 +132,19 @@ more direct way of extracting a [`BadRequest`] error message from [`tonic::Status`]. [`tonic::Status`]: https://docs.rs/tonic/latest/tonic/struct.Status.html +[`tonic::Code`]: https://docs.rs/tonic/latest/tonic/enum.Code.html [`tonic`]: https://docs.rs/tonic/latest/tonic/ [gRPC Richer Error Model]: https://www.grpc.io/docs/guides/error/ [`pb`]: https://docs.rs/tonic-types/latest/tonic_types/pb/index.html -[`StatusExt`]: https://docs.rs/tonic-types/latest/tonic_types/trait.StatusExt.html [examples]: https://github.com/hyperium/tonic/tree/master/examples +[`ErrorDetail`]: https://docs.rs/tonic-types/latest/tonic_types/enum.ErrorDetail.html [`ErrorDetails`]: https://docs.rs/tonic-types/latest/tonic_types/struct.ErrorDetails.html [error_details.proto]: https://github.com/googleapis/googleapis/blob/master/google/rpc/error_details.proto -[`ErrorDetail`]: https://docs.rs/tonic-types/latest/tonic_types/enum.ErrorDetail.html +[`StatusExt`]: https://docs.rs/tonic-types/latest/tonic_types/trait.StatusExt.html [`StatusExt::with_error_details_vec`]: https://docs.rs/tonic-types/latest/tonic_types/trait.StatusExt.html#tymethod.with_error_details_vec [`StatusExt::get_error_details_vec`]: https://docs.rs/tonic-types/latest/tonic_types/trait.StatusExt.html#tymethod.get_error_details_vec -[Richer Error example]: https://github.com/hyperium/tonic/tree/master/examples/src/richer-error [`StatusExt::get_details_bad_request`]: https://docs.rs/tonic-types/latest/tonic_types/trait.StatusExt.html#tymethod.get_details_bad_request -[`BadRequest`]: https://docs.rs/tonic-types/latest/tonic_types/struct.BadRequest.html \ No newline at end of file +[`CodeExt`]: https://docs.rs/tonic-types/latest/tonic_types/trait.CodeExt.html +[`http::status::StatusCode`]: https://docs.rs/http/latest/http/status/struct.StatusCode.html +[Richer Error example]: https://github.com/hyperium/tonic/tree/master/examples/src/richer-error +[`BadRequest`]: https://docs.rs/tonic-types/latest/tonic_types/struct.BadRequest.html diff --git a/tonic-types/src/lib.rs b/tonic-types/src/lib.rs index fe2886a60..e2d8cdc9f 100644 --- a/tonic-types/src/lib.rs +++ b/tonic-types/src/lib.rs @@ -182,12 +182,7 @@ pub mod pb { pub use pb::Status; mod richer_error; - -pub use richer_error::{ - BadRequest, DebugInfo, ErrorDetail, ErrorDetails, ErrorInfo, FieldViolation, Help, HelpLink, - LocalizedMessage, PreconditionFailure, PreconditionViolation, QuotaFailure, QuotaViolation, - RequestInfo, ResourceInfo, RetryInfo, RpcStatusExt, StatusExt, -}; +pub use richer_error::*; mod sealed { pub trait Sealed {} diff --git a/tonic-types/src/richer_error/code_ext.rs b/tonic-types/src/richer_error/code_ext.rs new file mode 100644 index 000000000..02432d5de --- /dev/null +++ b/tonic-types/src/richer_error/code_ext.rs @@ -0,0 +1,32 @@ +use http::status::StatusCode as HttpCode; +use tonic::Code; + +/// Add gRPC Richer Error Model functionality to [`tonic::Code`]. +pub trait CodeExt: crate::sealed::Sealed { + /// gRPC to HTTP status code mappings as described in + /// . + fn http_status(&self) -> http::status::StatusCode; +} + +impl CodeExt for Code { + fn http_status(&self) -> HttpCode { + match self { + Code::Ok => HttpCode::OK, + Code::InvalidArgument | Code::FailedPrecondition | Code::OutOfRange => { + HttpCode::BAD_REQUEST + } + Code::PermissionDenied => HttpCode::FORBIDDEN, + Code::NotFound => HttpCode::NOT_FOUND, + Code::Aborted | Code::AlreadyExists => HttpCode::CONFLICT, + Code::Unauthenticated => HttpCode::UNAUTHORIZED, + Code::ResourceExhausted => HttpCode::TOO_MANY_REQUESTS, + Code::Cancelled => HttpCode::from_u16(499).expect("ivalid HTTP status code"), + Code::DataLoss | Code::Unknown | Code::Internal => HttpCode::INTERNAL_SERVER_ERROR, + Code::Unimplemented => HttpCode::NOT_IMPLEMENTED, + Code::Unavailable => HttpCode::SERVICE_UNAVAILABLE, + Code::DeadlineExceeded => HttpCode::GATEWAY_TIMEOUT, + } + } +} + +impl crate::sealed::Sealed for Code {} diff --git a/tonic-types/src/richer_error/error_details/mod.rs b/tonic-types/src/richer_error/error_details/mod.rs index 70074aa9c..ebad231a7 100644 --- a/tonic-types/src/richer_error/error_details/mod.rs +++ b/tonic-types/src/richer_error/error_details/mod.rs @@ -1,12 +1,8 @@ -use std::{collections::HashMap, time}; +pub(super) mod vec; -use super::std_messages::{ - BadRequest, DebugInfo, ErrorInfo, FieldViolation, Help, HelpLink, LocalizedMessage, - PreconditionFailure, PreconditionViolation, QuotaFailure, QuotaViolation, RequestInfo, - ResourceInfo, RetryInfo, -}; +use std::{collections::HashMap, time}; -pub(crate) mod vec; +use super::std_messages::*; /// Groups the standard error messages structs. Provides associated /// functions and methods to setup and edit each error message independently. diff --git a/tonic-types/src/richer_error/error_details/vec.rs b/tonic-types/src/richer_error/error_details/vec.rs index 9a09285e8..978d92a32 100644 --- a/tonic-types/src/richer_error/error_details/vec.rs +++ b/tonic-types/src/richer_error/error_details/vec.rs @@ -1,7 +1,4 @@ -use super::super::std_messages::{ - BadRequest, DebugInfo, ErrorInfo, Help, LocalizedMessage, PreconditionFailure, QuotaFailure, - RequestInfo, ResourceInfo, RetryInfo, -}; +use super::super::std_messages::*; /// Wraps the structs corresponding to the standard error messages, allowing /// the implementation and handling of vectors containing any of them. diff --git a/tonic-types/src/richer_error/helpers.rs b/tonic-types/src/richer_error/helpers.rs new file mode 100644 index 000000000..0c160de9a --- /dev/null +++ b/tonic-types/src/richer_error/helpers.rs @@ -0,0 +1,33 @@ +use prost::{ + bytes::{Bytes, BytesMut}, + DecodeError, Message, +}; +use prost_types::Any; +use tonic::Code; + +use crate::pb; + +pub(super) trait IntoAny { + fn into_any(self) -> Any; +} + +pub(super) trait FromAnyRef { + fn from_any_ref(any: &Any) -> Result + where + Self: Sized; +} + +pub(super) fn gen_details_bytes(code: Code, message: &str, details: Vec) -> Bytes { + let status = pb::Status { + code: code as i32, + message: message.to_owned(), + details, + }; + + let mut buf = BytesMut::with_capacity(status.encoded_len()); + + // Should never panic since `buf` is initialized with sufficient capacity + status.encode(&mut buf).unwrap(); + + buf.freeze() +} diff --git a/tonic-types/src/richer_error/mod.rs b/tonic-types/src/richer_error/mod.rs index 927bbc97d..d8883c7f6 100644 --- a/tonic-types/src/richer_error/mod.rs +++ b/tonic-types/src/richer_error/mod.rs @@ -1,1068 +1,17 @@ -use prost::{ - bytes::{Bytes, BytesMut}, - DecodeError, Message, -}; -use prost_types::Any; -use tonic::{metadata::MetadataMap, Code}; - mod error_details; -mod std_messages; - -use super::pb; - pub use error_details::{vec::ErrorDetail, ErrorDetails}; -pub use std_messages::{ - BadRequest, DebugInfo, ErrorInfo, FieldViolation, Help, HelpLink, LocalizedMessage, - PreconditionFailure, PreconditionViolation, QuotaFailure, QuotaViolation, RequestInfo, - ResourceInfo, RetryInfo, -}; - -trait IntoAny { - fn into_any(self) -> Any; -} - -#[allow(dead_code)] -trait FromAny { - fn from_any(any: Any) -> Result - where - Self: Sized; -} - -trait FromAnyRef { - fn from_any_ref(any: &Any) -> Result - where - Self: Sized; -} - -fn gen_details_bytes(code: Code, message: &str, details: Vec) -> Bytes { - let status = pb::Status { - code: code as i32, - message: message.to_owned(), - details, - }; - - let mut buf = BytesMut::with_capacity(status.encoded_len()); - - // Should never panic since `buf` is initialized with sufficient capacity - status.encode(&mut buf).unwrap(); - - buf.freeze() -} - -/// Used to implement associated functions and methods on `tonic::Status`, that -/// allow the addition and extraction of standard error details. This trait is -/// sealed and not meant to be implemented outside of `tonic-types`. -pub trait StatusExt: crate::sealed::Sealed { - /// Generates a `tonic::Status` with error details obtained from an - /// [`ErrorDetails`] struct, and custom metadata. - /// - /// # Examples - /// - /// ``` - /// use tonic::{metadata::MetadataMap, Code, Status}; - /// use tonic_types::{ErrorDetails, StatusExt}; - /// - /// let status = Status::with_error_details_and_metadata( - /// Code::InvalidArgument, - /// "bad request", - /// ErrorDetails::with_bad_request_violation("field", "description"), - /// MetadataMap::new() - /// ); - /// ``` - fn with_error_details_and_metadata( - code: Code, - message: impl Into, - details: ErrorDetails, - metadata: MetadataMap, - ) -> tonic::Status; - - /// Generates a `tonic::Status` with error details obtained from an - /// [`ErrorDetails`] struct. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Code, Status}; - /// use tonic_types::{ErrorDetails, StatusExt}; - /// - /// let status = Status::with_error_details( - /// Code::InvalidArgument, - /// "bad request", - /// ErrorDetails::with_bad_request_violation("field", "description"), - /// ); - /// ``` - fn with_error_details( - code: Code, - message: impl Into, - details: ErrorDetails, - ) -> tonic::Status; - - /// Generates a `tonic::Status` with error details provided in a vector of - /// [`ErrorDetail`] enums, and custom metadata. - /// - /// # Examples - /// - /// ``` - /// use tonic::{metadata::MetadataMap, Code, Status}; - /// use tonic_types::{BadRequest, StatusExt}; - /// - /// let status = Status::with_error_details_vec_and_metadata( - /// Code::InvalidArgument, - /// "bad request", - /// vec![ - /// BadRequest::with_violation("field", "description").into(), - /// ], - /// MetadataMap::new() - /// ); - /// ``` - fn with_error_details_vec_and_metadata( - code: Code, - message: impl Into, - details: impl IntoIterator, - metadata: MetadataMap, - ) -> tonic::Status; - - /// Generates a `tonic::Status` with error details provided in a vector of - /// [`ErrorDetail`] enums. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Code, Status}; - /// use tonic_types::{BadRequest, StatusExt}; - /// - /// let status = Status::with_error_details_vec( - /// Code::InvalidArgument, - /// "bad request", - /// vec![ - /// BadRequest::with_violation("field", "description").into(), - /// ] - /// ); - /// ``` - fn with_error_details_vec( - code: Code, - message: impl Into, - details: impl IntoIterator, - ) -> tonic::Status; - - /// Can be used to check if the error details contained in `tonic::Status` - /// are malformed or not. Tries to get an [`ErrorDetails`] struct from a - /// `tonic::Status`. If some `prost::DecodeError` occurs, it will be - /// returned. If not debugging, consider using - /// [`StatusExt::get_error_details`] or - /// [`StatusExt::get_error_details_vec`]. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => match status.check_error_details() { - /// Ok(err_details) => { - /// // Handle extracted details - /// } - /// Err(decode_error) => { - /// // Handle decode_error - /// } - /// } - /// }; - /// } - /// ``` - fn check_error_details(&self) -> Result; - - /// Get an [`ErrorDetails`] struct from `tonic::Status`. If some - /// `prost::DecodeError` occurs, an empty [`ErrorDetails`] struct will be - /// returned. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// let err_details = status.get_error_details(); - /// if let Some(bad_request) = err_details.bad_request() { - /// // Handle bad_request details - /// } - /// // ... - /// } - /// }; - /// } - /// ``` - fn get_error_details(&self) -> ErrorDetails; - - /// Can be used to check if the error details contained in `tonic::Status` - /// are malformed or not. Tries to get a vector of [`ErrorDetail`] enums - /// from a `tonic::Status`. If some `prost::DecodeError` occurs, it will be - /// returned. If not debugging, consider using - /// [`StatusExt::get_error_details_vec`] or - /// [`StatusExt::get_error_details`]. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => match status.check_error_details_vec() { - /// Ok(err_details) => { - /// // Handle extracted details - /// } - /// Err(decode_error) => { - /// // Handle decode_error - /// } - /// } - /// }; - /// } - /// ``` - fn check_error_details_vec(&self) -> Result, DecodeError>; - - /// Get a vector of [`ErrorDetail`] enums from `tonic::Status`. If some - /// `prost::DecodeError` occurs, an empty vector will be returned. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::{ErrorDetail, StatusExt}; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// let err_details = status.get_error_details_vec(); - /// for err_detail in err_details.iter() { - /// match err_detail { - /// ErrorDetail::BadRequest(bad_request) => { - /// // Handle bad_request details - /// } - /// // ... - /// _ => {} - /// } - /// } - /// } - /// }; - /// } - /// ``` - fn get_error_details_vec(&self) -> Vec; - - /// Get first [`RetryInfo`] details found on `tonic::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(retry_info) = status.get_details_retry_info() { - /// // Handle retry_info details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_retry_info(&self) -> Option; - - /// Get first [`DebugInfo`] details found on `tonic::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(debug_info) = status.get_details_debug_info() { - /// // Handle debug_info details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_debug_info(&self) -> Option; - - /// Get first [`QuotaFailure`] details found on `tonic::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(quota_failure) = status.get_details_quota_failure() { - /// // Handle quota_failure details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_quota_failure(&self) -> Option; - - /// Get first [`ErrorInfo`] details found on `tonic::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(error_info) = status.get_details_error_info() { - /// // Handle error_info details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_error_info(&self) -> Option; - - /// Get first [`PreconditionFailure`] details found on `tonic::Status`, - /// if any. If some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(precondition_failure) = status.get_details_precondition_failure() { - /// // Handle precondition_failure details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_precondition_failure(&self) -> Option; - - /// Get first [`BadRequest`] details found on `tonic::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(bad_request) = status.get_details_bad_request() { - /// // Handle bad_request details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_bad_request(&self) -> Option; - - /// Get first [`RequestInfo`] details found on `tonic::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(request_info) = status.get_details_request_info() { - /// // Handle request_info details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_request_info(&self) -> Option; - - /// Get first [`ResourceInfo`] details found on `tonic::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(resource_info) = status.get_details_resource_info() { - /// // Handle resource_info details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_resource_info(&self) -> Option; - - /// Get first [`Help`] details found on `tonic::Status`, if any. If some - /// `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(help) = status.get_details_help() { - /// // Handle help details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_help(&self) -> Option; - - /// Get first [`LocalizedMessage`] details found on `tonic::Status`, if - /// any. If some `prost::DecodeError` occurs, returns `None`. - /// - /// # Examples - /// - /// ``` - /// use tonic::{Status, Response}; - /// use tonic_types::StatusExt; - /// - /// fn handle_request_result(req_result: Result, Status>) { - /// match req_result { - /// Ok(_) => {}, - /// Err(status) => { - /// if let Some(localized_message) = status.get_details_localized_message() { - /// // Handle localized_message details - /// } - /// } - /// }; - /// } - /// ``` - fn get_details_localized_message(&self) -> Option; -} - -impl crate::sealed::Sealed for tonic::Status {} - -impl StatusExt for tonic::Status { - fn with_error_details_and_metadata( - code: Code, - message: impl Into, - details: ErrorDetails, - metadata: MetadataMap, - ) -> Self { - let message: String = message.into(); - - let mut conv_details: Vec = Vec::with_capacity(10); - - if let Some(retry_info) = details.retry_info { - conv_details.push(retry_info.into_any()); - } - - if let Some(debug_info) = details.debug_info { - conv_details.push(debug_info.into_any()); - } - - if let Some(quota_failure) = details.quota_failure { - conv_details.push(quota_failure.into_any()); - } - - if let Some(error_info) = details.error_info { - conv_details.push(error_info.into_any()); - } - - if let Some(precondition_failure) = details.precondition_failure { - conv_details.push(precondition_failure.into_any()); - } - - if let Some(bad_request) = details.bad_request { - conv_details.push(bad_request.into_any()); - } - - if let Some(request_info) = details.request_info { - conv_details.push(request_info.into_any()); - } - - if let Some(resource_info) = details.resource_info { - conv_details.push(resource_info.into_any()); - } - - if let Some(help) = details.help { - conv_details.push(help.into_any()); - } - - if let Some(localized_message) = details.localized_message { - conv_details.push(localized_message.into_any()); - } - - let details = gen_details_bytes(code, &message, conv_details); - - tonic::Status::with_details_and_metadata(code, message, details, metadata) - } - - fn with_error_details(code: Code, message: impl Into, details: ErrorDetails) -> Self { - tonic::Status::with_error_details_and_metadata(code, message, details, MetadataMap::new()) - } - - fn with_error_details_vec_and_metadata( - code: Code, - message: impl Into, - details: impl IntoIterator, - metadata: MetadataMap, - ) -> Self { - let message: String = message.into(); - - let mut conv_details: Vec = Vec::new(); - - for error_detail in details.into_iter() { - match error_detail { - ErrorDetail::RetryInfo(retry_info) => { - conv_details.push(retry_info.into_any()); - } - ErrorDetail::DebugInfo(debug_info) => { - conv_details.push(debug_info.into_any()); - } - ErrorDetail::QuotaFailure(quota_failure) => { - conv_details.push(quota_failure.into_any()); - } - ErrorDetail::ErrorInfo(error_info) => { - conv_details.push(error_info.into_any()); - } - ErrorDetail::PreconditionFailure(prec_failure) => { - conv_details.push(prec_failure.into_any()); - } - ErrorDetail::BadRequest(bad_req) => { - conv_details.push(bad_req.into_any()); - } - ErrorDetail::RequestInfo(req_info) => { - conv_details.push(req_info.into_any()); - } - ErrorDetail::ResourceInfo(res_info) => { - conv_details.push(res_info.into_any()); - } - ErrorDetail::Help(help) => { - conv_details.push(help.into_any()); - } - ErrorDetail::LocalizedMessage(loc_message) => { - conv_details.push(loc_message.into_any()); - } - } - } - - let details = gen_details_bytes(code, &message, conv_details); - - tonic::Status::with_details_and_metadata(code, message, details, metadata) - } - - fn with_error_details_vec( - code: Code, - message: impl Into, - details: impl IntoIterator, - ) -> Self { - tonic::Status::with_error_details_vec_and_metadata( - code, - message, - details, - MetadataMap::new(), - ) - } - - fn check_error_details(&self) -> Result { - let status = pb::Status::decode(self.details())?; - - status.check_error_details() - } - - fn get_error_details(&self) -> ErrorDetails { - self.check_error_details().unwrap_or_default() - } - - fn check_error_details_vec(&self) -> Result, DecodeError> { - let status = pb::Status::decode(self.details())?; - - status.check_error_details_vec() - } - - fn get_error_details_vec(&self) -> Vec { - self.check_error_details_vec().unwrap_or_default() - } - - fn get_details_retry_info(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_retry_info() - } - - fn get_details_debug_info(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_debug_info() - } - - fn get_details_quota_failure(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_quota_failure() - } - - fn get_details_error_info(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_error_info() - } - - fn get_details_precondition_failure(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - status.get_details_precondition_failure() - } - - fn get_details_bad_request(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_bad_request() - } - - fn get_details_request_info(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_request_info() - } - - fn get_details_resource_info(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_resource_info() - } - - fn get_details_help(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_help() - } - - fn get_details_localized_message(&self) -> Option { - let status = pb::Status::decode(self.details()).ok()?; - - status.get_details_localized_message() - } -} - -impl crate::sealed::Sealed for pb::Status {} - -/// Used to implement associated functions and methods on `pb::Status`, that -/// allow the extraction of standard error details. This trait is -/// sealed and not meant to be implemented outside of `tonic-types`. -pub trait RpcStatusExt: crate::sealed::Sealed { - /// Can be used to check if the error details contained in `pb::Status` - /// are malformed or not. Tries to get an [`ErrorDetails`] struct from a - /// `pb::Status`. If some `prost::DecodeError` occurs, it will be - /// returned. If not debugging, consider using - /// [`RpcStatusExt::get_error_details`] or - /// [`RpcStatusExt::get_error_details_vec`]. - fn check_error_details(&self) -> Result; - - /// Get an [`ErrorDetails`] struct from `pb::Status`. If some - /// `prost::DecodeError` occurs, an empty [`ErrorDetails`] struct will be - /// returned. - fn get_error_details(&self) -> ErrorDetails; - - /// Can be used to check if the error details contained in `pb::Status` - /// are malformed or not. Tries to get a vector of [`ErrorDetail`] enums - /// from a `pb::Status`. If some `prost::DecodeError` occurs, it will be - /// returned. If not debugging, consider using - /// [`StatusExt::get_error_details_vec`] or - /// [`StatusExt::get_error_details`]. - fn check_error_details_vec(&self) -> Result, DecodeError>; - - /// Get a vector of [`ErrorDetail`] enums from `pb::Status`. If some - /// `prost::DecodeError` occurs, an empty vector will be returned. - fn get_error_details_vec(&self) -> Vec; - - /// Get first [`RetryInfo`] details found on `pb::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - fn get_details_retry_info(&self) -> Option; - - /// Get first [`DebugInfo`] details found on `pb::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - fn get_details_debug_info(&self) -> Option; - - /// Get first [`QuotaFailure`] details found on `pb::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - fn get_details_quota_failure(&self) -> Option; - - /// Get first [`ErrorInfo`] details found on `pb::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - fn get_details_error_info(&self) -> Option; - - /// Get first [`PreconditionFailure`] details found on `pb::Status`, - /// if any. If some `prost::DecodeError` occurs, returns `None`. - fn get_details_precondition_failure(&self) -> Option; - - /// Get first [`BadRequest`] details found on `pb::Status`, if any. If - /// some `prost::DecodeError` occurs, returns `None`. - fn get_details_bad_request(&self) -> Option; - - /// Get first [`RequestInfo`] details found on `pb::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - fn get_details_request_info(&self) -> Option; - - /// Get first [`ResourceInfo`] details found on `pb::Status`, if any. - /// If some `prost::DecodeError` occurs, returns `None`. - fn get_details_resource_info(&self) -> Option; - - /// Get first [`Help`] details found on `pb::Status`, if any. If some - /// `prost::DecodeError` occurs, returns `None`. - fn get_details_help(&self) -> Option; - - /// Get first [`LocalizedMessage`] details found on `pb::Status`, if - /// any. If some `prost::DecodeError` occurs, returns `None`. - fn get_details_localized_message(&self) -> Option; -} - -impl RpcStatusExt for pb::Status { - fn check_error_details(&self) -> Result { - let mut details = ErrorDetails::new(); - - for any in self.details.iter() { - match any.type_url.as_str() { - RetryInfo::TYPE_URL => { - details.retry_info = Some(RetryInfo::from_any_ref(any)?); - } - DebugInfo::TYPE_URL => { - details.debug_info = Some(DebugInfo::from_any_ref(any)?); - } - QuotaFailure::TYPE_URL => { - details.quota_failure = Some(QuotaFailure::from_any_ref(any)?); - } - ErrorInfo::TYPE_URL => { - details.error_info = Some(ErrorInfo::from_any_ref(any)?); - } - PreconditionFailure::TYPE_URL => { - details.precondition_failure = Some(PreconditionFailure::from_any_ref(any)?); - } - BadRequest::TYPE_URL => { - details.bad_request = Some(BadRequest::from_any_ref(any)?); - } - RequestInfo::TYPE_URL => { - details.request_info = Some(RequestInfo::from_any_ref(any)?); - } - ResourceInfo::TYPE_URL => { - details.resource_info = Some(ResourceInfo::from_any_ref(any)?); - } - Help::TYPE_URL => { - details.help = Some(Help::from_any_ref(any)?); - } - LocalizedMessage::TYPE_URL => { - details.localized_message = Some(LocalizedMessage::from_any_ref(any)?); - } - _ => {} - } - } - - Ok(details) - } - - fn get_error_details(&self) -> ErrorDetails { - self.check_error_details().unwrap_or_default() - } - - fn check_error_details_vec(&self) -> Result, DecodeError> { - let mut details: Vec = Vec::with_capacity(self.details.len()); - - for any in self.details.iter() { - match any.type_url.as_str() { - RetryInfo::TYPE_URL => { - details.push(RetryInfo::from_any_ref(any)?.into()); - } - DebugInfo::TYPE_URL => { - details.push(DebugInfo::from_any_ref(any)?.into()); - } - QuotaFailure::TYPE_URL => { - details.push(QuotaFailure::from_any_ref(any)?.into()); - } - ErrorInfo::TYPE_URL => { - details.push(ErrorInfo::from_any_ref(any)?.into()); - } - PreconditionFailure::TYPE_URL => { - details.push(PreconditionFailure::from_any_ref(any)?.into()); - } - BadRequest::TYPE_URL => { - details.push(BadRequest::from_any_ref(any)?.into()); - } - RequestInfo::TYPE_URL => { - details.push(RequestInfo::from_any_ref(any)?.into()); - } - ResourceInfo::TYPE_URL => { - details.push(ResourceInfo::from_any_ref(any)?.into()); - } - Help::TYPE_URL => { - details.push(Help::from_any_ref(any)?.into()); - } - LocalizedMessage::TYPE_URL => { - details.push(LocalizedMessage::from_any_ref(any)?.into()); - } - _ => {} - } - } - - Ok(details) - } - - fn get_error_details_vec(&self) -> Vec { - self.check_error_details_vec().unwrap_or_default() - } - - fn get_details_retry_info(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == RetryInfo::TYPE_URL { - if let Ok(detail) = RetryInfo::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_debug_info(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == DebugInfo::TYPE_URL { - if let Ok(detail) = DebugInfo::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_quota_failure(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == QuotaFailure::TYPE_URL { - if let Ok(detail) = QuotaFailure::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_error_info(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == ErrorInfo::TYPE_URL { - if let Ok(detail) = ErrorInfo::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_precondition_failure(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == PreconditionFailure::TYPE_URL { - if let Ok(detail) = PreconditionFailure::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_bad_request(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == BadRequest::TYPE_URL { - if let Ok(detail) = BadRequest::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_request_info(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == RequestInfo::TYPE_URL { - if let Ok(detail) = RequestInfo::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_resource_info(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == ResourceInfo::TYPE_URL { - if let Ok(detail) = ResourceInfo::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_help(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == Help::TYPE_URL { - if let Ok(detail) = Help::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } - - fn get_details_localized_message(&self) -> Option { - for any in self.details.iter() { - if any.type_url.as_str() == LocalizedMessage::TYPE_URL { - if let Ok(detail) = LocalizedMessage::from_any_ref(any) { - return Some(detail); - } - } - } - - None - } -} - -#[cfg(test)] -mod tests { - use std::{collections::HashMap, time::Duration}; - use tonic::{Code, Status}; - - use super::{ - BadRequest, DebugInfo, ErrorDetails, ErrorInfo, Help, LocalizedMessage, - PreconditionFailure, QuotaFailure, RequestInfo, ResourceInfo, RetryInfo, StatusExt, - }; - - #[test] - fn gen_status_with_details() { - let mut metadata = HashMap::new(); - metadata.insert("limitPerRequest".into(), "100".into()); - - let mut err_details = ErrorDetails::new(); - - err_details - .set_retry_info(Some(Duration::from_secs(5))) - .set_debug_info( - vec!["trace3".into(), "trace2".into(), "trace1".into()], - "details", - ) - .add_quota_failure_violation("clientip:", "description") - .set_error_info("SOME_INFO", "example.local", metadata.clone()) - .add_precondition_failure_violation("TOS", "example.local", "description") - .add_bad_request_violation("field", "description") - .set_request_info("request-id", "some-request-data") - .set_resource_info("resource-type", "resource-name", "owner", "description") - .add_help_link("link to resource", "resource.example.local") - .set_localized_message("en-US", "message for the user"); - - let fmt_details = format!("{:?}", err_details); - - let err_details_vec = vec![ - RetryInfo::new(Some(Duration::from_secs(5))).into(), - DebugInfo::new( - vec!["trace3".into(), "trace2".into(), "trace1".into()], - "details", - ) - .into(), - QuotaFailure::with_violation("clientip:", "description").into(), - ErrorInfo::new("SOME_INFO", "example.local", metadata).into(), - PreconditionFailure::with_violation("TOS", "example.local", "description").into(), - BadRequest::with_violation("field", "description").into(), - RequestInfo::new("request-id", "some-request-data").into(), - ResourceInfo::new("resource-type", "resource-name", "owner", "description").into(), - Help::with_link("link to resource", "resource.example.local").into(), - LocalizedMessage::new("en-US", "message for the user").into(), - ]; - - let fmt_details_vec = format!("{:?}", err_details_vec); - - let status_from_struct = Status::with_error_details( - Code::InvalidArgument, - "error with bad request details", - err_details, - ); - - let status_from_vec = Status::with_error_details_vec( - Code::InvalidArgument, - "error with bad request details", - err_details_vec, - ); - - let ext_details = match status_from_vec.check_error_details() { - Ok(ext_details) => ext_details, - Err(err) => panic!( - "Error extracting details struct from status_from_vec: {:?}", - err - ), - }; - - let fmt_ext_details = format!("{:?}", ext_details); +mod std_messages; +pub use std_messages::*; - assert!( - fmt_ext_details.eq(&fmt_details), - "Extracted details struct differs from original details struct" - ); +mod status_ext; +pub use status_ext::StatusExt; - let ext_details_vec = match status_from_struct.check_error_details_vec() { - Ok(ext_details) => ext_details, - Err(err) => panic!( - "Error extracting details_vec from status_from_struct: {:?}", - err - ), - }; +mod rpc_status_ext; +pub use rpc_status_ext::RpcStatusExt; - let fmt_ext_details_vec = format!("{:?}", ext_details_vec); +mod code_ext; +pub use code_ext::CodeExt; - assert!( - fmt_ext_details_vec.eq(&fmt_details_vec), - "Extracted details vec differs from original details vec" - ); - } -} +mod helpers; +use helpers::{gen_details_bytes, FromAnyRef, IntoAny}; diff --git a/tonic-types/src/richer_error/rpc_status_ext.rs b/tonic-types/src/richer_error/rpc_status_ext.rs new file mode 100644 index 000000000..444350873 --- /dev/null +++ b/tonic-types/src/richer_error/rpc_status_ext.rs @@ -0,0 +1,291 @@ +use prost::DecodeError; + +use crate::pb; + +use super::*; + +/// Used to implement associated functions and methods on `pb::Status`, that +/// allow the extraction of standard error details. This trait is +/// sealed and not meant to be implemented outside of `tonic-types`. +pub trait RpcStatusExt: crate::sealed::Sealed { + /// Can be used to check if the error details contained in `pb::Status` + /// are malformed or not. Tries to get an [`ErrorDetails`] struct from a + /// `pb::Status`. If some `prost::DecodeError` occurs, it will be + /// returned. If not debugging, consider using + /// [`RpcStatusExt::get_error_details`] or + /// [`RpcStatusExt::get_error_details_vec`]. + fn check_error_details(&self) -> Result; + + /// Get an [`ErrorDetails`] struct from `pb::Status`. If some + /// `prost::DecodeError` occurs, an empty [`ErrorDetails`] struct will be + /// returned. + fn get_error_details(&self) -> ErrorDetails; + + /// Can be used to check if the error details contained in `pb::Status` + /// are malformed or not. Tries to get a vector of [`ErrorDetail`] enums + /// from a `pb::Status`. If some `prost::DecodeError` occurs, it will be + /// returned. If not debugging, consider using + /// [`StatusExt::get_error_details_vec`] or + /// [`StatusExt::get_error_details`]. + fn check_error_details_vec(&self) -> Result, DecodeError>; + + /// Get a vector of [`ErrorDetail`] enums from `pb::Status`. If some + /// `prost::DecodeError` occurs, an empty vector will be returned. + fn get_error_details_vec(&self) -> Vec; + + /// Get first [`RetryInfo`] details found on `pb::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + fn get_details_retry_info(&self) -> Option; + + /// Get first [`DebugInfo`] details found on `pb::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + fn get_details_debug_info(&self) -> Option; + + /// Get first [`QuotaFailure`] details found on `pb::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + fn get_details_quota_failure(&self) -> Option; + + /// Get first [`ErrorInfo`] details found on `pb::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + fn get_details_error_info(&self) -> Option; + + /// Get first [`PreconditionFailure`] details found on `pb::Status`, + /// if any. If some `prost::DecodeError` occurs, returns `None`. + fn get_details_precondition_failure(&self) -> Option; + + /// Get first [`BadRequest`] details found on `pb::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + fn get_details_bad_request(&self) -> Option; + + /// Get first [`RequestInfo`] details found on `pb::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + fn get_details_request_info(&self) -> Option; + + /// Get first [`ResourceInfo`] details found on `pb::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + fn get_details_resource_info(&self) -> Option; + + /// Get first [`Help`] details found on `pb::Status`, if any. If some + /// `prost::DecodeError` occurs, returns `None`. + fn get_details_help(&self) -> Option; + + /// Get first [`LocalizedMessage`] details found on `pb::Status`, if + /// any. If some `prost::DecodeError` occurs, returns `None`. + fn get_details_localized_message(&self) -> Option; +} + +impl RpcStatusExt for pb::Status { + fn check_error_details(&self) -> Result { + let mut details = ErrorDetails::new(); + + for any in self.details.iter() { + match any.type_url.as_str() { + RetryInfo::TYPE_URL => { + details.retry_info = Some(RetryInfo::from_any_ref(any)?); + } + DebugInfo::TYPE_URL => { + details.debug_info = Some(DebugInfo::from_any_ref(any)?); + } + QuotaFailure::TYPE_URL => { + details.quota_failure = Some(QuotaFailure::from_any_ref(any)?); + } + ErrorInfo::TYPE_URL => { + details.error_info = Some(ErrorInfo::from_any_ref(any)?); + } + PreconditionFailure::TYPE_URL => { + details.precondition_failure = Some(PreconditionFailure::from_any_ref(any)?); + } + BadRequest::TYPE_URL => { + details.bad_request = Some(BadRequest::from_any_ref(any)?); + } + RequestInfo::TYPE_URL => { + details.request_info = Some(RequestInfo::from_any_ref(any)?); + } + ResourceInfo::TYPE_URL => { + details.resource_info = Some(ResourceInfo::from_any_ref(any)?); + } + Help::TYPE_URL => { + details.help = Some(Help::from_any_ref(any)?); + } + LocalizedMessage::TYPE_URL => { + details.localized_message = Some(LocalizedMessage::from_any_ref(any)?); + } + _ => {} + } + } + + Ok(details) + } + + fn get_error_details(&self) -> ErrorDetails { + self.check_error_details().unwrap_or_default() + } + + fn check_error_details_vec(&self) -> Result, DecodeError> { + let mut details: Vec = Vec::with_capacity(self.details.len()); + + for any in self.details.iter() { + match any.type_url.as_str() { + RetryInfo::TYPE_URL => { + details.push(RetryInfo::from_any_ref(any)?.into()); + } + DebugInfo::TYPE_URL => { + details.push(DebugInfo::from_any_ref(any)?.into()); + } + QuotaFailure::TYPE_URL => { + details.push(QuotaFailure::from_any_ref(any)?.into()); + } + ErrorInfo::TYPE_URL => { + details.push(ErrorInfo::from_any_ref(any)?.into()); + } + PreconditionFailure::TYPE_URL => { + details.push(PreconditionFailure::from_any_ref(any)?.into()); + } + BadRequest::TYPE_URL => { + details.push(BadRequest::from_any_ref(any)?.into()); + } + RequestInfo::TYPE_URL => { + details.push(RequestInfo::from_any_ref(any)?.into()); + } + ResourceInfo::TYPE_URL => { + details.push(ResourceInfo::from_any_ref(any)?.into()); + } + Help::TYPE_URL => { + details.push(Help::from_any_ref(any)?.into()); + } + LocalizedMessage::TYPE_URL => { + details.push(LocalizedMessage::from_any_ref(any)?.into()); + } + _ => {} + } + } + + Ok(details) + } + + fn get_error_details_vec(&self) -> Vec { + self.check_error_details_vec().unwrap_or_default() + } + + fn get_details_retry_info(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == RetryInfo::TYPE_URL { + if let Ok(detail) = RetryInfo::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_debug_info(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == DebugInfo::TYPE_URL { + if let Ok(detail) = DebugInfo::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_quota_failure(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == QuotaFailure::TYPE_URL { + if let Ok(detail) = QuotaFailure::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_error_info(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == ErrorInfo::TYPE_URL { + if let Ok(detail) = ErrorInfo::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_precondition_failure(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == PreconditionFailure::TYPE_URL { + if let Ok(detail) = PreconditionFailure::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_bad_request(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == BadRequest::TYPE_URL { + if let Ok(detail) = BadRequest::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_request_info(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == RequestInfo::TYPE_URL { + if let Ok(detail) = RequestInfo::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_resource_info(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == ResourceInfo::TYPE_URL { + if let Ok(detail) = ResourceInfo::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_help(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == Help::TYPE_URL { + if let Ok(detail) = Help::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } + + fn get_details_localized_message(&self) -> Option { + for any in self.details.iter() { + if any.type_url.as_str() == LocalizedMessage::TYPE_URL { + if let Ok(detail) = LocalizedMessage::from_any_ref(any) { + return Some(detail); + } + } + } + + None + } +} + +impl crate::sealed::Sealed for pb::Status {} diff --git a/tonic-types/src/richer_error/status_ext.rs b/tonic-types/src/richer_error/status_ext.rs new file mode 100644 index 000000000..a134e21ac --- /dev/null +++ b/tonic-types/src/richer_error/status_ext.rs @@ -0,0 +1,739 @@ +use prost::{DecodeError, Message}; +use prost_types::Any; +use tonic::{metadata::MetadataMap, Code}; + +use crate::pb; + +use super::*; + +/// Used to implement associated functions and methods on `tonic::Status`, that +/// allow the addition and extraction of standard error details. This trait is +/// sealed and not meant to be implemented outside of `tonic-types`. +pub trait StatusExt: crate::sealed::Sealed { + /// Generates a `tonic::Status` with error details obtained from an + /// [`ErrorDetails`] struct, and custom metadata. + /// + /// # Examples + /// + /// ``` + /// use tonic::{metadata::MetadataMap, Code, Status}; + /// use tonic_types::{ErrorDetails, StatusExt}; + /// + /// let status = Status::with_error_details_and_metadata( + /// Code::InvalidArgument, + /// "bad request", + /// ErrorDetails::with_bad_request_violation("field", "description"), + /// MetadataMap::new() + /// ); + /// ``` + fn with_error_details_and_metadata( + code: Code, + message: impl Into, + details: ErrorDetails, + metadata: MetadataMap, + ) -> tonic::Status; + + /// Generates a `tonic::Status` with error details obtained from an + /// [`ErrorDetails`] struct. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Code, Status}; + /// use tonic_types::{ErrorDetails, StatusExt}; + /// + /// let status = Status::with_error_details( + /// Code::InvalidArgument, + /// "bad request", + /// ErrorDetails::with_bad_request_violation("field", "description"), + /// ); + /// ``` + fn with_error_details( + code: Code, + message: impl Into, + details: ErrorDetails, + ) -> tonic::Status; + + /// Generates a `tonic::Status` with error details provided in a vector of + /// [`ErrorDetail`] enums, and custom metadata. + /// + /// # Examples + /// + /// ``` + /// use tonic::{metadata::MetadataMap, Code, Status}; + /// use tonic_types::{BadRequest, StatusExt}; + /// + /// let status = Status::with_error_details_vec_and_metadata( + /// Code::InvalidArgument, + /// "bad request", + /// vec![ + /// BadRequest::with_violation("field", "description").into(), + /// ], + /// MetadataMap::new() + /// ); + /// ``` + fn with_error_details_vec_and_metadata( + code: Code, + message: impl Into, + details: impl IntoIterator, + metadata: MetadataMap, + ) -> tonic::Status; + + /// Generates a `tonic::Status` with error details provided in a vector of + /// [`ErrorDetail`] enums. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Code, Status}; + /// use tonic_types::{BadRequest, StatusExt}; + /// + /// let status = Status::with_error_details_vec( + /// Code::InvalidArgument, + /// "bad request", + /// vec![ + /// BadRequest::with_violation("field", "description").into(), + /// ] + /// ); + /// ``` + fn with_error_details_vec( + code: Code, + message: impl Into, + details: impl IntoIterator, + ) -> tonic::Status; + + /// Can be used to check if the error details contained in `tonic::Status` + /// are malformed or not. Tries to get an [`ErrorDetails`] struct from a + /// `tonic::Status`. If some `prost::DecodeError` occurs, it will be + /// returned. If not debugging, consider using + /// [`StatusExt::get_error_details`] or + /// [`StatusExt::get_error_details_vec`]. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => match status.check_error_details() { + /// Ok(err_details) => { + /// // Handle extracted details + /// } + /// Err(decode_error) => { + /// // Handle decode_error + /// } + /// } + /// }; + /// } + /// ``` + fn check_error_details(&self) -> Result; + + /// Get an [`ErrorDetails`] struct from `tonic::Status`. If some + /// `prost::DecodeError` occurs, an empty [`ErrorDetails`] struct will be + /// returned. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// let err_details = status.get_error_details(); + /// if let Some(bad_request) = err_details.bad_request() { + /// // Handle bad_request details + /// } + /// // ... + /// } + /// }; + /// } + /// ``` + fn get_error_details(&self) -> ErrorDetails; + + /// Can be used to check if the error details contained in `tonic::Status` + /// are malformed or not. Tries to get a vector of [`ErrorDetail`] enums + /// from a `tonic::Status`. If some `prost::DecodeError` occurs, it will be + /// returned. If not debugging, consider using + /// [`StatusExt::get_error_details_vec`] or + /// [`StatusExt::get_error_details`]. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => match status.check_error_details_vec() { + /// Ok(err_details) => { + /// // Handle extracted details + /// } + /// Err(decode_error) => { + /// // Handle decode_error + /// } + /// } + /// }; + /// } + /// ``` + fn check_error_details_vec(&self) -> Result, DecodeError>; + + /// Get a vector of [`ErrorDetail`] enums from `tonic::Status`. If some + /// `prost::DecodeError` occurs, an empty vector will be returned. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::{ErrorDetail, StatusExt}; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// let err_details = status.get_error_details_vec(); + /// for err_detail in err_details.iter() { + /// match err_detail { + /// ErrorDetail::BadRequest(bad_request) => { + /// // Handle bad_request details + /// } + /// // ... + /// _ => {} + /// } + /// } + /// } + /// }; + /// } + /// ``` + fn get_error_details_vec(&self) -> Vec; + + /// Get first [`RetryInfo`] details found on `tonic::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(retry_info) = status.get_details_retry_info() { + /// // Handle retry_info details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_retry_info(&self) -> Option; + + /// Get first [`DebugInfo`] details found on `tonic::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(debug_info) = status.get_details_debug_info() { + /// // Handle debug_info details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_debug_info(&self) -> Option; + + /// Get first [`QuotaFailure`] details found on `tonic::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(quota_failure) = status.get_details_quota_failure() { + /// // Handle quota_failure details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_quota_failure(&self) -> Option; + + /// Get first [`ErrorInfo`] details found on `tonic::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(error_info) = status.get_details_error_info() { + /// // Handle error_info details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_error_info(&self) -> Option; + + /// Get first [`PreconditionFailure`] details found on `tonic::Status`, + /// if any. If some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(precondition_failure) = status.get_details_precondition_failure() { + /// // Handle precondition_failure details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_precondition_failure(&self) -> Option; + + /// Get first [`BadRequest`] details found on `tonic::Status`, if any. If + /// some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(bad_request) = status.get_details_bad_request() { + /// // Handle bad_request details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_bad_request(&self) -> Option; + + /// Get first [`RequestInfo`] details found on `tonic::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(request_info) = status.get_details_request_info() { + /// // Handle request_info details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_request_info(&self) -> Option; + + /// Get first [`ResourceInfo`] details found on `tonic::Status`, if any. + /// If some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(resource_info) = status.get_details_resource_info() { + /// // Handle resource_info details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_resource_info(&self) -> Option; + + /// Get first [`Help`] details found on `tonic::Status`, if any. If some + /// `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(help) = status.get_details_help() { + /// // Handle help details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_help(&self) -> Option; + + /// Get first [`LocalizedMessage`] details found on `tonic::Status`, if + /// any. If some `prost::DecodeError` occurs, returns `None`. + /// + /// # Examples + /// + /// ``` + /// use tonic::{Status, Response}; + /// use tonic_types::StatusExt; + /// + /// fn handle_request_result(req_result: Result, Status>) { + /// match req_result { + /// Ok(_) => {}, + /// Err(status) => { + /// if let Some(localized_message) = status.get_details_localized_message() { + /// // Handle localized_message details + /// } + /// } + /// }; + /// } + /// ``` + fn get_details_localized_message(&self) -> Option; +} + +impl crate::sealed::Sealed for tonic::Status {} + +impl StatusExt for tonic::Status { + fn with_error_details_and_metadata( + code: Code, + message: impl Into, + details: ErrorDetails, + metadata: MetadataMap, + ) -> Self { + let message: String = message.into(); + + let mut conv_details: Vec = Vec::with_capacity(10); + + if let Some(retry_info) = details.retry_info { + conv_details.push(retry_info.into_any()); + } + + if let Some(debug_info) = details.debug_info { + conv_details.push(debug_info.into_any()); + } + + if let Some(quota_failure) = details.quota_failure { + conv_details.push(quota_failure.into_any()); + } + + if let Some(error_info) = details.error_info { + conv_details.push(error_info.into_any()); + } + + if let Some(precondition_failure) = details.precondition_failure { + conv_details.push(precondition_failure.into_any()); + } + + if let Some(bad_request) = details.bad_request { + conv_details.push(bad_request.into_any()); + } + + if let Some(request_info) = details.request_info { + conv_details.push(request_info.into_any()); + } + + if let Some(resource_info) = details.resource_info { + conv_details.push(resource_info.into_any()); + } + + if let Some(help) = details.help { + conv_details.push(help.into_any()); + } + + if let Some(localized_message) = details.localized_message { + conv_details.push(localized_message.into_any()); + } + + let details = gen_details_bytes(code, &message, conv_details); + + tonic::Status::with_details_and_metadata(code, message, details, metadata) + } + + fn with_error_details(code: Code, message: impl Into, details: ErrorDetails) -> Self { + tonic::Status::with_error_details_and_metadata(code, message, details, MetadataMap::new()) + } + + fn with_error_details_vec_and_metadata( + code: Code, + message: impl Into, + details: impl IntoIterator, + metadata: MetadataMap, + ) -> Self { + let message: String = message.into(); + + let mut conv_details: Vec = Vec::new(); + + for error_detail in details.into_iter() { + match error_detail { + ErrorDetail::RetryInfo(retry_info) => { + conv_details.push(retry_info.into_any()); + } + ErrorDetail::DebugInfo(debug_info) => { + conv_details.push(debug_info.into_any()); + } + ErrorDetail::QuotaFailure(quota_failure) => { + conv_details.push(quota_failure.into_any()); + } + ErrorDetail::ErrorInfo(error_info) => { + conv_details.push(error_info.into_any()); + } + ErrorDetail::PreconditionFailure(prec_failure) => { + conv_details.push(prec_failure.into_any()); + } + ErrorDetail::BadRequest(bad_req) => { + conv_details.push(bad_req.into_any()); + } + ErrorDetail::RequestInfo(req_info) => { + conv_details.push(req_info.into_any()); + } + ErrorDetail::ResourceInfo(res_info) => { + conv_details.push(res_info.into_any()); + } + ErrorDetail::Help(help) => { + conv_details.push(help.into_any()); + } + ErrorDetail::LocalizedMessage(loc_message) => { + conv_details.push(loc_message.into_any()); + } + } + } + + let details = gen_details_bytes(code, &message, conv_details); + + tonic::Status::with_details_and_metadata(code, message, details, metadata) + } + + fn with_error_details_vec( + code: Code, + message: impl Into, + details: impl IntoIterator, + ) -> Self { + tonic::Status::with_error_details_vec_and_metadata( + code, + message, + details, + MetadataMap::new(), + ) + } + + fn check_error_details(&self) -> Result { + let status = pb::Status::decode(self.details())?; + + status.check_error_details() + } + + fn get_error_details(&self) -> ErrorDetails { + self.check_error_details().unwrap_or_default() + } + + fn check_error_details_vec(&self) -> Result, DecodeError> { + let status = pb::Status::decode(self.details())?; + + status.check_error_details_vec() + } + + fn get_error_details_vec(&self) -> Vec { + self.check_error_details_vec().unwrap_or_default() + } + + fn get_details_retry_info(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_retry_info() + } + + fn get_details_debug_info(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_debug_info() + } + + fn get_details_quota_failure(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_quota_failure() + } + + fn get_details_error_info(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_error_info() + } + + fn get_details_precondition_failure(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_precondition_failure() + } + + fn get_details_bad_request(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_bad_request() + } + + fn get_details_request_info(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_request_info() + } + + fn get_details_resource_info(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_resource_info() + } + + fn get_details_help(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_help() + } + + fn get_details_localized_message(&self) -> Option { + let status = pb::Status::decode(self.details()).ok()?; + + status.get_details_localized_message() + } +} + +#[cfg(test)] +mod tests { + use std::{collections::HashMap, time::Duration}; + use tonic::{Code, Status}; + + use super::{ + BadRequest, DebugInfo, ErrorDetails, ErrorInfo, Help, LocalizedMessage, + PreconditionFailure, QuotaFailure, RequestInfo, ResourceInfo, RetryInfo, StatusExt, + }; + + #[test] + fn gen_status_with_details() { + let mut metadata = HashMap::new(); + metadata.insert("limitPerRequest".into(), "100".into()); + + let mut err_details = ErrorDetails::new(); + + err_details + .set_retry_info(Some(Duration::from_secs(5))) + .set_debug_info( + vec!["trace3".into(), "trace2".into(), "trace1".into()], + "details", + ) + .add_quota_failure_violation("clientip:", "description") + .set_error_info("SOME_INFO", "example.local", metadata.clone()) + .add_precondition_failure_violation("TOS", "example.local", "description") + .add_bad_request_violation("field", "description") + .set_request_info("request-id", "some-request-data") + .set_resource_info("resource-type", "resource-name", "owner", "description") + .add_help_link("link to resource", "resource.example.local") + .set_localized_message("en-US", "message for the user"); + + let fmt_details = format!("{:?}", err_details); + + let err_details_vec = vec![ + RetryInfo::new(Some(Duration::from_secs(5))).into(), + DebugInfo::new( + vec!["trace3".into(), "trace2".into(), "trace1".into()], + "details", + ) + .into(), + QuotaFailure::with_violation("clientip:", "description").into(), + ErrorInfo::new("SOME_INFO", "example.local", metadata).into(), + PreconditionFailure::with_violation("TOS", "example.local", "description").into(), + BadRequest::with_violation("field", "description").into(), + RequestInfo::new("request-id", "some-request-data").into(), + ResourceInfo::new("resource-type", "resource-name", "owner", "description").into(), + Help::with_link("link to resource", "resource.example.local").into(), + LocalizedMessage::new("en-US", "message for the user").into(), + ]; + + let fmt_details_vec = format!("{:?}", err_details_vec); + + let status_from_struct = Status::with_error_details( + Code::InvalidArgument, + "error with bad request details", + err_details, + ); + + let status_from_vec = Status::with_error_details_vec( + Code::InvalidArgument, + "error with bad request details", + err_details_vec, + ); + + let ext_details = match status_from_vec.check_error_details() { + Ok(ext_details) => ext_details, + Err(err) => panic!( + "Error extracting details struct from status_from_vec: {:?}", + err + ), + }; + + let fmt_ext_details = format!("{:?}", ext_details); + + assert!( + fmt_ext_details.eq(&fmt_details), + "Extracted details struct differs from original details struct" + ); + + let ext_details_vec = match status_from_struct.check_error_details_vec() { + Ok(ext_details) => ext_details, + Err(err) => panic!( + "Error extracting details_vec from status_from_struct: {:?}", + err + ), + }; + + let fmt_ext_details_vec = format!("{:?}", ext_details_vec); + + assert!( + fmt_ext_details_vec.eq(&fmt_details_vec), + "Extracted details vec differs from original details vec" + ); + } +} diff --git a/tonic-types/src/richer_error/std_messages/bad_request.rs b/tonic-types/src/richer_error/std_messages/bad_request.rs index c9fcb984a..042b07963 100644 --- a/tonic-types/src/richer_error/std_messages/bad_request.rs +++ b/tonic-types/src/richer_error/std_messages/bad_request.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used at the `field_violations` field of the [`BadRequest`] struct. /// Describes a single bad request field. @@ -110,13 +110,6 @@ impl IntoAny for BadRequest { } } -impl FromAny for BadRequest { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for BadRequest { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -144,8 +137,7 @@ impl From for pb::BadRequest { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::BadRequest; + use super::*; #[test] fn gen_bad_request() { @@ -192,7 +184,7 @@ mod tests { "Any from filled BadRequest differs from expected result" ); - let br_details = match BadRequest::from_any(gen_any) { + let br_details = match BadRequest::from_any_ref(&gen_any) { Err(error) => panic!("Error generating BadRequest from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/debug_info.rs b/tonic-types/src/richer_error/std_messages/debug_info.rs index 9c450c774..2f0d8ccba 100644 --- a/tonic-types/src/richer_error/std_messages/debug_info.rs +++ b/tonic-types/src/richer_error/std_messages/debug_info.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used to encode/decode the `DebugInfo` standard error message described in /// [error_details.proto]. Describes additional debugging info. @@ -48,13 +48,6 @@ impl IntoAny for DebugInfo { } } -impl FromAny for DebugInfo { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for DebugInfo { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -84,8 +77,7 @@ impl From for pb::DebugInfo { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::DebugInfo; + use super::*; #[test] fn gen_debug_info() { @@ -114,7 +106,7 @@ mod tests { "Any from filled DebugInfo differs from expected result" ); - let br_details = match DebugInfo::from_any(gen_any) { + let br_details = match DebugInfo::from_any_ref(&gen_any) { Err(error) => panic!("Error generating DebugInfo from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/error_info.rs b/tonic-types/src/richer_error/std_messages/error_info.rs index 8d535a157..9388788f4 100644 --- a/tonic-types/src/richer_error/std_messages/error_info.rs +++ b/tonic-types/src/richer_error/std_messages/error_info.rs @@ -3,9 +3,9 @@ use std::collections::HashMap; use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used to encode/decode the `ErrorInfo` standard error message described in /// [error_details.proto]. Describes the cause of the error with structured @@ -64,13 +64,6 @@ impl IntoAny for ErrorInfo { } } -impl FromAny for ErrorInfo { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for ErrorInfo { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -102,10 +95,7 @@ impl From for pb::ErrorInfo { #[cfg(test)] mod tests { - use std::collections::HashMap; - - use super::super::super::{FromAny, IntoAny}; - use super::ErrorInfo; + use super::*; #[test] fn gen_error_info() { @@ -135,7 +125,7 @@ mod tests { "Any from filled ErrorInfo differs from expected result" ); - let br_details = match ErrorInfo::from_any(gen_any) { + let br_details = match ErrorInfo::from_any_ref(&gen_any) { Err(error) => panic!("Error generating ErrorInfo from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/help.rs b/tonic-types/src/richer_error/std_messages/help.rs index 08549b2e9..1aa8e9ee3 100644 --- a/tonic-types/src/richer_error/std_messages/help.rs +++ b/tonic-types/src/richer_error/std_messages/help.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used at the `links` field of the [`Help`] struct. Describes a URL link. #[derive(Clone, Debug)] @@ -106,13 +106,6 @@ impl IntoAny for Help { } } -impl FromAny for Help { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for Help { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -140,8 +133,7 @@ impl From for pb::Help { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::Help; + use super::*; #[test] fn gen_help() { @@ -188,7 +180,7 @@ mod tests { "Any from filled Help differs from expected result" ); - let br_details = match Help::from_any(gen_any) { + let br_details = match Help::from_any_ref(&gen_any) { Err(error) => panic!("Error generating Help from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/loc_message.rs b/tonic-types/src/richer_error/std_messages/loc_message.rs index 98ab72bce..be28fbbaf 100644 --- a/tonic-types/src/richer_error/std_messages/loc_message.rs +++ b/tonic-types/src/richer_error/std_messages/loc_message.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used to encode/decode the `LocalizedMessage` standard error message /// described in [error_details.proto]. Provides a localized error message @@ -52,13 +52,6 @@ impl IntoAny for LocalizedMessage { } } -impl FromAny for LocalizedMessage { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for LocalizedMessage { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -88,8 +81,7 @@ impl From for pb::LocalizedMessage { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::LocalizedMessage; + use super::*; #[test] fn gen_localized_message() { @@ -117,7 +109,7 @@ mod tests { "Any from filled LocalizedMessage differs from expected result" ); - let br_details = match LocalizedMessage::from_any(gen_any) { + let br_details = match LocalizedMessage::from_any_ref(&gen_any) { Err(error) => panic!("Error generating LocalizedMessage from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/mod.rs b/tonic-types/src/richer_error/std_messages/mod.rs index 25b773b0b..b39f6d25e 100644 --- a/tonic-types/src/richer_error/std_messages/mod.rs +++ b/tonic-types/src/richer_error/std_messages/mod.rs @@ -1,39 +1,29 @@ mod retry_info; - pub use retry_info::RetryInfo; mod debug_info; - pub use debug_info::DebugInfo; mod quota_failure; - pub use quota_failure::{QuotaFailure, QuotaViolation}; mod error_info; - pub use error_info::ErrorInfo; mod prec_failure; - pub use prec_failure::{PreconditionFailure, PreconditionViolation}; mod bad_request; - pub use bad_request::{BadRequest, FieldViolation}; mod request_info; - pub use request_info::RequestInfo; mod resource_info; - pub use resource_info::ResourceInfo; mod help; - pub use help::{Help, HelpLink}; mod loc_message; - pub use loc_message::LocalizedMessage; diff --git a/tonic-types/src/richer_error/std_messages/prec_failure.rs b/tonic-types/src/richer_error/std_messages/prec_failure.rs index a8a5c50f1..5752a0f50 100644 --- a/tonic-types/src/richer_error/std_messages/prec_failure.rs +++ b/tonic-types/src/richer_error/std_messages/prec_failure.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used at the `violations` field of the [`PreconditionFailure`] struct. /// Describes a single precondition failure. @@ -130,13 +130,6 @@ impl IntoAny for PreconditionFailure { } } -impl FromAny for PreconditionFailure { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for PreconditionFailure { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -164,8 +157,7 @@ impl From for pb::PreconditionFailure { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::PreconditionFailure; + use super::*; #[test] fn gen_prec_failure() { @@ -213,7 +205,7 @@ mod tests { "Any from filled PreconditionFailure differs from expected result" ); - let br_details = match PreconditionFailure::from_any(gen_any) { + let br_details = match PreconditionFailure::from_any_ref(&gen_any) { Err(error) => panic!("Error generating PreconditionFailure from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/quota_failure.rs b/tonic-types/src/richer_error/std_messages/quota_failure.rs index 23d160521..4045c4b93 100644 --- a/tonic-types/src/richer_error/std_messages/quota_failure.rs +++ b/tonic-types/src/richer_error/std_messages/quota_failure.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used at the `violations` field of the [`QuotaFailure`] struct. Describes a /// single quota violation. @@ -107,13 +107,6 @@ impl IntoAny for QuotaFailure { } } -impl FromAny for QuotaFailure { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for QuotaFailure { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -141,8 +134,7 @@ impl From for pb::QuotaFailure { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::QuotaFailure; + use super::*; #[test] fn gen_quota_failure() { @@ -190,7 +182,7 @@ mod tests { "Any from filled QuotaFailure differs from expected result" ); - let br_details = match QuotaFailure::from_any(gen_any) { + let br_details = match QuotaFailure::from_any_ref(&gen_any) { Err(error) => panic!("Error generating QuotaFailure from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/request_info.rs b/tonic-types/src/richer_error/std_messages/request_info.rs index bb09f0c5b..e0f9a51b6 100644 --- a/tonic-types/src/richer_error/std_messages/request_info.rs +++ b/tonic-types/src/richer_error/std_messages/request_info.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used to encode/decode the `RequestInfo` standard error message described /// in [error_details.proto]. Contains metadata about the request that @@ -51,13 +51,6 @@ impl IntoAny for RequestInfo { } } -impl FromAny for RequestInfo { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for RequestInfo { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -87,8 +80,7 @@ impl From for pb::RequestInfo { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::RequestInfo; + use super::*; #[test] fn gen_request_info() { @@ -116,7 +108,7 @@ mod tests { "Any from filled RequestInfo differs from expected result" ); - let br_details = match RequestInfo::from_any(gen_any) { + let br_details = match RequestInfo::from_any_ref(&gen_any) { Err(error) => panic!("Error generating RequestInfo from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/resource_info.rs b/tonic-types/src/richer_error/std_messages/resource_info.rs index ccb23e1a7..8aad6e413 100644 --- a/tonic-types/src/richer_error/std_messages/resource_info.rs +++ b/tonic-types/src/richer_error/std_messages/resource_info.rs @@ -1,9 +1,9 @@ use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::*; /// Used to encode/decode the `ResourceInfo` standard error message described /// in [error_details.proto]. Describes the resource that is being accessed. @@ -64,13 +64,6 @@ impl IntoAny for ResourceInfo { } } -impl FromAny for ResourceInfo { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for ResourceInfo { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -104,8 +97,7 @@ impl From for pb::ResourceInfo { #[cfg(test)] mod tests { - use super::super::super::{FromAny, IntoAny}; - use super::ResourceInfo; + use super::*; #[test] fn gen_resource_info() { @@ -132,7 +124,7 @@ mod tests { "Any from filled ResourceInfo differs from expected result" ); - let br_details = match ResourceInfo::from_any(gen_any) { + let br_details = match ResourceInfo::from_any_ref(&gen_any) { Err(error) => panic!("Error generating ResourceInfo from Any: {:?}", error), Ok(from_any) => from_any, }; diff --git a/tonic-types/src/richer_error/std_messages/retry_info.rs b/tonic-types/src/richer_error/std_messages/retry_info.rs index a26e62076..55ebf5579 100644 --- a/tonic-types/src/richer_error/std_messages/retry_info.rs +++ b/tonic-types/src/richer_error/std_messages/retry_info.rs @@ -3,9 +3,9 @@ use std::time; use prost::{DecodeError, Message}; use prost_types::Any; -use crate::richer_error::FromAnyRef; +use crate::pb; -use super::super::{pb, FromAny, IntoAny}; +use super::super::{FromAnyRef, IntoAny}; /// Used to encode/decode the `RetryInfo` standard error message described in /// [error_details.proto]. Describes when the clients can retry a failed @@ -63,13 +63,6 @@ impl IntoAny for RetryInfo { } } -impl FromAny for RetryInfo { - #[inline] - fn from_any(any: Any) -> Result { - FromAnyRef::from_any_ref(&any) - } -} - impl FromAnyRef for RetryInfo { fn from_any_ref(any: &Any) -> Result { let buf: &[u8] = &any.value; @@ -119,8 +112,7 @@ impl From for pb::RetryInfo { mod tests { use core::time::Duration; - use super::super::super::{FromAny, IntoAny}; - use super::RetryInfo; + use super::*; #[test] fn gen_retry_info() { @@ -152,7 +144,7 @@ mod tests { "Any from filled RetryInfo differs from expected result" ); - let br_details = match RetryInfo::from_any(gen_any) { + let br_details = match RetryInfo::from_any_ref(&gen_any) { Err(error) => panic!("Error generating RetryInfo from Any: {:?}", error), Ok(from_any) => from_any, };