Skip to content

Commit

Permalink
feat(types): add support for DebugInfo error message type (#1179)
Browse files Browse the repository at this point in the history
  • Loading branch information
flemosr authored Dec 12, 2022
1 parent d00fd08 commit 3076e82
Show file tree
Hide file tree
Showing 6 changed files with 241 additions and 5 deletions.
2 changes: 1 addition & 1 deletion tonic-types/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ pub use pb::Status;
mod richer_error;

pub use richer_error::{
BadRequest, ErrorDetail, ErrorDetails, FieldViolation, RetryInfo, StatusExt,
BadRequest, DebugInfo, ErrorDetail, ErrorDetails, FieldViolation, RetryInfo, StatusExt,
};

mod sealed {
Expand Down
53 changes: 52 additions & 1 deletion tonic-types/src/richer_error/error_details/mod.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
use std::time;

use super::std_messages::{BadRequest, FieldViolation, RetryInfo};
use super::std_messages::{BadRequest, DebugInfo, FieldViolation, RetryInfo};

pub(crate) mod vec;

Expand All @@ -14,6 +14,9 @@ pub struct ErrorDetails {
/// This field stores [`RetryInfo`] data, if any.
pub(crate) retry_info: Option<RetryInfo>,

/// This field stores [`DebugInfo`] data, if any.
pub(crate) debug_info: Option<DebugInfo>,

/// This field stores [`BadRequest`] data, if any.
pub(crate) bad_request: Option<BadRequest>,
}
Expand All @@ -31,6 +34,7 @@ impl ErrorDetails {
pub fn new() -> Self {
ErrorDetails {
retry_info: None,
debug_info: None,
bad_request: None,
}
}
Expand All @@ -53,6 +57,25 @@ impl ErrorDetails {
}
}

/// Generates an [`ErrorDetails`] struct with [`DebugInfo`] details and
/// remaining fields set to `None`.
///
/// # Examples
///
/// ```
/// use tonic_types::{ErrorDetails};
///
/// let err_stack = vec!["...".into(), "...".into()];
///
/// let err_details = ErrorDetails::with_debug_info(err_stack, "error details");
/// ```
pub fn with_debug_info(stack_entries: Vec<String>, detail: impl Into<String>) -> Self {
ErrorDetails {
debug_info: Some(DebugInfo::new(stack_entries, detail)),
..ErrorDetails::new()
}
}

/// Generates an [`ErrorDetails`] struct with [`BadRequest`] details and
/// remaining fields set to `None`.
///
Expand Down Expand Up @@ -101,6 +124,11 @@ impl ErrorDetails {
self.retry_info.clone()
}

/// Get [`DebugInfo`] details, if any
pub fn debug_info(&self) -> Option<DebugInfo> {
self.debug_info.clone()
}

/// Get [`BadRequest`] details, if any
pub fn bad_request(&self) -> Option<BadRequest> {
self.bad_request.clone()
Expand All @@ -124,6 +152,29 @@ impl ErrorDetails {
self
}

/// Set [`DebugInfo`] details. Can be chained with other `.set_` and
/// `.add_` [`ErrorDetails`] methods.
///
/// # Examples
///
/// ```
/// use tonic_types::{ErrorDetails};
///
/// let mut err_details = ErrorDetails::new();
///
/// let err_stack = vec!["...".into(), "...".into()];
///
/// err_details.set_debug_info(err_stack, "error details");
/// ```
pub fn set_debug_info(
&mut self,
stack_entries: Vec<String>,
detail: impl Into<String>,
) -> &mut Self {
self.debug_info = Some(DebugInfo::new(stack_entries, detail));
self
}

/// Set [`BadRequest`] details. Can be chained with other `.set_` and
/// `.add_` [`ErrorDetails`] methods.
///
Expand Down
11 changes: 10 additions & 1 deletion tonic-types/src/richer_error/error_details/vec.rs
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
use super::super::std_messages::{BadRequest, RetryInfo};
use super::super::std_messages::{BadRequest, DebugInfo, RetryInfo};

/// Wraps the structs corresponding to the standard error messages, allowing
/// the implementation and handling of vectors containing any of them.
Expand All @@ -8,6 +8,9 @@ pub enum ErrorDetail {
/// Wraps the [`RetryInfo`] struct.
RetryInfo(RetryInfo),

/// Wraps the [`DebugInfo`] struct.
DebugInfo(DebugInfo),

/// Wraps the [`BadRequest`] struct.
BadRequest(BadRequest),
}
Expand All @@ -18,6 +21,12 @@ impl From<RetryInfo> for ErrorDetail {
}
}

impl From<DebugInfo> for ErrorDetail {
fn from(err_detail: DebugInfo) -> Self {
ErrorDetail::DebugInfo(err_detail)
}
}

impl From<BadRequest> for ErrorDetail {
fn from(err_detail: BadRequest) -> Self {
ErrorDetail::BadRequest(err_detail)
Expand Down
64 changes: 62 additions & 2 deletions tonic-types/src/richer_error/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ mod std_messages;
use super::pb;

pub use error_details::{vec::ErrorDetail, ErrorDetails};
pub use std_messages::{BadRequest, FieldViolation, RetryInfo};
pub use std_messages::{BadRequest, DebugInfo, FieldViolation, RetryInfo};

trait IntoAny {
fn into_any(self) -> Any;
Expand Down Expand Up @@ -266,6 +266,28 @@ pub trait StatusExt: crate::sealed::Sealed {
/// ```
fn get_details_retry_info(&self) -> Option<RetryInfo>;

/// 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<T>(req_result: Result<Response<T>, 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<DebugInfo>;

/// Get first [`BadRequest`] details found on `tonic::Status`, if any. If
/// some `prost::DecodeError` occurs, returns `None`.
///
Expand Down Expand Up @@ -306,6 +328,10 @@ impl StatusExt for tonic::Status {
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(bad_request) = details.bad_request {
conv_details.push(bad_request.into_any());
}
Expand Down Expand Up @@ -334,6 +360,9 @@ impl StatusExt for tonic::Status {
ErrorDetail::RetryInfo(retry_info) => {
conv_details.push(retry_info.into_any());
}
ErrorDetail::DebugInfo(debug_info) => {
conv_details.push(debug_info.into_any());
}
ErrorDetail::BadRequest(bad_req) => {
conv_details.push(bad_req.into_any());
}
Expand Down Expand Up @@ -368,6 +397,9 @@ impl StatusExt for tonic::Status {
RetryInfo::TYPE_URL => {
details.retry_info = Some(RetryInfo::from_any(any)?);
}
DebugInfo::TYPE_URL => {
details.debug_info = Some(DebugInfo::from_any(any)?);
}
BadRequest::TYPE_URL => {
details.bad_request = Some(BadRequest::from_any(any)?);
}
Expand All @@ -392,6 +424,9 @@ impl StatusExt for tonic::Status {
RetryInfo::TYPE_URL => {
details.push(RetryInfo::from_any(any)?.into());
}
DebugInfo::TYPE_URL => {
details.push(DebugInfo::from_any(any)?.into());
}
BadRequest::TYPE_URL => {
details.push(BadRequest::from_any(any)?.into());
}
Expand Down Expand Up @@ -422,6 +457,22 @@ impl StatusExt for tonic::Status {
None
}

fn get_details_debug_info(&self) -> Option<DebugInfo> {
let status = pb::Status::decode(self.details()).ok()?;

for any in status.details.into_iter() {
match any.type_url.as_str() {
DebugInfo::TYPE_URL => match DebugInfo::from_any(any) {
Ok(detail) => return Some(detail),
Err(_) => {}
},
_ => {}
}
}

None
}

fn get_details_bad_request(&self) -> Option<BadRequest> {
let status = pb::Status::decode(self.details()).ok()?;

Expand All @@ -444,20 +495,29 @@ mod tests {
use std::time::Duration;
use tonic::{Code, Status};

use super::{BadRequest, ErrorDetails, RetryInfo, StatusExt};
use super::{BadRequest, DebugInfo, ErrorDetails, RetryInfo, StatusExt};

#[test]
fn gen_status_with_details() {
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_bad_request_violation("field", "description");

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(),
BadRequest::with_violation("field", "description").into(),
];

Expand Down
112 changes: 112 additions & 0 deletions tonic-types/src/richer_error/std_messages/debug_info.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
use prost::{DecodeError, Message};
use prost_types::Any;

use super::super::{pb, FromAny, IntoAny};

/// Used to encode/decode the `DebugInfo` standard error message described in
/// [error_details.proto]. Describes additional debugging info.
///
/// [error_details.proto]: https://github.com/googleapis/googleapis/blob/master/google/rpc/error_details.proto
#[derive(Clone, Debug)]
pub struct DebugInfo {
/// Stack trace entries indicating where the error occurred.
pub stack_entries: Vec<String>,

/// Additional debugging information provided by the server.
pub detail: String,
}

impl DebugInfo {
/// Type URL of the `DebugInfo` standard error message type.
pub const TYPE_URL: &'static str = "type.googleapis.com/google.rpc.DebugInfo";

/// Creates a new [`DebugInfo`] struct.
pub fn new(stack_entries: Vec<String>, detail: impl Into<String>) -> Self {
DebugInfo {
stack_entries,
detail: detail.into(),
}
}
}

impl DebugInfo {
/// Returns `true` if [`DebugInfo`] fields are empty, and `false` if they
/// are not.
pub fn is_empty(&self) -> bool {
self.stack_entries.is_empty() && self.detail.is_empty()
}
}

impl IntoAny for DebugInfo {
fn into_any(self) -> Any {
let detail_data = pb::DebugInfo {
stack_entries: self.stack_entries,
detail: self.detail,
};

Any {
type_url: DebugInfo::TYPE_URL.to_string(),
value: detail_data.encode_to_vec(),
}
}
}

impl FromAny for DebugInfo {
fn from_any(any: Any) -> Result<Self, DecodeError> {
let buf: &[u8] = &any.value;
let debug_info = pb::DebugInfo::decode(buf)?;

let debug_info = DebugInfo {
stack_entries: debug_info.stack_entries,
detail: debug_info.detail,
};

Ok(debug_info)
}
}

#[cfg(test)]
mod tests {
use super::super::super::{FromAny, IntoAny};
use super::DebugInfo;

#[test]
fn gen_debug_info() {
let debug_info = DebugInfo::new(
vec!["trace 3".into(), "trace 2".into(), "trace 1".into()],
"details about the error",
);

let formatted = format!("{:?}", debug_info);

let expected_filled = "DebugInfo { stack_entries: [\"trace 3\", \"trace 2\", \"trace 1\"], detail: \"details about the error\" }";

assert!(
formatted.eq(expected_filled),
"filled DebugInfo differs from expected result"
);

let gen_any = debug_info.into_any();
let formatted = format!("{:?}", gen_any);

let expected =
"Any { type_url: \"type.googleapis.com/google.rpc.DebugInfo\", value: [10, 7, 116, 114, 97, 99, 101, 32, 51, 10, 7, 116, 114, 97, 99, 101, 32, 50, 10, 7, 116, 114, 97, 99, 101, 32, 49, 18, 23, 100, 101, 116, 97, 105, 108, 115, 32, 97, 98, 111, 117, 116, 32, 116, 104, 101, 32, 101, 114, 114, 111, 114] }";

assert!(
formatted.eq(expected),
"Any from filled DebugInfo differs from expected result"
);

let br_details = match DebugInfo::from_any(gen_any) {
Err(error) => panic!("Error generating DebugInfo from Any: {:?}", error),
Ok(from_any) => from_any,
};

let formatted = format!("{:?}", br_details);

assert!(
formatted.eq(expected_filled),
"DebugInfo from Any differs from expected result"
);
}
}
4 changes: 4 additions & 0 deletions tonic-types/src/richer_error/std_messages/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,10 @@ mod retry_info;

pub use retry_info::RetryInfo;

mod debug_info;

pub use debug_info::DebugInfo;

mod bad_request;

pub use bad_request::{BadRequest, FieldViolation};

0 comments on commit 3076e82

Please sign in to comment.