1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
use serde::{Deserialize, Serialize};
use crate::error::JsonRpcError;
pub use anyhow::Error;
#[cfg(feature = "client")]
use client_imports::*;
pub use serde::de::DeserializeOwned;
pub use serde_json::{from_value, json, Value};
#[cfg(feature = "client")]
mod client_imports {
pub use std::sync::atomic::{AtomicU64, Ordering};
pub use std::sync::Arc;
pub use std::time::Duration;
pub use reqwest::{Client as ClientR, Url};
pub use serde::Deserialize;
pub use crate::error::JsonRpcError;
pub use crate::params::Params;
}
pub mod error;
pub mod params;
#[cfg(feature = "client")]
#[derive(Clone, Debug)]
pub struct Client {
inner: ClientR,
url: Url,
id: Arc<AtomicU64>,
}
#[cfg(feature = "client")]
impl Client {
#[must_use]
pub fn new(url: &Url, timeout: Option<Duration>, connection_timeout: Option<Duration>) -> Self {
let builder = ClientR::builder();
let client = builder
.connect_timeout(connection_timeout.unwrap_or_else(|| Duration::from_secs(10)))
.timeout(timeout.unwrap_or_else(|| Duration::from_secs(5)))
.build()
.expect("Shouldn't happen on this set of options");
Client {
inner: client,
url: url.clone(),
id: Arc::new(AtomicU64::new(0)),
}
}
pub async fn request<M, P, Ret>(&self, method: M, params: P) -> Result<Ret, Error>
where
M: AsRef<str> + Send,
P: Serialize,
Ret: DeserializeOwned,
{
#[derive(Deserialize, Debug)]
struct JsonRpcData {
result: Option<Value>,
error: Option<Value>,
}
let client = &self.inner;
let id = { self.id.fetch_add(1, Ordering::SeqCst) };
let json_payload = json!({
"jsonrpc": "2.0",
"method": method.as_ref(),
"params": params,
"id": id
});
let res = client
.post(self.url.clone())
.json(&json_payload)
.send()
.await?
.text()
.await?;
parse_response(&res)
}
}
pub fn parse_response<Ret>(data: &str) -> Result<Ret, anyhow::Error>
where
Ret: DeserializeOwned,
{
#[derive(Deserialize, Debug)]
struct JsonRpcData {
result: Option<Value>,
error: Option<Value>,
}
let response: JsonRpcData = serde_json::from_str(data)?;
match response.error {
Some(a) => Err(parse_error(a)?.into()),
None => match response.result {
Some(a) => Ok(from_value(a)?),
None => Err(Error::msg("Bad server answer")),
},
}
}
fn parse_error(value: Value) -> Result<JsonRpcError, Error> {
#[derive(Deserialize)]
struct ErrorObj {
code: i32,
message: String,
data: Option<String>,
}
let error_obj: ErrorObj = serde_json::from_value(value)?;
Ok(JsonRpcError {
code: error_obj.code,
data: error_obj.data,
message: error_obj.message,
})
}