mirror of
https://github.com/ElementsProject/lightning.git
synced 2025-02-22 14:42:40 +01:00
88 lines
2.7 KiB
Rust
88 lines
2.7 KiB
Rust
//! Common structs to handle JSON-RPC decoding and encoding. They are
|
|
//! generic over the Notification and Request types.
|
|
|
|
use serde::ser::{SerializeStruct, Serializer};
|
|
use serde::de::{self, Deserializer};
|
|
use serde::{Deserialize, Serialize};
|
|
use serde_json::Value;
|
|
use std::fmt::Debug;
|
|
|
|
#[derive(Debug)]
|
|
pub enum JsonRpc<N, R> {
|
|
Request(serde_json::Value, R),
|
|
Notification(N),
|
|
}
|
|
|
|
/// This function disentangles the various cases:
|
|
///
|
|
/// 1) If we have an `id` then it is a request
|
|
///
|
|
/// 2) Otherwise it's a notification that doesn't require a
|
|
/// response.
|
|
///
|
|
/// Furthermore we distinguish between the built-in types and the
|
|
/// custom user notifications/methods:
|
|
///
|
|
/// 1) We either match a built-in type above,
|
|
///
|
|
/// 2) Or it's a custom one, so we pass it around just as a
|
|
/// `serde_json::Value`
|
|
impl<'de, N, R> Deserialize<'de> for JsonRpc<N, R>
|
|
where
|
|
N: Deserialize<'de> + Debug,
|
|
R: Deserialize<'de> + Debug,
|
|
{
|
|
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
|
|
where
|
|
D: Deserializer<'de>,
|
|
{
|
|
#[derive(Deserialize, Debug)]
|
|
struct IdHelper {
|
|
id: Option<serde_json::Value>,
|
|
}
|
|
|
|
let v = Value::deserialize(deserializer)?;
|
|
let helper = IdHelper::deserialize(&v).map_err(de::Error::custom)?;
|
|
match helper.id {
|
|
Some(id) => {
|
|
let r = R::deserialize(v).map_err(de::Error::custom)?;
|
|
Ok(JsonRpc::Request(id, r))
|
|
}
|
|
None => {
|
|
let n = N::deserialize(v).map_err(de::Error::custom)?;
|
|
Ok(JsonRpc::Notification(n))
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<N, R> Serialize for JsonRpc<N, R>
|
|
where
|
|
N: Serialize + Debug,
|
|
R: Serialize + Debug,
|
|
{
|
|
fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
|
|
where
|
|
S: Serializer,
|
|
{
|
|
match self {
|
|
JsonRpc::Notification(r) => {
|
|
let r = serde_json::to_value(r).unwrap();
|
|
let mut s = serializer.serialize_struct("Notification", 3)?;
|
|
s.serialize_field("jsonrpc", "2.0")?;
|
|
s.serialize_field("method", &r["method"])?;
|
|
s.serialize_field("params", &r["params"])?;
|
|
s.end()
|
|
}
|
|
JsonRpc::Request(id, r) => {
|
|
let r = serde_json::to_value(r).unwrap();
|
|
let mut s = serializer.serialize_struct("Request", 4)?;
|
|
s.serialize_field("jsonrpc", "2.0")?;
|
|
s.serialize_field("id", id)?;
|
|
s.serialize_field("method", &r["method"])?;
|
|
s.serialize_field("params", &r["params"])?;
|
|
s.end()
|
|
}
|
|
}
|
|
}
|
|
}
|