2014-07-18 13:56:17 +00:00
|
|
|
// Rust Bitcoin Library
|
|
|
|
// Written in 2014 by
|
|
|
|
// Andrew Poelstra <apoelstra@wpsoftware.net>
|
|
|
|
//
|
|
|
|
// 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 <http://creativecommons.org/publicdomain/zero/1.0/>.
|
|
|
|
//
|
|
|
|
|
|
|
|
//! # Sockets
|
|
|
|
//!
|
|
|
|
//! This module provides support for low-level network communication.
|
|
|
|
//!
|
|
|
|
|
|
|
|
use time::now;
|
|
|
|
use std::rand::task_rng;
|
|
|
|
use rand::Rng;
|
2014-07-24 15:27:40 +00:00
|
|
|
use std::io::{BufferedReader, BufferedWriter};
|
2014-07-18 13:56:17 +00:00
|
|
|
use std::io::{IoError, IoResult, NotConnected, OtherIoError, standard_error};
|
|
|
|
use std::io::net::{ip, tcp};
|
2014-07-24 15:27:40 +00:00
|
|
|
use std::sync::{Arc, Mutex};
|
2014-07-18 13:56:17 +00:00
|
|
|
|
|
|
|
use network::constants;
|
|
|
|
use network::address::Address;
|
2014-08-01 16:01:39 +00:00
|
|
|
use network::encodable::{ConsensusEncodable, ConsensusDecodable};
|
2014-07-18 13:56:17 +00:00
|
|
|
use network::message::{RawNetworkMessage, NetworkMessage, Version};
|
|
|
|
use network::message_network::VersionMessage;
|
2014-08-01 16:01:39 +00:00
|
|
|
use network::serialize::{RawEncoder, RawDecoder};
|
2014-07-23 18:22:35 +00:00
|
|
|
use util::misc::prepend_err;
|
2014-07-18 13:56:17 +00:00
|
|
|
|
|
|
|
/// Format an IP address in the 16-byte bitcoin protocol serialization
|
|
|
|
fn ipaddr_to_bitcoin_addr(ipaddr: &ip::IpAddr) -> [u8, ..16] {
|
|
|
|
match *ipaddr {
|
|
|
|
ip::Ipv4Addr(a, b, c, d) =>
|
|
|
|
[0, 0, 0, 0, 0, 0, 0, 0,
|
|
|
|
0, 0, 0xff, 0xff, a, b, c, d],
|
|
|
|
ip::Ipv6Addr(a, b, c, d, e, f, g, h) =>
|
|
|
|
[(a / 0x100) as u8, (a % 0x100) as u8, (b / 0x100) as u8, (b % 0x100) as u8,
|
|
|
|
(c / 0x100) as u8, (c % 0x100) as u8, (d / 0x100) as u8, (d % 0x100) as u8,
|
|
|
|
(e / 0x100) as u8, (e % 0x100) as u8, (f / 0x100) as u8, (f % 0x100) as u8,
|
|
|
|
(g / 0x100) as u8, (g % 0x100) as u8, (h / 0x100) as u8, (h % 0x100) as u8 ]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A network socket along with information about the peer
|
|
|
|
#[deriving(Clone)]
|
|
|
|
pub struct Socket {
|
2014-07-24 15:27:40 +00:00
|
|
|
/// The underlying socket, which is only used directly to (a) get
|
|
|
|
/// information about the socket, and (b) to close down the socket,
|
|
|
|
/// quickly cancelling any read/writes and unlocking the Mutexes.
|
|
|
|
socket: Option<tcp::TcpStream>,
|
|
|
|
/// The underlying network data stream read buffer
|
|
|
|
buffered_reader: Arc<Mutex<Option<BufferedReader<tcp::TcpStream>>>>,
|
|
|
|
/// The underlying network data stream write buffer
|
|
|
|
buffered_writer: Arc<Mutex<Option<BufferedWriter<tcp::TcpStream>>>>,
|
2014-07-18 13:56:17 +00:00
|
|
|
/// 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
|
2014-07-18 21:38:35 +00:00
|
|
|
pub fn new(network: constants::Network) -> Socket {
|
2014-07-18 13:56:17 +00:00
|
|
|
let mut rng = task_rng();
|
|
|
|
Socket {
|
2014-07-24 15:27:40 +00:00
|
|
|
socket: None,
|
|
|
|
buffered_reader: Arc::new(Mutex::new(None)),
|
|
|
|
buffered_writer: Arc::new(Mutex::new(None)),
|
2014-07-18 13:56:17 +00:00
|
|
|
services: 0,
|
|
|
|
version_nonce: rng.gen(),
|
|
|
|
user_agent: String::from_str(constants::USER_AGENT),
|
2014-07-18 21:38:35 +00:00
|
|
|
magic: constants::magic(network)
|
2014-07-18 13:56:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Connect to the peer
|
|
|
|
pub fn connect(&mut self, host: &str, port: u16) -> IoResult<()> {
|
2014-07-24 15:27:40 +00:00
|
|
|
// Boot off any lingering readers or writers
|
|
|
|
if self.socket.is_some() {
|
|
|
|
let _ = self.socket.get_mut_ref().close_read();
|
|
|
|
let _ = self.socket.get_mut_ref().close_write();
|
|
|
|
}
|
|
|
|
// These locks should just pop open now
|
|
|
|
let mut reader_lock = self.buffered_reader.lock();
|
|
|
|
let mut writer_lock = self.buffered_writer.lock();
|
2014-07-18 13:56:17 +00:00
|
|
|
match tcp::TcpStream::connect(host, port) {
|
|
|
|
Ok(s) => {
|
2014-07-24 15:27:40 +00:00
|
|
|
*reader_lock = Some(BufferedReader::new(s.clone()));
|
|
|
|
*writer_lock = Some(BufferedWriter::new(s.clone()));
|
|
|
|
self.socket = Some(s);
|
2014-07-18 13:56:17 +00:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
Err(e) => Err(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-24 15:27:40 +00:00
|
|
|
/// 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)
|
|
|
|
}
|
|
|
|
|
2014-07-18 13:56:17 +00:00
|
|
|
/// Peer address
|
|
|
|
pub fn receiver_address(&mut self) -> IoResult<Address> {
|
2014-07-24 15:27:40 +00:00
|
|
|
match self.socket {
|
2014-07-18 13:56:17 +00:00
|
|
|
Some(ref mut s) => match s.peer_name() {
|
|
|
|
Ok(addr) => {
|
|
|
|
Ok(Address {
|
|
|
|
services: self.services,
|
|
|
|
address: ipaddr_to_bitcoin_addr(&addr.ip),
|
|
|
|
port: addr.port
|
|
|
|
})
|
|
|
|
}
|
|
|
|
Err(e) => Err(e)
|
|
|
|
},
|
|
|
|
None => Err(standard_error(NotConnected))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Our own address
|
|
|
|
pub fn sender_address(&mut self) -> IoResult<Address> {
|
2014-07-24 15:27:40 +00:00
|
|
|
match self.socket {
|
2014-07-18 13:56:17 +00:00
|
|
|
Some(ref mut s) => match s.socket_name() {
|
|
|
|
Ok(addr) => {
|
|
|
|
Ok(Address {
|
|
|
|
services: self.services,
|
|
|
|
address: ipaddr_to_bitcoin_addr(&addr.ip),
|
|
|
|
port: addr.port
|
|
|
|
})
|
|
|
|
}
|
|
|
|
Err(e) => Err(e)
|
|
|
|
},
|
|
|
|
None => Err(standard_error(NotConnected))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Produce a version message appropriate for this socket
|
|
|
|
pub fn version_message(&mut self, start_height: i32) -> IoResult<NetworkMessage> {
|
|
|
|
let timestamp = now().to_timespec().sec;
|
|
|
|
let recv_addr = self.receiver_address();
|
|
|
|
let send_addr = self.sender_address();
|
|
|
|
// If we are not connected, we might not be able to get these address.s
|
|
|
|
match recv_addr {
|
|
|
|
Err(e) => { return Err(e); }
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
match send_addr {
|
|
|
|
Err(e) => { return Err(e); }
|
|
|
|
_ => {}
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(Version(VersionMessage {
|
|
|
|
version: constants::PROTOCOL_VERSION,
|
|
|
|
services: constants::SERVICES,
|
|
|
|
timestamp: timestamp,
|
|
|
|
receiver: recv_addr.unwrap(),
|
|
|
|
sender: send_addr.unwrap(),
|
|
|
|
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) -> IoResult<()> {
|
2014-07-24 15:27:40 +00:00
|
|
|
let mut writer_lock = self.buffered_writer.lock();
|
|
|
|
match *writer_lock.deref_mut() {
|
|
|
|
None => Err(standard_error(NotConnected)),
|
|
|
|
Some(ref mut writer) => {
|
|
|
|
let message = RawNetworkMessage { magic: self.magic, payload: payload };
|
2014-08-01 16:01:39 +00:00
|
|
|
try!(message.consensus_encode(&mut RawEncoder::new(writer.by_ref())));
|
2014-07-24 15:27:40 +00:00
|
|
|
writer.flush()
|
2014-07-18 13:56:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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) -> IoResult<NetworkMessage> {
|
2014-07-24 15:27:40 +00:00
|
|
|
let mut reader_lock = self.buffered_reader.lock();
|
|
|
|
match *reader_lock.deref_mut() {
|
2014-07-18 13:56:17 +00:00
|
|
|
None => Err(standard_error(NotConnected)),
|
2014-07-24 15:27:40 +00:00
|
|
|
Some(ref mut buf) => {
|
2014-07-18 13:56:17 +00:00
|
|
|
// 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.
|
2014-08-01 16:01:39 +00:00
|
|
|
let mut decoder = RawDecoder::new(buf.by_ref());
|
|
|
|
let decode: IoResult<RawNetworkMessage> = ConsensusDecodable::consensus_decode(&mut decoder);
|
|
|
|
match decode {
|
|
|
|
// Check for parse errors...
|
|
|
|
Err(e) => {
|
|
|
|
prepend_err("network_decode", Err(e))
|
2014-07-24 15:27:40 +00:00
|
|
|
},
|
2014-08-01 16:01:39 +00:00
|
|
|
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(IoError {
|
|
|
|
kind: OtherIoError,
|
|
|
|
desc: "bad magic",
|
|
|
|
detail: Some(format!("got magic {:x}, expected {:x}", ret.magic, self.magic)),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Ok(ret.payload)
|
2014-07-18 13:56:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|