2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2018-02-02 05:03:12 +03:00
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2018-02-10 01:32:16 +03:00
|
|
|
use std::fs::File;
|
2018-03-04 03:19:54 +03:00
|
|
|
use std::net::{Shutdown, SocketAddr, TcpListener, TcpStream};
|
2018-10-09 10:27:34 +03:00
|
|
|
use std::sync::Arc;
|
2018-02-02 05:03:12 +03:00
|
|
|
use std::time::Duration;
|
2018-06-14 15:16:14 +03:00
|
|
|
use std::{io, thread};
|
2018-02-02 05:03:12 +03:00
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::lmdb;
|
2018-06-22 11:08:06 +03:00
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::core::core;
|
|
|
|
use crate::core::core::hash::Hash;
|
2018-12-31 02:15:58 +03:00
|
|
|
use crate::core::global;
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::core::pow::Difficulty;
|
|
|
|
use crate::handshake::Handshake;
|
|
|
|
use crate::peer::Peer;
|
|
|
|
use crate::peers::Peers;
|
|
|
|
use crate::store::PeerStore;
|
|
|
|
use crate::types::{Capabilities, ChainAdapter, Error, NetAdapter, P2PConfig, TxHashSetRead};
|
2018-12-11 14:07:41 +03:00
|
|
|
use crate::util::{Mutex, StopState};
|
2018-10-13 01:53:50 +03:00
|
|
|
use chrono::prelude::{DateTime, Utc};
|
2018-02-02 05:03:12 +03:00
|
|
|
|
|
|
|
/// P2P server implementation, handling bootstrapping to find and connect to
|
|
|
|
/// peers, receiving connections from other peers and keep track of all of them.
|
|
|
|
pub struct Server {
|
2018-03-27 20:17:01 +03:00
|
|
|
pub config: P2PConfig,
|
2018-02-02 05:03:12 +03:00
|
|
|
capabilities: Capabilities,
|
|
|
|
handshake: Arc<Handshake>,
|
2018-02-13 03:38:52 +03:00
|
|
|
pub peers: Arc<Peers>,
|
2018-12-11 14:07:41 +03:00
|
|
|
stop_state: Arc<Mutex<StopState>>,
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO TLS
|
|
|
|
impl Server {
|
|
|
|
/// Creates a new idle p2p server with no peers
|
|
|
|
pub fn new(
|
2018-06-22 11:08:06 +03:00
|
|
|
db_env: Arc<lmdb::Environment>,
|
2018-10-23 15:01:19 +03:00
|
|
|
capab: Capabilities,
|
2018-02-02 05:03:12 +03:00
|
|
|
config: P2PConfig,
|
2018-12-08 02:59:40 +03:00
|
|
|
adapter: Arc<dyn ChainAdapter>,
|
2018-02-02 05:03:12 +03:00
|
|
|
genesis: Hash,
|
2018-12-11 14:07:41 +03:00
|
|
|
stop_state: Arc<Mutex<StopState>>,
|
2018-02-02 05:03:12 +03:00
|
|
|
) -> Result<Server, Error> {
|
|
|
|
Ok(Server {
|
|
|
|
config: config.clone(),
|
|
|
|
capabilities: capab,
|
|
|
|
handshake: Arc::new(Handshake::new(genesis, config.clone())),
|
2018-06-22 11:08:06 +03:00
|
|
|
peers: Arc::new(Peers::new(PeerStore::new(db_env)?, adapter, config)),
|
2018-12-11 14:07:41 +03:00
|
|
|
stop_state,
|
2018-02-02 05:03:12 +03:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2018-02-05 22:52:11 +03:00
|
|
|
/// Starts a new TCP server and listen to incoming connections. This is a
|
|
|
|
/// blocking call until the TCP server stops.
|
2018-02-02 05:03:12 +03:00
|
|
|
pub fn listen(&self) -> Result<(), Error> {
|
2018-02-05 22:52:11 +03:00
|
|
|
// start TCP listener and handle incoming connections
|
2018-02-02 05:03:12 +03:00
|
|
|
let addr = SocketAddr::new(self.config.host, self.config.port);
|
|
|
|
let listener = TcpListener::bind(addr)?;
|
2018-02-13 03:38:52 +03:00
|
|
|
listener.set_nonblocking(true)?;
|
2018-02-02 05:03:12 +03:00
|
|
|
|
2018-02-13 03:38:52 +03:00
|
|
|
let sleep_time = Duration::from_millis(1);
|
|
|
|
loop {
|
2018-11-10 06:27:52 +03:00
|
|
|
// Pause peer ingress connection request. Only for tests.
|
2018-12-11 14:07:41 +03:00
|
|
|
if self.stop_state.lock().is_paused() {
|
2018-11-10 06:27:52 +03:00
|
|
|
thread::sleep(Duration::from_secs(1));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-02-13 03:38:52 +03:00
|
|
|
match listener.accept() {
|
|
|
|
Ok((stream, peer_addr)) => {
|
2018-02-02 05:03:12 +03:00
|
|
|
if !self.check_banned(&stream) {
|
|
|
|
if let Err(e) = self.handle_new_peer(stream) {
|
2018-10-21 23:30:56 +03:00
|
|
|
warn!("Error accepting peer {}: {:?}", peer_addr.to_string(), e);
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-02-13 03:38:52 +03:00
|
|
|
Err(ref e) if e.kind() == io::ErrorKind::WouldBlock => {
|
|
|
|
// nothing to do, will retry in next iteration
|
|
|
|
}
|
2018-02-02 05:03:12 +03:00
|
|
|
Err(e) => {
|
2018-10-21 23:30:56 +03:00
|
|
|
warn!("Couldn't establish new client connection: {:?}", e);
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
}
|
2018-12-11 14:07:41 +03:00
|
|
|
if self.stop_state.lock().is_stopped() {
|
2018-02-13 03:38:52 +03:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
thread::sleep(sleep_time);
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Asks the server to connect to a new peer. Directly returns the peer if
|
|
|
|
/// we're already connected to the provided address.
|
2018-10-09 10:27:34 +03:00
|
|
|
pub fn connect(&self, addr: &SocketAddr) -> Result<Arc<Peer>, Error> {
|
2018-02-02 05:03:12 +03:00
|
|
|
if Peer::is_denied(&self.config, &addr) {
|
2018-10-21 23:30:56 +03:00
|
|
|
debug!("connect_peer: peer {} denied, not connecting.", addr);
|
2018-02-02 05:03:12 +03:00
|
|
|
return Err(Error::ConnectionClose);
|
|
|
|
}
|
|
|
|
|
2018-12-31 02:15:58 +03:00
|
|
|
if global::is_production_mode() {
|
|
|
|
let hs = self.handshake.clone();
|
|
|
|
let addrs = hs.addrs.read();
|
|
|
|
if addrs.contains(&addr) {
|
2018-12-31 14:24:30 +03:00
|
|
|
debug!("connect: ignore connecting to PeerWithSelf, addr: {}", addr);
|
2018-12-31 02:15:58 +03:00
|
|
|
return Err(Error::PeerWithSelf);
|
|
|
|
}
|
2018-09-04 11:52:11 +03:00
|
|
|
}
|
|
|
|
|
2018-02-02 05:03:12 +03:00
|
|
|
if let Some(p) = self.peers.get_connected_peer(addr) {
|
|
|
|
// if we're already connected to the addr, just return the peer
|
2018-10-21 23:30:56 +03:00
|
|
|
trace!("connect_peer: already connected {}", addr);
|
2018-02-02 05:03:12 +03:00
|
|
|
return Ok(p);
|
|
|
|
}
|
|
|
|
|
2018-09-04 11:52:11 +03:00
|
|
|
trace!(
|
|
|
|
"connect_peer: on {}:{}. connecting to {}",
|
|
|
|
self.config.host,
|
|
|
|
self.config.port,
|
|
|
|
addr
|
|
|
|
);
|
2018-02-02 05:03:12 +03:00
|
|
|
match TcpStream::connect_timeout(addr, Duration::from_secs(10)) {
|
|
|
|
Ok(mut stream) => {
|
|
|
|
let addr = SocketAddr::new(self.config.host, self.config.port);
|
|
|
|
let total_diff = self.peers.total_difficulty();
|
|
|
|
|
2018-10-09 10:27:34 +03:00
|
|
|
let mut peer = Peer::connect(
|
2018-02-02 05:03:12 +03:00
|
|
|
&mut stream,
|
|
|
|
self.capabilities,
|
|
|
|
total_diff,
|
|
|
|
addr,
|
|
|
|
&self.handshake,
|
2018-02-13 03:38:52 +03:00
|
|
|
self.peers.clone(),
|
2018-10-09 10:27:34 +03:00
|
|
|
)?;
|
|
|
|
peer.start(stream);
|
|
|
|
let peer = Arc::new(peer);
|
2018-10-02 17:17:29 +03:00
|
|
|
self.peers.add_connected(peer.clone())?;
|
|
|
|
Ok(peer)
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
Err(e) => {
|
2018-11-11 02:30:14 +03:00
|
|
|
trace!(
|
2018-09-04 11:52:11 +03:00
|
|
|
"connect_peer: on {}:{}. Could not connect to {}: {:?}",
|
2018-11-11 02:30:14 +03:00
|
|
|
self.config.host,
|
|
|
|
self.config.port,
|
|
|
|
addr,
|
|
|
|
e
|
2018-09-04 11:52:11 +03:00
|
|
|
);
|
2018-02-02 05:03:12 +03:00
|
|
|
Err(Error::Connection(e))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn handle_new_peer(&self, mut stream: TcpStream) -> Result<(), Error> {
|
|
|
|
let total_diff = self.peers.total_difficulty();
|
|
|
|
|
|
|
|
// accept the peer and add it to the server map
|
2018-10-09 10:27:34 +03:00
|
|
|
let mut peer = Peer::accept(
|
2018-02-02 05:03:12 +03:00
|
|
|
&mut stream,
|
|
|
|
self.capabilities,
|
|
|
|
total_diff,
|
|
|
|
&self.handshake,
|
2018-02-13 03:38:52 +03:00
|
|
|
self.peers.clone(),
|
2018-10-09 10:27:34 +03:00
|
|
|
)?;
|
|
|
|
peer.start(stream);
|
|
|
|
self.peers.add_connected(Arc::new(peer))?;
|
2018-02-02 05:03:12 +03:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_banned(&self, stream: &TcpStream) -> bool {
|
|
|
|
// peer has been banned, go away!
|
|
|
|
if let Ok(peer_addr) = stream.peer_addr() {
|
|
|
|
if self.peers.is_banned(peer_addr) {
|
2018-10-21 23:30:56 +03:00
|
|
|
debug!("Peer {} banned, refusing connection.", peer_addr);
|
2018-02-02 05:03:12 +03:00
|
|
|
if let Err(e) = stream.shutdown(Shutdown::Both) {
|
2018-10-21 23:30:56 +03:00
|
|
|
debug!("Error shutting down conn: {:?}", e);
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
false
|
|
|
|
}
|
2018-02-13 03:38:52 +03:00
|
|
|
|
|
|
|
pub fn stop(&self) {
|
2018-12-11 14:07:41 +03:00
|
|
|
self.stop_state.lock().stop();
|
2018-02-13 03:38:52 +03:00
|
|
|
self.peers.stop();
|
|
|
|
}
|
2018-11-10 06:27:52 +03:00
|
|
|
|
|
|
|
/// Pause means: stop all the current peers connection, only for tests.
|
|
|
|
/// Note:
|
|
|
|
/// 1. must pause the 'seed' thread also, to avoid the new egress peer connection
|
|
|
|
/// 2. must pause the 'p2p-server' thread also, to avoid the new ingress peer connection.
|
|
|
|
pub fn pause(&self) {
|
|
|
|
self.peers.stop();
|
|
|
|
}
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A no-op network adapter used for testing.
|
|
|
|
pub struct DummyAdapter {}
|
|
|
|
|
|
|
|
impl ChainAdapter for DummyAdapter {
|
|
|
|
fn total_difficulty(&self) -> Difficulty {
|
2018-10-18 22:18:16 +03:00
|
|
|
Difficulty::min()
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
fn total_height(&self) -> u64 {
|
|
|
|
0
|
|
|
|
}
|
2018-11-07 12:28:17 +03:00
|
|
|
fn get_transaction(&self, _h: Hash) -> Option<core::Transaction> {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
fn tx_kernel_received(&self, _h: Hash, _addr: SocketAddr) {}
|
2018-03-23 13:13:57 +03:00
|
|
|
fn transaction_received(&self, _: core::Transaction, _stem: bool) {}
|
2018-03-04 03:19:54 +03:00
|
|
|
fn compact_block_received(&self, _cb: core::CompactBlock, _addr: SocketAddr) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
fn header_received(&self, _bh: core::BlockHeader, _addr: SocketAddr) -> bool {
|
|
|
|
true
|
|
|
|
}
|
|
|
|
fn block_received(&self, _: core::Block, _: SocketAddr) -> bool {
|
|
|
|
true
|
|
|
|
}
|
2018-12-05 19:50:32 +03:00
|
|
|
fn headers_received(&self, _: &[core::BlockHeader], _: SocketAddr) -> bool {
|
2018-08-17 05:30:05 +03:00
|
|
|
true
|
|
|
|
}
|
2018-12-05 19:50:32 +03:00
|
|
|
fn locate_headers(&self, _: &[Hash]) -> Vec<core::BlockHeader> {
|
2018-02-02 05:03:12 +03:00
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
fn get_block(&self, _: Hash) -> Option<core::Block> {
|
|
|
|
None
|
|
|
|
}
|
2018-03-05 22:33:44 +03:00
|
|
|
fn txhashset_read(&self, _h: Hash) -> Option<TxHashSetRead> {
|
2018-02-10 01:32:16 +03:00
|
|
|
unimplemented!()
|
|
|
|
}
|
|
|
|
|
2018-07-12 19:06:52 +03:00
|
|
|
fn txhashset_receive_ready(&self) -> bool {
|
|
|
|
false
|
|
|
|
}
|
|
|
|
|
2018-08-01 12:44:07 +03:00
|
|
|
fn txhashset_write(&self, _h: Hash, _txhashset_data: File, _peer_addr: SocketAddr) -> bool {
|
2018-02-10 01:32:16 +03:00
|
|
|
false
|
|
|
|
}
|
2018-10-13 01:53:50 +03:00
|
|
|
|
|
|
|
fn txhashset_download_update(
|
|
|
|
&self,
|
|
|
|
_start_time: DateTime<Utc>,
|
|
|
|
_downloaded_size: u64,
|
|
|
|
_total_size: u64,
|
|
|
|
) -> bool {
|
|
|
|
false
|
|
|
|
}
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl NetAdapter for DummyAdapter {
|
|
|
|
fn find_peer_addrs(&self, _: Capabilities) -> Vec<SocketAddr> {
|
|
|
|
vec![]
|
|
|
|
}
|
|
|
|
fn peer_addrs_received(&self, _: Vec<SocketAddr>) {}
|
2018-03-04 03:19:54 +03:00
|
|
|
fn peer_difficulty(&self, _: SocketAddr, _: Difficulty, _: u64) {}
|
2018-03-27 19:09:41 +03:00
|
|
|
fn is_banned(&self, _: SocketAddr) -> bool {
|
|
|
|
false
|
|
|
|
}
|
2018-02-02 05:03:12 +03:00
|
|
|
}
|