mirror of
https://github.com/mimblewimble/grin.git
synced 2025-01-21 03:21:08 +03:00
Added a Connection wrapper to handle timeouts when we want information from a peer in a request/response style.
This commit is contained in:
parent
572c1951e1
commit
7f029cb4c0
5 changed files with 196 additions and 81 deletions
|
@ -11,6 +11,7 @@ log = "^0.3"
|
||||||
net2 = "0.2.0"
|
net2 = "0.2.0"
|
||||||
rand = "^0.3"
|
rand = "^0.3"
|
||||||
tokio-core="^0.1.1"
|
tokio-core="^0.1.1"
|
||||||
|
tokio-timer="^0.1.0"
|
||||||
time = "^0.1"
|
time = "^0.1"
|
||||||
enum_primitive = "^0.1.0"
|
enum_primitive = "^0.1.0"
|
||||||
num = "^0.1.36"
|
num = "^0.1.36"
|
||||||
|
|
146
p2p/src/conn.rs
146
p2p/src/conn.rs
|
@ -12,11 +12,14 @@
|
||||||
// See the License for the specific language governing permissions and
|
// See the License for the specific language governing permissions and
|
||||||
// limitations under the License.
|
// limitations under the License.
|
||||||
|
|
||||||
//! Provides a connection wrapper that handles the lower level tasks in sending or
|
//! Provides a connection wrapper that handles the lower level tasks in sending
|
||||||
|
//! or
|
||||||
//! receiving data from the TCP socket, as well as dealing with timeouts.
|
//! receiving data from the TCP socket, as well as dealing with timeouts.
|
||||||
|
|
||||||
use std::iter;
|
use std::iter;
|
||||||
|
use std::ops::Deref;
|
||||||
use std::sync::{Mutex, Arc};
|
use std::sync::{Mutex, Arc};
|
||||||
|
use std::time::{Instant, Duration};
|
||||||
|
|
||||||
use futures;
|
use futures;
|
||||||
use futures::{Stream, Future};
|
use futures::{Stream, Future};
|
||||||
|
@ -24,30 +27,42 @@ use futures::stream;
|
||||||
use futures::sync::mpsc::{Sender, UnboundedSender, UnboundedReceiver};
|
use futures::sync::mpsc::{Sender, UnboundedSender, UnboundedReceiver};
|
||||||
use tokio_core::io::{Io, WriteHalf, ReadHalf, write_all, read_exact};
|
use tokio_core::io::{Io, WriteHalf, ReadHalf, write_all, read_exact};
|
||||||
use tokio_core::net::TcpStream;
|
use tokio_core::net::TcpStream;
|
||||||
|
use tokio_timer::{Timer, TimerError};
|
||||||
|
|
||||||
|
use core::core::hash::{Hash, ZERO_HASH};
|
||||||
use core::ser;
|
use core::ser;
|
||||||
use msg::*;
|
use msg::*;
|
||||||
|
|
||||||
/// Handler to provide to the connection, will be called back anytime a message is
|
/// Handler to provide to the connection, will be called back anytime a message
|
||||||
/// received. The provided sender can be use to immediately send back another
|
/// is received. The provided sender can be use to immediately send back
|
||||||
/// message.
|
/// another message.
|
||||||
pub trait Handler: Sync + Send {
|
pub trait Handler: Sync + Send {
|
||||||
/// Handle function to implement to process incoming messages. A sender to reply
|
/// Handle function to implement to process incoming messages. A sender to
|
||||||
/// immediately as well as the message header and its unparsed body are provided.
|
/// reply immediately as well as the message header and its unparsed body
|
||||||
fn handle(&self, sender: UnboundedSender<Vec<u8>>, header: MsgHeader, body: Vec<u8>) -> Result<(), ser::Error>;
|
/// are provided.
|
||||||
|
fn handle(&self,
|
||||||
|
sender: UnboundedSender<Vec<u8>>,
|
||||||
|
header: MsgHeader,
|
||||||
|
body: Vec<u8>)
|
||||||
|
-> Result<Option<Hash>, ser::Error>;
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<F> Handler for F
|
impl<F> Handler for F
|
||||||
where F: Fn(UnboundedSender<Vec<u8>>, MsgHeader, Vec<u8>) -> Result<(), ser::Error>, F: Sync + Send {
|
where F: Fn(UnboundedSender<Vec<u8>>, MsgHeader, Vec<u8>) -> Result<Option<Hash>, ser::Error>,
|
||||||
|
F: Sync + Send
|
||||||
fn handle(&self, sender: UnboundedSender<Vec<u8>>, header: MsgHeader, body: Vec<u8>) -> Result<(), ser::Error> {
|
{
|
||||||
|
fn handle(&self,
|
||||||
|
sender: UnboundedSender<Vec<u8>>,
|
||||||
|
header: MsgHeader,
|
||||||
|
body: Vec<u8>)
|
||||||
|
-> Result<Option<Hash>, ser::Error> {
|
||||||
self(sender, header, body)
|
self(sender, header, body)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// A higher level connection wrapping the TcpStream. Maintains the amount of data
|
/// A higher level connection wrapping the TcpStream. Maintains the amount of
|
||||||
/// transmitted and deals with the low-level task of sending and receiving
|
/// data transmitted and deals with the low-level task of sending and
|
||||||
/// data, parsing message headers and timeouts.
|
/// receiving data, parsing message headers and timeouts.
|
||||||
pub struct Connection {
|
pub struct Connection {
|
||||||
// Channel to push bytes to the remote peer
|
// Channel to push bytes to the remote peer
|
||||||
outbound_chan: UnboundedSender<Vec<u8>>,
|
outbound_chan: UnboundedSender<Vec<u8>>,
|
||||||
|
@ -66,11 +81,14 @@ pub struct Connection {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Connection {
|
impl Connection {
|
||||||
|
/// Start listening on the provided connection and wraps it. Does not hang
|
||||||
/// Start listening on the provided connection and wraps it. Does not hang the
|
/// the current thread, instead just returns a future and the Connection
|
||||||
/// current thread, instead just returns a future and the Connection itself.
|
/// itself.
|
||||||
pub fn listen<F>(conn: TcpStream, handler: F) -> (Connection, Box<Future<Item = (), Error = ser::Error>>)
|
pub fn listen<F>(conn: TcpStream,
|
||||||
where F: Handler + 'static {
|
handler: F)
|
||||||
|
-> (Connection, Box<Future<Item = (), Error = ser::Error>>)
|
||||||
|
where F: Handler + 'static
|
||||||
|
{
|
||||||
|
|
||||||
let (reader, writer) = conn.split();
|
let (reader, writer) = conn.split();
|
||||||
|
|
||||||
|
@ -97,7 +115,8 @@ impl Connection {
|
||||||
let write_msg = me.write_msg(rx, writer).map(|_| ());
|
let write_msg = me.write_msg(rx, writer).map(|_| ());
|
||||||
|
|
||||||
// select between our different futures and return them
|
// select between our different futures and return them
|
||||||
let fut = Box::new(close_conn.select(read_msg.select(write_msg).map(|_| ()).map_err(|(e, _)| e))
|
let fut =
|
||||||
|
Box::new(close_conn.select(read_msg.select(write_msg).map(|_| ()).map_err(|(e, _)| e))
|
||||||
.map(|_| ())
|
.map(|_| ())
|
||||||
.map_err(|(e, _)| e));
|
.map_err(|(e, _)| e));
|
||||||
|
|
||||||
|
@ -132,7 +151,8 @@ impl Connection {
|
||||||
reader: ReadHalf<TcpStream>,
|
reader: ReadHalf<TcpStream>,
|
||||||
handler: F)
|
handler: F)
|
||||||
-> Box<Future<Item = ReadHalf<TcpStream>, Error = ser::Error>>
|
-> Box<Future<Item = ReadHalf<TcpStream>, Error = ser::Error>>
|
||||||
where F: Handler + 'static {
|
where F: Handler + 'static
|
||||||
|
{
|
||||||
|
|
||||||
// infinite iterator stream so we repeat the message reading logic until the
|
// infinite iterator stream so we repeat the message reading logic until the
|
||||||
// peer is stopped
|
// peer is stopped
|
||||||
|
@ -197,3 +217,89 @@ impl Connection {
|
||||||
(sent, recv)
|
(sent, recv)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Connection wrapper that handles a request/response oriented interaction with
|
||||||
|
/// a timeout.
|
||||||
|
pub struct TimeoutConnection {
|
||||||
|
underlying: Connection,
|
||||||
|
|
||||||
|
expected_responses: Arc<Mutex<Vec<(Type, Hash, Instant)>>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl TimeoutConnection {
|
||||||
|
/// Same as Connection
|
||||||
|
pub fn listen<F>(conn: TcpStream,
|
||||||
|
handler: F)
|
||||||
|
-> (TimeoutConnection, Box<Future<Item = (), Error = ser::Error>>)
|
||||||
|
where F: Handler + 'static
|
||||||
|
{
|
||||||
|
|
||||||
|
let expects = Arc::new(Mutex::new(vec![]));
|
||||||
|
|
||||||
|
// Decorates the handler to remove the "subscription" from the expected
|
||||||
|
// responses. We got our replies, so no timeout should occur.
|
||||||
|
let exp = expects.clone();
|
||||||
|
let (conn, fut) = Connection::listen(conn, move |sender, header: MsgHeader, data| {
|
||||||
|
let msg_type = header.msg_type;
|
||||||
|
let recv_h = try!(handler.handle(sender, header, data));
|
||||||
|
|
||||||
|
let mut expects = exp.lock().unwrap();
|
||||||
|
let filtered = expects.iter()
|
||||||
|
.filter(|&&(typ, h, _)| msg_type != typ || recv_h.is_some() && recv_h.unwrap() != h)
|
||||||
|
.map(|&x| x)
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
*expects = filtered;
|
||||||
|
|
||||||
|
Ok(recv_h)
|
||||||
|
});
|
||||||
|
|
||||||
|
// Registers a timer with the event loop to regularly check for timeouts.
|
||||||
|
let exp = expects.clone();
|
||||||
|
let timer = Timer::default()
|
||||||
|
.interval(Duration::new(2, 0))
|
||||||
|
.fold((), move |_, _| {
|
||||||
|
let exp = exp.lock().unwrap();
|
||||||
|
for &(_, _, t) in exp.deref() {
|
||||||
|
if Instant::now() - t > Duration::new(2, 0) {
|
||||||
|
return Err(TimerError::TooLong);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
})
|
||||||
|
.map_err(|_| ser::Error::CorruptedData);
|
||||||
|
|
||||||
|
let me = TimeoutConnection {
|
||||||
|
underlying: conn,
|
||||||
|
expected_responses: expects,
|
||||||
|
};
|
||||||
|
(me, Box::new(fut.join(timer).map(|_| ())))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Sends a request and registers a timer on the provided message type and
|
||||||
|
/// optionally the hash of the sent data.
|
||||||
|
pub fn send_request(&self,
|
||||||
|
t: Type,
|
||||||
|
body: &ser::Writeable,
|
||||||
|
expect_h: Option<Hash>)
|
||||||
|
-> Result<(), ser::Error> {
|
||||||
|
let sent = try!(self.underlying.send_msg(t, body));
|
||||||
|
|
||||||
|
let mut expects = self.expected_responses.lock().unwrap();
|
||||||
|
if let Some(h) = expect_h {
|
||||||
|
expects.push((t, h, Instant::now()));
|
||||||
|
} else {
|
||||||
|
expects.push((t, ZERO_HASH, Instant::now()));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Same as Connection
|
||||||
|
pub fn send_msg(&self, t: Type, body: &ser::Writeable) -> Result<(), ser::Error> {
|
||||||
|
self.underlying.send_msg(t, body)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Same as Connection
|
||||||
|
pub fn transmitted_bytes(&self) -> (u64, u64) {
|
||||||
|
self.underlying.transmitted_bytes()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -32,6 +32,7 @@ extern crate log;
|
||||||
extern crate futures;
|
extern crate futures;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
extern crate tokio_core;
|
extern crate tokio_core;
|
||||||
|
extern crate tokio_timer;
|
||||||
extern crate rand;
|
extern crate rand;
|
||||||
extern crate time;
|
extern crate time;
|
||||||
extern crate num;
|
extern crate num;
|
||||||
|
|
|
@ -44,7 +44,7 @@ pub enum ErrCodes {
|
||||||
|
|
||||||
/// Types of messages
|
/// Types of messages
|
||||||
enum_from_primitive! {
|
enum_from_primitive! {
|
||||||
#[derive(Debug, Clone, Copy)]
|
#[derive(Debug, Clone, Copy, PartialEq)]
|
||||||
pub enum Type {
|
pub enum Type {
|
||||||
Error,
|
Error,
|
||||||
Hand,
|
Hand,
|
||||||
|
|
|
@ -23,13 +23,13 @@ use tokio_core::net::TcpStream;
|
||||||
use core::core;
|
use core::core;
|
||||||
use core::core::hash::Hash;
|
use core::core::hash::Hash;
|
||||||
use core::ser;
|
use core::ser;
|
||||||
use conn::Connection;
|
use conn::TimeoutConnection;
|
||||||
use msg::*;
|
use msg::*;
|
||||||
use types::*;
|
use types::*;
|
||||||
use util::OneTime;
|
use util::OneTime;
|
||||||
|
|
||||||
pub struct ProtocolV1 {
|
pub struct ProtocolV1 {
|
||||||
conn: OneTime<Connection>,
|
conn: OneTime<TimeoutConnection>,
|
||||||
|
|
||||||
expected_responses: Mutex<Vec<(Type, Hash)>>,
|
expected_responses: Mutex<Vec<(Type, Hash)>>,
|
||||||
}
|
}
|
||||||
|
@ -50,7 +50,7 @@ impl Protocol for ProtocolV1 {
|
||||||
adapter: Arc<NetAdapter>)
|
adapter: Arc<NetAdapter>)
|
||||||
-> Box<Future<Item = (), Error = ser::Error>> {
|
-> Box<Future<Item = (), Error = ser::Error>> {
|
||||||
|
|
||||||
let (conn, listener) = Connection::listen(conn, move |sender, header, data| {
|
let (conn, listener) = TimeoutConnection::listen(conn, move |sender, header, data| {
|
||||||
let adapt = adapter.as_ref();
|
let adapt = adapter.as_ref();
|
||||||
handle_payload(adapt, sender, header, data)
|
handle_payload(adapt, sender, header, data)
|
||||||
});
|
});
|
||||||
|
@ -68,7 +68,7 @@ impl Protocol for ProtocolV1 {
|
||||||
/// Sends a ping message to the remote peer. Will panic if handle has never
|
/// Sends a ping message to the remote peer. Will panic if handle has never
|
||||||
/// been called on this protocol.
|
/// been called on this protocol.
|
||||||
fn send_ping(&self) -> Result<(), ser::Error> {
|
fn send_ping(&self) -> Result<(), ser::Error> {
|
||||||
self.send_msg(Type::Ping, &Empty {})
|
self.send_request(Type::Ping, &Empty {}, None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Serializes and sends a block to our remote peer
|
/// Serializes and sends a block to our remote peer
|
||||||
|
@ -88,12 +88,15 @@ impl Protocol for ProtocolV1 {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ProtocolV1 {
|
impl ProtocolV1 {
|
||||||
|
|
||||||
fn send_msg(&self, t: Type, body: &ser::Writeable) -> Result<(), ser::Error> {
|
fn send_msg(&self, t: Type, body: &ser::Writeable) -> Result<(), ser::Error> {
|
||||||
self.conn.borrow().send_msg(t, body)
|
self.conn.borrow().send_msg(t, body)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn send_request(&self, t: Type, body: &ser::Writeable, expect_resp: Option<(Type, Hash)>) -> Result<(), ser::Error> {
|
fn send_request(&self,
|
||||||
|
t: Type,
|
||||||
|
body: &ser::Writeable,
|
||||||
|
expect_resp: Option<(Type, Hash)>)
|
||||||
|
-> Result<(), ser::Error> {
|
||||||
let sent = self.send_msg(t, body);
|
let sent = self.send_msg(t, body);
|
||||||
|
|
||||||
if let Err(e) = sent {
|
if let Err(e) = sent {
|
||||||
|
@ -110,24 +113,28 @@ fn handle_payload(adapter: &NetAdapter,
|
||||||
sender: UnboundedSender<Vec<u8>>,
|
sender: UnboundedSender<Vec<u8>>,
|
||||||
header: MsgHeader,
|
header: MsgHeader,
|
||||||
buf: Vec<u8>)
|
buf: Vec<u8>)
|
||||||
-> Result<(), ser::Error> {
|
-> Result<Option<Hash>, ser::Error> {
|
||||||
match header.msg_type {
|
match header.msg_type {
|
||||||
Type::Ping => {
|
Type::Ping => {
|
||||||
let data = try!(ser::ser_vec(&MsgHeader::new(Type::Pong, 0)));
|
let data = try!(ser::ser_vec(&MsgHeader::new(Type::Pong, 0)));
|
||||||
sender.send(data);
|
sender.send(data);
|
||||||
|
Ok(None)
|
||||||
}
|
}
|
||||||
Type::Pong => {}
|
Type::Pong => Ok(None),
|
||||||
Type::Transaction => {
|
Type::Transaction => {
|
||||||
let tx = try!(ser::deserialize::<core::Transaction>(&mut &buf[..]));
|
let tx = try!(ser::deserialize::<core::Transaction>(&mut &buf[..]));
|
||||||
adapter.transaction_received(tx);
|
adapter.transaction_received(tx);
|
||||||
|
Ok(None)
|
||||||
}
|
}
|
||||||
Type::Block => {
|
Type::Block => {
|
||||||
let b = try!(ser::deserialize::<core::Block>(&mut &buf[..]));
|
let b = try!(ser::deserialize::<core::Block>(&mut &buf[..]));
|
||||||
|
let bh = b.hash();
|
||||||
adapter.block_received(b);
|
adapter.block_received(b);
|
||||||
|
Ok(Some(bh))
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
debug!("unknown message type {:?}", header.msg_type);
|
debug!("unknown message type {:?}", header.msg_type);
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
|
||||||
Ok(())
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue