limabean 0.4.1

Beancount frontend using Rust and Clojure and the Lima parser
Documentation
use serde::{Deserialize, Serialize};
use std::borrow::Cow;

use crate::api::types::{SyntheticSpan, booked};

use super::types::{Plugin, Report, raw::*};

#[derive(Deserialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct Request<'a> {
    pub(crate) jsonrpc: &'a str,
    pub(crate) id: Option<Id<'a>>,
    #[serde(borrow)]
    #[serde(flatten)]
    pub(crate) method: RequestMethod<'a>,
}

// TODO RequestMethod should be generic and the actual methods moved out of this module,
// but the deserialize lifetimes are a bit tricksy
#[derive(Deserialize, Clone, Debug)]
#[serde(tag = "method")]
#[serde(rename_all = "kebab-case")]
pub(crate) enum RequestMethod<'a> {
    Status,
    #[serde(rename = "parser.plugins")]
    ParserPlugins,
    #[serde(rename = "parser.directives")]
    ParserDirectives,
    #[serde(rename = "parser.options")]
    ParserOptions,
    #[serde(rename = "parser.format-errors")]
    ParserFormatErrors(Params<Vec<Report<'a>>>),
    #[serde(rename = "parser.format-warnings")]
    ParserFormatWarnings(Params<Vec<Report<'a>>>),
    #[serde(rename = "parser.resolve-span")]
    ParserResolveSpan(Params<Span>),
    #[serde(rename = "parser.create-synthetic-spans")]
    ParserCreateSyntheticSpans(Params<Vec<SyntheticSpan<'a>>>),
    #[serde(borrow)]
    Book(OptionalParams<Vec<Directive<'a>>>),
}

#[derive(Deserialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub struct Params<T> {
    pub(crate) params: T,
}

#[derive(Deserialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub struct OptionalParams<T> {
    #[serde(skip_serializing_if = "Option::is_none")]
    pub(crate) params: Option<T>,
}

#[derive(Serialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct ResultResponse<'i, 'a, 'b> {
    pub(crate) jsonrpc: &'static str,
    pub(crate) id: Option<Id<'i>>,
    pub(crate) result: ResultData<'a, 'b>,
}

impl<'i, 'a, 'b> ResultResponse<'i, 'a, 'b> {
    pub(crate) fn new(id: Option<Id<'i>>, result: ResultData<'a, 'b>) -> Self {
        ResultResponse {
            jsonrpc: JSONRPC_VERSION,
            id,
            result,
        }
    }
}

#[derive(Serialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
#[serde(untagged)]
pub(crate) enum ResultData<'a, 'b> {
    Ok,
    Plugins(&'b [Plugin<'a>]),
    #[serde(borrow)]
    RawDirectives(RawDirectives<'a>),
    Options(&'b Options<'a>),
    Report(Cow<'b, str>),
    ResolvedSpan(SpannedSource<'a>),
    Spans(Vec<Span>),
    // TODO also return warnings with booked
    BookedDirectives(BookedDirectives<'a>),
}

#[derive(Serialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct RawDirectives<'a> {
    pub(crate) directives: Vec<Directive<'a>>,
    pub(crate) warnings: Option<Vec<Report<'a>>>,
}

#[derive(Serialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct BookedDirectives<'a> {
    pub(crate) directives: Vec<booked::Directive<'a>>,
}

#[derive(Serialize, Clone, Debug)]
#[serde(rename_all = "kebab-case")]
pub(crate) struct ErrorResponse<'a, 'b, T>
where
    T: Serialize,
{
    pub(crate) jsonrpc: &'static str,
    pub(crate) id: Option<Id<'a>>,
    pub(crate) error: ErrorData<'b, T>,
}

impl<'a, 'b, T> ErrorResponse<'a, 'b, T>
where
    T: Serialize,
{
    pub(crate) fn new(
        id: Option<Id<'a>>,
        code: ErrorCode,
        message: Cow<'b, str>,
        data: Option<T>,
    ) -> Self {
        ErrorResponse {
            jsonrpc: JSONRPC_VERSION,
            id,
            error: ErrorData {
                code,
                message,
                data,
            },
        }
    }
}

#[derive(Serialize, Clone, Debug)]
pub(crate) struct ErrorData<'a, T>
where
    T: Serialize,
{
    code: i32,
    message: Cow<'a, str>,
    #[serde(skip_serializing_if = "Option::is_none")]
    data: Option<T>,
}

#[derive(Serialize, Deserialize, Copy, Clone, Debug)]
#[serde(untagged)]
pub(crate) enum Id<'a> {
    String(&'a str),
    Int(i32),
    Float(f64),
}

pub(crate) const JSONRPC_VERSION: &str = "2.0";

// https://www.jsonrpc.org/specification#error_object
pub(crate) type ErrorCode = i32;
pub(crate) const ERROR_BEANFILE_IO_ERROR: ErrorCode = 1;
pub(crate) const ERROR_REPORT: ErrorCode = 2;
pub(crate) const ERROR_INDEXED_REPORT: ErrorCode = 3;
pub(crate) const ERROR_PARSE: ErrorCode = -32700;
pub(crate) const ERROR_INVALID_REQUEST: ErrorCode = -32600;
pub(crate) const ERROR_INTERNAL: ErrorCode = -32603;