Destroy socket listener on error rather than trying to reconnect; add #derivings
Reconnecting an existing socket simply was not working; the Rust socket did not expose any methods for reconnection, so I simply tried calling connect() again. As near as I can tell, this was a no-op --- which makes sense because both the sending and receiving threads had their own copy of the Socket, and it's not clear what the synchronization behaviour should have been. Instead if the connection fails, we relay this information to the main thread, wait for an acknowledgement, then simply destroy the listening thread. The caller can then simply call `start()` again.
This commit is contained in:
		
							parent
							
								
									1f41a67194
								
							
						
					
					
						commit
						ef11e8273b
					
				|  | @ -31,7 +31,7 @@ use blockdata::transaction::Transaction; | ||||||
| 
 | 
 | ||||||
| /// A block header, which contains all the block's information except
 | /// A block header, which contains all the block's information except
 | ||||||
| /// the actual transactions
 | /// the actual transactions
 | ||||||
| #[deriving(PartialEq, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct BlockHeader { | pub struct BlockHeader { | ||||||
|   /// The protocol version. Should always be 1.
 |   /// The protocol version. Should always be 1.
 | ||||||
|   pub version: u32, |   pub version: u32, | ||||||
|  | @ -50,7 +50,7 @@ pub struct BlockHeader { | ||||||
| 
 | 
 | ||||||
| /// A Bitcoin block, which is a collection of transactions with an attached
 | /// A Bitcoin block, which is a collection of transactions with an attached
 | ||||||
| /// proof of work.
 | /// proof of work.
 | ||||||
| #[deriving(PartialEq, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct Block { | pub struct Block { | ||||||
|   /// The block header
 |   /// The block header
 | ||||||
|   pub header: BlockHeader, |   pub header: BlockHeader, | ||||||
|  | @ -60,7 +60,7 @@ pub struct Block { | ||||||
| 
 | 
 | ||||||
| /// A block header with txcount attached, which is given in the `headers`
 | /// A block header with txcount attached, which is given in the `headers`
 | ||||||
| /// network message.
 | /// network message.
 | ||||||
| #[deriving(PartialEq, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct LoneBlockHeader { | pub struct LoneBlockHeader { | ||||||
|   /// The actual block header
 |   /// The actual block header
 | ||||||
|   pub header: BlockHeader, |   pub header: BlockHeader, | ||||||
|  |  | ||||||
|  | @ -69,6 +69,30 @@ impl fmt::Show for Address { | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | impl Clone for Address { | ||||||
|  |   fn clone(&self) -> Address { | ||||||
|  |     unsafe { | ||||||
|  |       use std::intrinsics::copy_nonoverlapping_memory; | ||||||
|  |       use std::mem; | ||||||
|  |       let mut ret = mem::uninitialized(); | ||||||
|  |       copy_nonoverlapping_memory(&mut ret, | ||||||
|  |                                  self, | ||||||
|  |                                  mem::size_of::<Address>()); | ||||||
|  |       ret | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl PartialEq for Address { | ||||||
|  |   fn eq(&self, other: &Address) -> bool { | ||||||
|  |     self.services == other.services && | ||||||
|  |     self.address.as_slice() == other.address.as_slice() && | ||||||
|  |     self.port == other.port | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Eq for Address {} | ||||||
|  | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod test { | mod test { | ||||||
|   use super::Address; |   use super::Address; | ||||||
|  |  | ||||||
|  | @ -51,11 +51,11 @@ pub trait ConsensusDecodable<D:SimpleDecoder<E>, E> { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// A variable-length unsigned integer
 | /// A variable-length unsigned integer
 | ||||||
| #[deriving(PartialEq, Show)] | #[deriving(PartialEq, Eq, PartialOrd, Ord, Clone, Show)] | ||||||
| pub struct VarInt(pub u64); | pub struct VarInt(pub u64); | ||||||
| 
 | 
 | ||||||
| /// Data which must be preceded by a 4-byte checksum
 | /// Data which must be preceded by a 4-byte checksum
 | ||||||
| #[deriving(PartialEq, Clone, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct CheckedData(pub Vec<u8>); | pub struct CheckedData(pub Vec<u8>); | ||||||
| 
 | 
 | ||||||
| // Primitive types
 | // Primitive types
 | ||||||
|  |  | ||||||
|  | @ -19,11 +19,9 @@ | ||||||
| //!
 | //!
 | ||||||
| 
 | 
 | ||||||
| use std::io::{IoResult, standard_error, ConnectionFailed}; | use std::io::{IoResult, standard_error, ConnectionFailed}; | ||||||
| use std::io::timer; |  | ||||||
| use std::time::Duration; |  | ||||||
| 
 | 
 | ||||||
| use network::constants::Network; | use network::constants::Network; | ||||||
| use network::message::{NetworkMessage, Verack}; | use network::message::{mod, SocketResponse, MessageReceived, Verack}; | ||||||
| use network::socket::Socket; | use network::socket::Socket; | ||||||
| 
 | 
 | ||||||
| /// A message which can be sent on the Bitcoin network
 | /// A message which can be sent on the Bitcoin network
 | ||||||
|  | @ -35,7 +33,7 @@ pub trait Listener { | ||||||
|   /// Return the network this `Listener` is operating on
 |   /// Return the network this `Listener` is operating on
 | ||||||
|   fn network(&self) -> Network; |   fn network(&self) -> Network; | ||||||
|   /// Main listen loop
 |   /// Main listen loop
 | ||||||
|   fn start(&self) -> IoResult<(Receiver<NetworkMessage>, Socket)> { |   fn start(&self) -> IoResult<(Receiver<SocketResponse>, Socket)> { | ||||||
|     // Open socket
 |     // Open socket
 | ||||||
|     let mut ret_sock = Socket::new(self.network()); |     let mut ret_sock = Socket::new(self.network()); | ||||||
|     match ret_sock.connect(self.peer(), self.port()) { |     match ret_sock.connect(self.peer(), self.port()) { | ||||||
|  | @ -75,20 +73,17 @@ pub trait Listener { | ||||||
|             // We have to pass the message to the main thread for processing,
 |             // We have to pass the message to the main thread for processing,
 | ||||||
|             // unfortunately, because sipa says we have to handle everything
 |             // unfortunately, because sipa says we have to handle everything
 | ||||||
|             // in order.
 |             // in order.
 | ||||||
|             recv_tx.send(payload); |             recv_tx.send(MessageReceived(payload)); | ||||||
|           } |           } | ||||||
|           Err(e) => { |           Err(e) => { | ||||||
|             println!("Received error {:} when decoding message. Pausing for 5 seconds then reconnecting.", e); |             // On failure we send an error message to the main thread, along with
 | ||||||
|             timer::sleep(Duration::seconds(5)); |             // a channel to receive an acknowledgement that we may tear down this
 | ||||||
|             // Reconnect
 |             // thread. (If we simply exited immediately, the channel would be torn
 | ||||||
|             sock.reconnect() |             // down and the main thread would never see the error message.)
 | ||||||
|               // Create version message
 |             let (tx, rx) = channel(); | ||||||
|               .and_then(|_| sock.version_message(0)) |             recv_tx.send(message::ConnectionFailed(e, tx)); | ||||||
|               // Send it out
 |             rx.recv(); | ||||||
|               .and_then(|msg| sock.send_message(msg)) |             break; | ||||||
|               // For now, not much we can do on error
 |  | ||||||
|               .unwrap_or_else(|e| println!("Error {} when reconnecting.", e)); |  | ||||||
|             handshake_complete = false; |  | ||||||
|           } |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  | @ -34,7 +34,7 @@ use network::serialize::{serialize, RawDecoder, SimpleEncoder, SimpleDecoder}; | ||||||
| use util::misc::prepend_err; | use util::misc::prepend_err; | ||||||
| 
 | 
 | ||||||
| /// Serializer for command string
 | /// Serializer for command string
 | ||||||
| #[deriving(PartialEq, Clone, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct CommandString(pub String); | pub struct CommandString(pub String); | ||||||
| 
 | 
 | ||||||
| impl<S:SimpleEncoder<E>, E> ConsensusEncodable<S, E> for CommandString { | impl<S:SimpleEncoder<E>, E> ConsensusEncodable<S, E> for CommandString { | ||||||
|  | @ -64,10 +64,18 @@ pub struct RawNetworkMessage { | ||||||
|   pub payload: NetworkMessage |   pub payload: NetworkMessage | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[deriving(Show)] | /// A response from the peer-connected socket
 | ||||||
|  | pub enum SocketResponse { | ||||||
|  |   /// A message was received
 | ||||||
|  |   MessageReceived(NetworkMessage), | ||||||
|  |   /// An error occured and the socket needs to close
 | ||||||
|  |   ConnectionFailed(IoError, Sender<()>) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[deriving(Clone, PartialEq, Eq, Show)] | ||||||
| /// A Network message payload. Proper documentation is available on the Bitcoin
 | /// A Network message payload. Proper documentation is available on the Bitcoin
 | ||||||
| /// wiki https://en.bitcoin.it/wiki/Protocol_specification
 | /// wiki https://en.bitcoin.it/wiki/Protocol_specification
 | ||||||
| pub enum NetworkMessage{ | pub enum NetworkMessage { | ||||||
|   /// `version`
 |   /// `version`
 | ||||||
|   Version(message_network::VersionMessage), |   Version(message_network::VersionMessage), | ||||||
|   /// `verack`
 |   /// `verack`
 | ||||||
|  | @ -98,7 +106,7 @@ pub enum NetworkMessage{ | ||||||
|   /// `ping`
 |   /// `ping`
 | ||||||
|   Ping(u64), |   Ping(u64), | ||||||
|   /// `pong`
 |   /// `pong`
 | ||||||
|   Pong(u64), |   Pong(u64) | ||||||
|   // TODO: reject,
 |   // TODO: reject,
 | ||||||
|   // TODO: bloom filtering
 |   // TODO: bloom filtering
 | ||||||
|   // TODO: alert
 |   // TODO: alert
 | ||||||
|  | @ -119,7 +127,7 @@ impl RawNetworkMessage { | ||||||
|       Block(_)   => "block", |       Block(_)   => "block", | ||||||
|       Headers(_) => "headers", |       Headers(_) => "headers", | ||||||
|       Ping(_)    => "ping", |       Ping(_)    => "ping", | ||||||
|       Pong(_)    => "pong" |       Pong(_)    => "pong", | ||||||
|     }.to_string() |     }.to_string() | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -23,7 +23,7 @@ use network::encodable::{ConsensusDecodable, ConsensusEncodable}; | ||||||
| use network::serialize::{SimpleDecoder, SimpleEncoder}; | use network::serialize::{SimpleDecoder, SimpleEncoder}; | ||||||
| use util::hash::Sha256dHash; | use util::hash::Sha256dHash; | ||||||
| 
 | 
 | ||||||
| #[deriving(Clone, PartialEq, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| /// The type of an inventory object
 | /// The type of an inventory object
 | ||||||
| pub enum InvType { | pub enum InvType { | ||||||
|   /// Error --- these inventories can be ignored
 |   /// Error --- these inventories can be ignored
 | ||||||
|  | @ -37,7 +37,7 @@ pub enum InvType { | ||||||
| // Some simple messages
 | // Some simple messages
 | ||||||
| 
 | 
 | ||||||
| /// The `getblocks` message
 | /// The `getblocks` message
 | ||||||
| #[deriving(Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct GetBlocksMessage { | pub struct GetBlocksMessage { | ||||||
|   /// The protocol version
 |   /// The protocol version
 | ||||||
|   pub version: u32, |   pub version: u32, | ||||||
|  | @ -50,7 +50,7 @@ pub struct GetBlocksMessage { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// The `getheaders` message
 | /// The `getheaders` message
 | ||||||
| #[deriving(Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct GetHeadersMessage { | pub struct GetHeadersMessage { | ||||||
|   /// The protocol version
 |   /// The protocol version
 | ||||||
|   pub version: u32, |   pub version: u32, | ||||||
|  | @ -63,7 +63,7 @@ pub struct GetHeadersMessage { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// An inventory object --- a reference to a Bitcoin object
 | /// An inventory object --- a reference to a Bitcoin object
 | ||||||
| #[deriving(Clone, Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct Inventory { | pub struct Inventory { | ||||||
|   /// The type of object that is referenced
 |   /// The type of object that is referenced
 | ||||||
|   pub inv_type: InvType, |   pub inv_type: InvType, | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ use network::socket::Socket; | ||||||
| /// Some simple messages
 | /// Some simple messages
 | ||||||
| 
 | 
 | ||||||
| /// The `version` message
 | /// The `version` message
 | ||||||
| #[deriving(Show)] | #[deriving(PartialEq, Eq, Clone, Show)] | ||||||
| pub struct VersionMessage { | pub struct VersionMessage { | ||||||
|   /// The P2P network protocol version
 |   /// The P2P network protocol version
 | ||||||
|   pub version: u32, |   pub version: u32, | ||||||
|  |  | ||||||
|  | @ -105,18 +105,6 @@ impl Socket { | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   /// Reset the connection to the peer
 |  | ||||||
|   pub fn reconnect(&mut self) -> IoResult<()> { |  | ||||||
|     let (host, port) = match self.socket { |  | ||||||
|       Some(ref mut s) => match s.peer_name() { |  | ||||||
|         Ok(addr) => (format!("{}", addr.ip), addr.port), |  | ||||||
|         Err(e) => { return Err(e); } |  | ||||||
|       }, |  | ||||||
|       None => { return Err(standard_error(NotConnected)); } |  | ||||||
|     }; |  | ||||||
|     self.connect(host.as_slice(), port) |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /// Peer address
 |   /// Peer address
 | ||||||
|   pub fn receiver_address(&mut self) -> IoResult<Address> { |   pub fn receiver_address(&mut self) -> IoResult<Address> { | ||||||
|     match self.socket { |     match self.socket { | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue