use std::collections::HashMap;
type ErrorDatum = Box<dyn erased_serde::Serialize + Send + 'static>;
#[derive(serde::Serialize)]
pub struct RpcError {
message: String,
#[serde(serialize_with = "ser_code")]
code: RpcErrorKind,
#[serde(serialize_with = "ser_kind")]
kinds: AnyErrorKind,
#[serde(skip_serializing_if = "Option::is_none")]
data: Option<HashMap<String, ErrorDatum>>,
}
impl RpcError {
pub fn new(message: String, code: RpcErrorKind) -> Self {
Self {
message,
code,
kinds: AnyErrorKind::Rpc(code),
data: None,
}
}
pub fn set_kind(&mut self, kind: tor_error::ErrorKind) {
self.kinds = AnyErrorKind::Tor(kind);
}
pub fn set_datum<D>(&mut self, keyword: String, datum: D)
where
D: serde::Serialize + Send + 'static,
{
self.data
.get_or_insert_with(HashMap::new)
.insert(keyword, Box::new(datum) as _);
}
pub fn is_internal(&self) -> bool {
matches!(
self.kinds,
AnyErrorKind::Tor(tor_error::ErrorKind::Internal)
| AnyErrorKind::Rpc(RpcErrorKind::InternalError)
)
}
}
impl<T> From<T> for RpcError
where
T: std::error::Error + tor_error::HasKind + Send + 'static,
{
fn from(value: T) -> RpcError {
use tor_error::ErrorReport as _;
let message = value.report().to_string();
let code = kind_to_code(value.kind());
let kinds = AnyErrorKind::Tor(value.kind());
RpcError {
message,
code,
kinds,
data: None,
}
}
}
fn ser_kind<S: serde::Serializer>(kind: &AnyErrorKind, s: S) -> Result<S::Ok, S::Error> {
use serde::ser::SerializeSeq;
let mut seq = s.serialize_seq(None)?;
match kind {
AnyErrorKind::Tor(kind) => seq.serialize_element(&format!("arti:{:?}", kind))?,
AnyErrorKind::Rpc(kind) => seq.serialize_element(&format!("rpc:{:?}", kind))?,
}
seq.end()
}
fn ser_code<S: serde::Serializer>(kind: &RpcErrorKind, s: S) -> Result<S::Ok, S::Error> {
s.serialize_i32(*kind as i32)
}
#[derive(Clone, Copy, Debug)]
enum AnyErrorKind {
Tor(tor_error::ErrorKind),
#[allow(unused)]
Rpc(RpcErrorKind),
}
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
#[repr(i32)]
#[non_exhaustive]
pub enum RpcErrorKind {
InvalidRequest = -32600,
NoSuchMethod = -32601,
InvalidMethodParameters = -32602,
InternalError = -32603,
ObjectNotFound = 1,
RequestError = 2,
MethodNotImpl = 3,
RequestCancelled = 4,
FeatureNotPresent = 5,
}
fn kind_to_code(kind: tor_error::ErrorKind) -> RpcErrorKind {
use tor_error::ErrorKind as EK;
use RpcErrorKind as RC;
match kind {
EK::Internal | EK::BadApiUsage => RC::InternalError,
_ => RC::RequestError, }
}
impl std::fmt::Debug for RpcError {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("RpcError")
.field("message", &self.message)
.field("code", &self.code)
.field("kinds", &self.kinds)
.finish()
}
}
#[cfg(test)]
mod test {
#![allow(clippy::bool_assert_comparison)]
#![allow(clippy::clone_on_copy)]
#![allow(clippy::dbg_macro)]
#![allow(clippy::mixed_attributes_style)]
#![allow(clippy::print_stderr)]
#![allow(clippy::print_stdout)]
#![allow(clippy::single_char_pattern)]
#![allow(clippy::unwrap_used)]
#![allow(clippy::unchecked_duration_subtraction)]
#![allow(clippy::useless_vec)]
#![allow(clippy::needless_pass_by_value)]
use super::*;
#[derive(Debug, thiserror::Error, serde::Serialize)]
enum ExampleError {
#[error("The {} exploded because {}", what, why)]
SomethingExploded { what: String, why: String },
#[error("I'm hiding the {0} in my {1}")]
SomethingWasHidden(String, String),
#[error("The {0} was missing")]
SomethingWasMissing(String),
#[error("I don't feel up to it today")]
ProgramUnwilling,
}
impl tor_error::HasKind for ExampleError {
fn kind(&self) -> tor_error::ErrorKind {
match self {
Self::SomethingExploded { .. } => tor_error::ErrorKind::Other,
Self::SomethingWasHidden(_, _) => tor_error::ErrorKind::RemoteHostNotFound,
Self::SomethingWasMissing(_) => tor_error::ErrorKind::FeatureDisabled,
Self::ProgramUnwilling => tor_error::ErrorKind::Internal,
}
}
}
macro_rules! assert_json_eq {
($a:expr, $b:expr) => {
let json_a: serde_json::Value = serde_json::from_str($a).unwrap();
let json_b: serde_json::Value = serde_json::from_str($b).unwrap();
assert_eq!(json_a, json_b);
};
}
#[test]
fn serialize_error() {
let err = ExampleError::SomethingExploded {
what: "previous implementation".into(),
why: "worse things happen at C".into(),
};
let err = RpcError::from(err);
assert_eq!(err.code, RpcErrorKind::RequestError);
let serialized = serde_json::to_string(&err).unwrap();
let expected_json = r#"
{
"message": "error: The previous implementation exploded because worse things happen at C",
"code": 2,
"kinds": ["arti:Other"]
}
"#;
assert_json_eq!(&serialized, expected_json);
let err = ExampleError::SomethingWasHidden(
"zircon-encrusted tweezers".into(),
"chrome dinette".into(),
);
let err = RpcError::from(err);
let serialized = serde_json::to_string(&err).unwrap();
let expected = r#"
{
"message": "error: I'm hiding the zircon-encrusted tweezers in my chrome dinette",
"code": 2,
"kinds": ["arti:RemoteHostNotFound"]
}
"#;
assert_json_eq!(&serialized, expected);
let err = ExampleError::SomethingWasMissing("turbo-encabulator".into());
let err = RpcError::from(err);
let serialized = serde_json::to_string(&err).unwrap();
let expected = r#"
{
"message": "error: The turbo-encabulator was missing",
"code": 2,
"kinds": ["arti:FeatureDisabled"]
}
"#;
assert_json_eq!(&serialized, expected);
let err = ExampleError::ProgramUnwilling;
let err = RpcError::from(err);
let serialized = serde_json::to_string(&err).unwrap();
let expected = r#"
{
"message": "error: I don't feel up to it today",
"code": -32603,
"kinds": ["arti:Internal"]
}
"#;
assert_json_eq!(&serialized, expected);
}
#[test]
fn create_error() {
let mut e = RpcError::new("Example error".to_string(), RpcErrorKind::RequestError);
e.set_kind(tor_error::ErrorKind::CacheCorrupted);
e.set_datum("rpc:example".to_string(), "Hello world".to_string());
let serialized = serde_json::to_string(&e).unwrap();
let expected = r#"
{
"message": "Example error",
"code": 2,
"kinds": ["arti:CacheCorrupted"],
"data": {
"rpc:example": "Hello world"
}
}
"#;
assert_json_eq!(&serialized, expected);
}
}