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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
use crate::*;
use std::sync::Arc;
use futures_util::{StreamExt, SinkExt};
use tokio::sync::RwLock;
use tokio::net::TcpListener;
use tokio_tungstenite::tungstenite::Message;
use ZeroMQMessage::*;
#[derive(Debug, Clone)]
pub struct WebSocketRelay {
ws_endpoint: String,
stop: Arc<RwLock<bool>>,
ready: Arc<RwLock<bool>>,
}
impl WebSocketRelay {
pub fn new(ws_endpoint: &str) -> Self {
Self {
ws_endpoint: ws_endpoint.to_string(),
stop: Arc::new(RwLock::new(false)),
ready: Arc::new(RwLock::new(false)),
}
}
pub async fn run(&self, receiver: tokio::sync::watch::Receiver<ZeroMQMessage>) {
let stop = self.stop.clone();
tokio::spawn(async move {
tokio::signal::ctrl_c().await.expect("Failed to install CTRL+C signal handler.");
*stop.write().await = true;
});
let ws_endpoint = self.ws_endpoint.clone();
let (tx, rx) = tokio::sync::watch::channel("".to_string());
let ready = self.ready.clone();
tokio::spawn(async move {
let listener = TcpListener::bind(&ws_endpoint).await.unwrap();
println!("WebSocketRelay: listening on {}", ws_endpoint);
*ready.write().await = true;
loop {
if let Ok((stream, _)) = listener.accept().await {
let mut rx = rx.clone();
tokio::spawn(async move {
let addr = stream.peer_addr().unwrap();
let ws_stream = tokio_tungstenite::accept_async(stream).await;
if ws_stream.is_err() {
return;
}
println!("WebSocketRelay: new connection from {}.", addr);
let (mut write, _read) = ws_stream.unwrap().split();
while rx.changed().await.is_ok() {
let message = (*rx.borrow()).to_string();
match write.send(Message::Text(message)).await {
Ok(_) => {},
Err(_) => break,
}
}
});
}
}
});
let mut last_message = Init;
loop {
if *self.stop.read().await {
break;
}
let message = (*receiver.borrow()).clone();
if message != last_message {
last_message = message.clone();
match message {
HashBlock(block_hash) => {
let json = serde_json::to_string(&vec!["hashblock", &hex::encode(&consensus_encode(&block_hash))]).unwrap();
tx.send(json).unwrap();
},
RawTx(transaction) => {
let txid = transaction.txid();
let json = serde_json::to_string(&vec!["hashtx", &txid.to_string()]).unwrap();
tx.send(json).unwrap();
},
Init => {},
}
} else {
tokio::time::sleep(std::time::Duration::from_millis(100)).await;
}
}
println!("WebSocketRelay stopped.");
}
pub async fn ready(&self) -> bool {
*self.ready.read().await
}
pub async fn wait_for_ready(&self) {
while !self.ready().await {
tokio::time::sleep(std::time::Duration::from_millis(1)).await;
}
}
pub async fn stop(&self) {
*self.stop.write().await = true;
}
}
#[cfg(test)]
mod tests {
use tokio::sync::watch::channel;
use super::*;
const WS_PORT: u16 = 6666;
const BLOCK_HASH: &str = "0123456789abcdef0123456789abcdef0123456789abcdef0123456789abcdef";
const RAW_TX: &str = "0200000000010122f1294bc73da293dfe1a9088c6d26d71564bf538940c7ce9c4e6212f099c3b90000000000ffffffff011e272d0100000000160014af73f777fcd64ec6d9b22ac9e1a57e127ea169ee0247304402205fea552c7d5ed3330aa4a8b5c90a980c1d3bdc72abd13c2d7bccba91fbb978f5022027fac985cfb83339fc9227e1c653b8a824c63a49cda4f9f97d48d5c07e047608012102acc07439373cc2902d0ad6602ed6f5a1b7abdf7608d265c089160ac826a4600600000000";
#[tokio::test(flavor = "multi_thread")]
async fn web_socket_relay() {
let (tx, rx) = channel(Init);
let relay = WebSocketRelay::new(&format!("localhost:{}", WS_PORT));
let handle = {
let relay = relay.clone();
tokio::spawn(async move {
relay.run(rx).await;
})
};
relay.wait_for_ready().await;
println!("Creating WebSocket client...");
let (ws_stream, _) = tokio_tungstenite::connect_async(&format!("ws://localhost:{}", WS_PORT)).await.unwrap();
let (_write, mut read) = ws_stream.split();
println!("Sending \"hashblock\"...");
tx.send(HashBlock(consensus_decode(&hex::decode(BLOCK_HASH).unwrap()))).unwrap();
println!("Reading a message from WebSocket...");
let msg = read.next().await.unwrap().unwrap().into_data();
assert_eq!(String::from_utf8(msg).unwrap(), format!("[\"hashblock\",\"{}\"]", BLOCK_HASH));
println!("Sending \"hashtx\"...");
let transaction: bitcoin::Transaction = consensus_decode(&hex::decode(RAW_TX).unwrap());
let txid = transaction.txid();
tx.send(RawTx(transaction)).unwrap();
println!("Reading a message from WebSocket...");
let msg = read.next().await.unwrap().unwrap().into_data();
assert_eq!(String::from_utf8(msg).unwrap(), format!("[\"hashtx\",\"{}\"]", txid));
relay.stop().await;
handle.await.unwrap();
}
}