use std::{error, fmt};
use serde::{Deserialize, Deserializer, Serialize, Serializer};
use serde_json::Value;
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
pub enum ErrorCode {
ParseError,
InvalidRequest,
MethodNotFound,
InvalidParams,
InternalError,
ServerError(i64),
}
impl From<i64> for ErrorCode {
fn from(code: i64) -> Self {
match code {
-32700 => ErrorCode::ParseError,
-32600 => ErrorCode::InvalidRequest,
-32601 => ErrorCode::MethodNotFound,
-32602 => ErrorCode::InvalidParams,
-32603 => ErrorCode::InternalError,
code => ErrorCode::ServerError(code),
}
}
}
impl Serialize for ErrorCode {
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_i64(self.code())
}
}
impl<'de> Deserialize<'de> for ErrorCode {
fn deserialize<D>(deserializer: D) -> Result<ErrorCode, D::Error>
where
D: Deserializer<'de>,
{
let code: i64 = Deserialize::deserialize(deserializer)?;
Ok(ErrorCode::from(code))
}
}
impl ErrorCode {
pub fn code(&self) -> i64 {
match self {
ErrorCode::ParseError => -32700,
ErrorCode::InvalidRequest => -32600,
ErrorCode::MethodNotFound => -32601,
ErrorCode::InvalidParams => -32602,
ErrorCode::InternalError => -32603,
ErrorCode::ServerError(code) => *code,
}
}
pub fn description(&self) -> String {
let desc = match self {
ErrorCode::ParseError => "Parse error",
ErrorCode::InvalidRequest => "Invalid request",
ErrorCode::MethodNotFound => "Method not found",
ErrorCode::InvalidParams => "Invalid params",
ErrorCode::InternalError => "Internal error",
ErrorCode::ServerError(_) => "Server error",
};
desc.to_string()
}
}
#[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize)]
#[serde(deny_unknown_fields)]
pub struct Error {
pub code: ErrorCode,
pub message: String,
#[serde(skip_serializing_if = "Option::is_none")]
pub data: Option<Value>,
}
impl fmt::Display for Error {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}: {}", self.code.description(), self.message)
}
}
impl error::Error for Error {}
impl Error {
pub fn new(code: ErrorCode) -> Self {
Error {
message: code.description(),
code,
data: None,
}
}
pub fn parse_error() -> Self {
Self::new(ErrorCode::ParseError)
}
pub fn invalid_request() -> Self {
Self::new(ErrorCode::InvalidRequest)
}
pub fn method_not_found() -> Self {
Self::new(ErrorCode::MethodNotFound)
}
pub fn invalid_params<M>(message: M) -> Self
where
M: fmt::Display,
{
Error {
code: ErrorCode::InvalidParams,
message: format!("Invalid parameters: {}", message),
data: None,
}
}
pub fn invalid_params_with_details<M, D>(message: M, details: D) -> Self
where
M: fmt::Display,
D: fmt::Display,
{
Error {
code: ErrorCode::InvalidParams,
message: format!("Invalid parameters: {}", message),
data: Some(Value::String(details.to_string())),
}
}
pub fn internal_error() -> Self {
Self::new(ErrorCode::InternalError)
}
pub fn invalid_version() -> Self {
Error {
code: ErrorCode::InvalidRequest,
message: "Unsupported JSON-RPC protocol version".to_owned(),
data: None,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn error_serialization() {
assert_eq!(
serde_json::to_string(&Error::parse_error()).unwrap(),
r#"{"code":-32700,"message":"Parse error"}"#
);
assert_eq!(
serde_json::to_string(&Error::invalid_request()).unwrap(),
r#"{"code":-32600,"message":"Invalid request"}"#
);
assert_eq!(
serde_json::to_string(&Error::method_not_found()).unwrap(),
r#"{"code":-32601,"message":"Method not found"}"#
);
assert_eq!(
serde_json::to_string(&Error::invalid_params("unexpected params")).unwrap(),
r#"{"code":-32602,"message":"Invalid parameters: unexpected params"}"#
);
assert_eq!(
serde_json::to_string(&Error::invalid_params_with_details("unexpected params", "details")).unwrap(),
r#"{"code":-32602,"message":"Invalid parameters: unexpected params","data":"details"}"#
);
assert_eq!(
serde_json::to_string(&Error::internal_error()).unwrap(),
r#"{"code":-32603,"message":"Internal error"}"#
);
assert_eq!(
serde_json::to_string(&Error::invalid_version()).unwrap(),
r#"{"code":-32600,"message":"Unsupported JSON-RPC protocol version"}"#
);
}
}