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
use crate::{JsonRpcClient, Middleware, Provider, TransactionStream};
use ethers_core::types::{TxHash, U256};
use futures_util::stream::Stream;
use pin_project::{pin_project, pinned_drop};
use serde::de::DeserializeOwned;
use serde_json::Value;
use std::{
marker::PhantomData,
pin::Pin,
task::{Context, Poll},
};
pub trait PubsubClient: JsonRpcClient {
type NotificationStream: futures_core::Stream<Item = Value> + Send + Unpin;
fn subscribe<T: Into<U256>>(&self, id: T) -> Result<Self::NotificationStream, Self::Error>;
fn unsubscribe<T: Into<U256>>(&self, id: T) -> Result<(), Self::Error>;
}
#[must_use = "subscriptions do nothing unless you stream them"]
#[pin_project(PinnedDrop)]
pub struct SubscriptionStream<'a, P: PubsubClient, R: DeserializeOwned> {
pub id: U256,
provider: &'a Provider<P>,
#[pin]
rx: P::NotificationStream,
ret: PhantomData<R>,
}
impl<'a, P, R> SubscriptionStream<'a, P, R>
where
P: PubsubClient,
R: DeserializeOwned,
{
pub fn new(id: U256, provider: &'a Provider<P>) -> Result<Self, P::Error> {
let rx = provider.as_ref().subscribe(id)?;
Ok(Self {
id,
provider,
rx,
ret: PhantomData,
})
}
pub async fn unsubscribe(&self) -> Result<bool, crate::ProviderError> {
self.provider.unsubscribe(self.id).await
}
}
impl<'a, P, R> Stream for SubscriptionStream<'a, P, R>
where
P: PubsubClient,
R: DeserializeOwned,
{
type Item = R;
fn poll_next(self: Pin<&mut Self>, ctx: &mut Context) -> Poll<Option<Self::Item>> {
let this = self.project();
match futures_util::ready!(this.rx.poll_next(ctx)) {
Some(item) => match serde_json::from_value(item) {
Ok(res) => Poll::Ready(Some(res)),
_ => Poll::Pending,
},
None => Poll::Ready(None),
}
}
}
#[pinned_drop]
impl<P, R> PinnedDrop for SubscriptionStream<'_, P, R>
where
P: PubsubClient,
R: DeserializeOwned,
{
fn drop(self: Pin<&mut Self>) {
let _ = (*self.provider).as_ref().unsubscribe(self.id);
}
}
impl<'a, P> SubscriptionStream<'a, P, TxHash>
where
P: PubsubClient,
{
pub fn transactions_unordered(self, n: usize) -> TransactionStream<'a, P, Self> {
TransactionStream::new(self.provider, self, n)
}
}