// Rust Bitcoin Library // Written in 2014 by // Andrew Poelstra // // To the extent possible under law, the author(s) have dedicated all // copyright and related and neighboring rights to this software to // the public domain worldwide. This software is distributed without // any warranty. // // You should have received a copy of the CC0 Public Domain Dedication // along with this software. // If not, see . // //! # Sockets //! //! This module provides support for low-level network communication. //! use time::now; use rand::{thread_rng, Rng}; use std::io::{self, Cursor, Write}; use std::net::{ip, tcp}; use std::sync::{Arc, Mutex}; use network::constants; use network::address::Address; use network::encodable::{ConsensusEncodable, ConsensusDecodable}; use network::message::{RawNetworkMessage, NetworkMessage}; use network::message::NetworkMessage::Version; use network::message_network::VersionMessage; use network::serialize::{RawEncoder, RawDecoder}; use util::{self, propagate_err}; /// Format an IP address in the 16-byte bitcoin protocol serialization fn ipaddr_to_bitcoin_addr(ipaddr: &ip::IpAddr) -> [u16; 8] { match *ipaddr { ip::IpAddr::V4(ref addr) => &addr.to_ipv6_mapped(), ip::IpAddr::V6(ref addr) => addr }.segments() } /// A network socket along with information about the peer #[derive(Clone)] pub struct Socket { /// The underlying TCP socket socket: Arc>>, /// Services supported by us pub services: u64, /// Our user agent pub user_agent: String, /// Nonce to identify our `version` messages pub version_nonce: u64, /// Network magic pub magic: u32 } impl Socket { // TODO: we fix services to 0 /// Construct a new socket pub fn new(network: constants::Network) -> Socket { let mut rng = thread_rng(); Socket { socket: Arc::new(Mutex::new(None)), services: 0, version_nonce: rng.gen(), user_agent: String::from_str(constants::USER_AGENT), magic: constants::magic(network) } } /// (Re)connect to the peer pub fn connect(&mut self, host: &str, port: u16) -> Result<(), util::Error> { // Entirely replace the Mutex, in case it was poisoned; // this will also drop any preexisting socket that might be open match tcp::TcpStream::connect((host, port)) { Ok(s) => { self.socket = Arc::new(Mutex::new(Some(s))); Ok(()) } Err(e) => { self.socket = Arc::new(Mutex::new(None)); Err(util::Error::Io(e)) } } } fn socket(&mut self) -> Result<&mut tcp::TcpStream, util::Error> { let mut sock_lock = self.socket.lock(); match sock_lock { Err(_) => { let io_err = io::Error::new(io::ErrorKind::NotConnected, "socket: socket mutex was poisoned"); Err(util::Error::Io(io_err)) } Ok(guard) => { match *guard { Some(ref mut sock) => Ok(sock), None => { let io_err = io::Error::new(io::ErrorKind::NotConnected, "socket: not connected to peer"); Err(util::Error::Io(io_err)) } } } } } /// Peer address pub fn receiver_address(&mut self) -> Result { let sock = try!(self.socket()); match sock.peer_addr() { Ok(addr) => { Ok(Address { services: self.services, address: ipaddr_to_bitcoin_addr(&addr.ip()), port: addr.port() }) }, Err(e) => Err(util::Error::Io(e)) } } /// Our own address pub fn sender_address(&mut self) -> Result { let sock = try!(self.socket()); match sock.local_addr() { Ok(addr) => { Ok(Address { services: self.services, address: ipaddr_to_bitcoin_addr(&addr.ip()), port: addr.port() }) }, Err(e) => Err(util::Error::Io(e)) } } /// Produce a version message appropriate for this socket pub fn version_message(&mut self, start_height: i32) -> Result { let recv_addr = try!(self.receiver_address()); let send_addr = try!(self.sender_address()); let timestamp = now().to_timespec().sec; Ok(Version(VersionMessage { version: constants::PROTOCOL_VERSION, services: constants::SERVICES, timestamp: timestamp, receiver: recv_addr, sender: send_addr, nonce: self.version_nonce, user_agent: self.user_agent.clone(), start_height: start_height, relay: false })) } /// Send a general message across the line pub fn send_message(&mut self, payload: NetworkMessage) -> Result<(), util::Error> { let sock = try!(self.socket()); let message = RawNetworkMessage { magic: self.magic, payload: payload }; try!(message.consensus_encode(&mut RawEncoder::new(sock))); sock.flush().map_err(util::Error::Io) } /// Receive the next message from the peer, decoding the network header /// and verifying its correctness. Returns the undecoded payload. pub fn receive_message(&mut self) -> Result { let sock = try!(self.socket()); // We need a new scope since the closure in here borrows read_err, // and we try to read it afterward. Letting `iter` go out fixes it. let mut decoder = RawDecoder::new(sock); let decode: Result = ConsensusDecodable::consensus_decode(&mut decoder); match decode { // Check for parse errors... Err(e) => { propagate_err("receive_message".to_string(), Err(e)) }, Ok(ret) => { // Then for magic (this should come before parse error, but we can't // get to it if the deserialization failed). TODO restructure this if ret.magic != self.magic { Err(util::Error::BadNetworkMagic(self.magic, ret.magic)) } else { Ok(ret.payload) } } } } }