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
use super::{Connector, ConnectorError};
use crate::rpc::eth_unsubscribe;
use crate::transport::{Request, Subscribe, TransportError};
use crate::types::U128;
use log::{error, info, trace};
use serde::de::DeserializeOwned;
use serde_json::Value;
use std::fmt::Debug;
use std::marker::PhantomData;
use thiserror::Error;
pub struct Subscription<T: DeserializeOwned + Debug, U: Subscribe + Request> {
pub id: U128,
pub(crate) connector: Connector<U>,
pub(crate) result_type: PhantomData<T>,
}
impl<T: DeserializeOwned + Debug, U: Subscribe + Request> Subscription<T, U> {
pub fn next_item(&mut self) -> Result<T, SubscriptionError> {
trace!("Fetching next item from subscription");
let response = self.connector.connection.read_next()?;
deserialize_from_sub(&response)
}
pub fn close(self) {
info!("Closing subscription with id {}", self.id);
}
}
impl<T: DeserializeOwned + Debug, U: Subscribe + Request> Drop for Subscription<T, U> {
fn drop(&mut self) {
match self.connector.call(eth_unsubscribe(self.id)) {
Ok(true) => (),
Ok(_) => error!("Unable to cancel subscription"),
Err(err) => error!("{}", err),
}
}
}
fn deserialize_from_sub<U: DeserializeOwned + Debug>(
response: &str,
) -> Result<U, SubscriptionError> {
trace!("Deserializing response {}", response);
let value = serde_json::from_str::<Value>(response)?;
serde_json::from_value::<U>(value["params"]["result"].clone()).map_err(SubscriptionError::from)
}
#[allow(clippy::large_enum_variant)]
#[derive(Debug, Error)]
pub enum SubscriptionError {
#[error("Subscription Transport Error {0}")]
Read(#[from] TransportError),
#[error("Subscription Error during canceling subscription: {0}")]
Cancel(#[from] ConnectorError),
#[error("Subscription De-/Serialization Error: {0}")]
Serde(#[from] serde_json::Error),
}