lspower 1.5.0

A lightweight framework for implementing LSP servers.
Documentation
//! Error types defined by the JSON-RPC specification.

use serde::{de::Deserializer, ser::Serializer, Deserialize, Serialize};
use serde_json::Value;
use std::fmt::{self, Display, Formatter};

/// A list of numeric error codes used in JSON-RPC responses.
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum ErrorCode {
    /// Invalid JSON was received by the server.
    ParseError,
    /// The JSON sent is not a valid Request object.
    InvalidRequest,
    /// The method does not exist / is not available.
    MethodNotFound,
    /// Invalid method parameter(s).
    InvalidParams,
    /// Internal JSON-RPC error.
    InternalError,
    /// Reserved for implementation-defined server errors.
    ServerError(i64),
    /// The request was cancelled by the client.
    ///
    /// # Compatibility
    ///
    /// This error code is defined by the Language Server Protocol.
    RequestCancelled,
    /// The request was invalidated by another incoming request.
    ///
    /// # Compatibility
    ///
    /// This error code is specific to the Language Server Protocol.
    ContentModified,
}

impl ErrorCode {
    /// Returns the integer error code value.
    pub fn code(&self) -> i64 {
        match *self {
            ErrorCode::ParseError => -32700,
            ErrorCode::InvalidRequest => -32600,
            ErrorCode::MethodNotFound => -32601,
            ErrorCode::InvalidParams => -32602,
            ErrorCode::InternalError => -32603,
            ErrorCode::RequestCancelled => -32800,
            ErrorCode::ContentModified => -32801,
            ErrorCode::ServerError(code) => code,
        }
    }

    /// Returns a human-readable description of the error.
    pub fn description(&self) -> &'static str {
        match *self {
            ErrorCode::ParseError => "Parse error",
            ErrorCode::InvalidRequest => "Invalid request",
            ErrorCode::MethodNotFound => "Method not found",
            ErrorCode::InvalidParams => "Invalid params",
            ErrorCode::InternalError => "Internal error",
            ErrorCode::RequestCancelled => "Canceled",
            ErrorCode::ContentModified => "Content modified",
            ErrorCode::ServerError(_) => "Server error",
        }
    }
}

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,
            -32800 => ErrorCode::RequestCancelled,
            -32801 => ErrorCode::ContentModified,
            code => ErrorCode::ServerError(code),
        }
    }
}

impl Display for ErrorCode {
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        Display::fmt(&self.code(), f)
    }
}

impl<'a> Deserialize<'a> for ErrorCode {
    fn deserialize<D>(deserializer: D) -> std::result::Result<Self, D::Error>
    where
        D: Deserializer<'a>,
    {
        let code: i64 = Deserialize::deserialize(deserializer)?;
        Ok(ErrorCode::from(code))
    }
}

impl Serialize for ErrorCode {
    fn serialize<S>(&self, serializer: S) -> std::result::Result<S::Ok, S::Error>
    where
        S: Serializer,
    {
        self.code().serialize(serializer)
    }
}

/// A JSON-RPC error object.
#[derive(Clone, Debug, PartialEq, Deserialize, Serialize)]
#[serde(deny_unknown_fields)]
pub struct Error {
    /// A number indicating the error type that occurred.
    pub code: ErrorCode,
    /// A short description of the error.
    pub message: String,
    /// Additional information about the error, if any.
    #[serde(skip_serializing_if = "Option::is_none")]
    pub data: Option<Value>,
}

impl Error {
    /// Creates a new error from the given `ErrorCode`.
    pub fn new(code: ErrorCode) -> Self {
        Error {
            code,
            message: code.description().to_string(),
            data: None,
        }
    }

    /// Creates a new parse error (`-32700`).
    pub fn parse_error() -> Self {
        Error::new(ErrorCode::ParseError)
    }

    /// Creates a new "invalid request" error (`-32600`).
    pub fn invalid_request() -> Self {
        Error::new(ErrorCode::InvalidRequest)
    }

    /// Creates a new "method not found" error (`-32601`).
    pub fn method_not_found() -> Self {
        Error::new(ErrorCode::MethodNotFound)
    }

    /// Creates a new "invalid params" error (`-32602`).
    pub fn invalid_params<M>(message: M) -> Self
    where
        M: Into<String>,
    {
        Error {
            code: ErrorCode::InvalidParams,
            message: message.into(),
            data: None,
        }
    }

    /// Creates a new internal error (`-32603`).
    pub fn internal_error() -> Self {
        Error::new(ErrorCode::InternalError)
    }

    /// Creates a new "request cancelled" error (`-32800`).
    ///
    /// # Compatibility
    ///
    /// This error code is defined by the Language Server Protocol.
    pub fn request_cancelled() -> Self {
        Error::new(ErrorCode::RequestCancelled)
    }

    /// Creates a new "content modified" error (`-32801`).
    ///
    /// # Compatibility
    ///
    /// This error code is defined by the Language Server Protocol.
    pub fn content_modified() -> Self {
        Error::new(ErrorCode::ContentModified)
    }
}

impl Display for Error {
    fn fmt(&self, f: &mut Formatter) -> fmt::Result {
        write!(f, "{}: {}", self.code.description(), self.message)
    }
}

impl std::error::Error for Error {
}

#[cfg(test)]
mod tests {
    use crate::jsonrpc::error::*;

    #[test]
    fn display_error() {
        let error = Error::parse_error();
        assert_eq!("Parse error: Parse error", format!("{}", error));
    }

    #[test]
    fn display_error_code() {
        let code = ErrorCode::ParseError;
        assert_eq!("-32700", format!("{}", code));
    }

    #[test]
    fn parse_error() {
        let code = ErrorCode::ParseError;
        assert_eq!(code, code.code().into());
        let error = Error::parse_error();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn invalid_request() {
        let code = ErrorCode::InvalidRequest;
        assert_eq!(code, code.code().into());
        let error = Error::invalid_request();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn method_not_found() {
        let code = ErrorCode::MethodNotFound;
        assert_eq!(code, code.code().into());
        let error = Error::method_not_found();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn invalid_params() {
        let code = ErrorCode::InvalidParams;
        assert_eq!(code, code.code().into());
        let error = Error::invalid_params(code.description());
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn internal_error() {
        let code = ErrorCode::InternalError;
        assert_eq!(code, code.code().into());
        let error = Error::internal_error();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn request_cancelled() {
        let code = ErrorCode::RequestCancelled;
        assert_eq!(code, code.code().into());
        let error = Error::request_cancelled();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn content_modified() {
        let code = ErrorCode::ContentModified;
        assert_eq!(code, code.code().into());
        let error = Error::content_modified();
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn server_error() {
        let code = ErrorCode::ServerError(42);
        assert_eq!(code, code.code().into());
        let error = Error::new(code);
        assert_eq!(code, error.code);
        assert_eq!(code.description(), error.message);
    }

    #[test]
    fn server_not_initialized() {
        let code = ErrorCode::ServerError(-32002);
        let error = crate::jsonrpc::not_initialized_error();
        assert_eq!(code, error.code);
        assert_eq!("Server not initialized", error.message);
    }
}