2019-03-25 12:02:51 +00:00
|
|
|
use crate::behaviour::{Behaviour, BehaviourEvent, PubsubMessage};
|
2019-03-12 06:28:11 +00:00
|
|
|
use crate::error;
|
2019-03-08 01:15:57 +00:00
|
|
|
use crate::multiaddr::Protocol;
|
2019-03-17 12:14:28 +00:00
|
|
|
use crate::rpc::RPCEvent;
|
2019-03-06 12:31:08 +00:00
|
|
|
use crate::NetworkConfig;
|
2019-04-03 05:00:09 +00:00
|
|
|
use crate::{TopicBuilder, TopicHash};
|
2019-06-25 04:51:45 +00:00
|
|
|
use crate::{BEACON_ATTESTATION_TOPIC, BEACON_PUBSUB_TOPIC};
|
2019-03-07 05:17:06 +00:00
|
|
|
use futures::prelude::*;
|
2019-03-12 06:28:11 +00:00
|
|
|
use futures::Stream;
|
2019-03-07 05:17:06 +00:00
|
|
|
use libp2p::core::{
|
2019-07-01 06:38:42 +00:00
|
|
|
identity::Keypair,
|
2019-06-25 08:02:11 +00:00
|
|
|
multiaddr::Multiaddr,
|
2019-03-07 05:17:06 +00:00
|
|
|
muxing::StreamMuxerBox,
|
|
|
|
nodes::Substream,
|
|
|
|
transport::boxed::Boxed,
|
2019-03-08 00:07:30 +00:00
|
|
|
upgrade::{InboundUpgradeExt, OutboundUpgradeExt},
|
2019-03-07 05:17:06 +00:00
|
|
|
};
|
2019-03-21 01:57:41 +00:00
|
|
|
use libp2p::{core, secio, PeerId, Swarm, Transport};
|
2019-03-13 04:37:44 +00:00
|
|
|
use slog::{debug, info, trace, warn};
|
2019-07-01 06:38:42 +00:00
|
|
|
use std::fs::File;
|
|
|
|
use std::io::prelude::*;
|
2019-03-07 05:17:06 +00:00
|
|
|
use std::io::{Error, ErrorKind};
|
|
|
|
use std::time::Duration;
|
2019-03-04 07:31:01 +00:00
|
|
|
|
2019-04-03 05:23:09 +00:00
|
|
|
type Libp2pStream = Boxed<(PeerId, StreamMuxerBox), Error>;
|
|
|
|
type Libp2pBehaviour = Behaviour<Substream<StreamMuxerBox>>;
|
|
|
|
|
2019-07-01 06:38:42 +00:00
|
|
|
const NETWORK_KEY_FILENAME: &str = "key";
|
|
|
|
|
2019-03-04 07:31:01 +00:00
|
|
|
/// The configuration and state of the libp2p components for the beacon node.
|
2019-03-06 12:31:08 +00:00
|
|
|
pub struct Service {
|
2019-03-07 00:43:55 +00:00
|
|
|
/// The libp2p Swarm handler.
|
2019-03-12 06:28:11 +00:00
|
|
|
//TODO: Make this private
|
2019-04-03 05:23:09 +00:00
|
|
|
pub swarm: Swarm<Libp2pStream, Libp2pBehaviour>,
|
2019-03-06 12:31:08 +00:00
|
|
|
/// This node's PeerId.
|
2019-04-03 05:23:09 +00:00
|
|
|
_local_peer_id: PeerId,
|
2019-03-12 06:28:11 +00:00
|
|
|
/// The libp2p logger handle.
|
|
|
|
pub log: slog::Logger,
|
2019-03-06 12:31:08 +00:00
|
|
|
}
|
2019-03-04 07:31:01 +00:00
|
|
|
|
|
|
|
impl Service {
|
2019-03-12 06:28:11 +00:00
|
|
|
pub fn new(config: NetworkConfig, log: slog::Logger) -> error::Result<Self> {
|
2019-06-25 04:51:45 +00:00
|
|
|
debug!(log, "Network-libp2p Service starting");
|
2019-03-06 12:31:08 +00:00
|
|
|
|
2019-07-01 06:38:42 +00:00
|
|
|
// load the private key from CLI flag, disk or generate a new one
|
|
|
|
let local_private_key = load_private_key(&config, &log);
|
|
|
|
|
2019-03-26 04:01:05 +00:00
|
|
|
let local_peer_id = PeerId::from(local_private_key.public());
|
2019-03-13 04:37:44 +00:00
|
|
|
info!(log, "Local peer id: {:?}", local_peer_id);
|
2019-03-06 12:31:08 +00:00
|
|
|
|
2019-03-08 00:07:30 +00:00
|
|
|
let mut swarm = {
|
2019-06-25 04:51:45 +00:00
|
|
|
// Set up the transport - tcp/ws with secio and mplex/yamux
|
|
|
|
let transport = build_transport(local_private_key.clone());
|
|
|
|
// Lighthouse network behaviour
|
|
|
|
let behaviour = Behaviour::new(&local_private_key, &config, &log)?;
|
|
|
|
Swarm::new(transport, behaviour, local_peer_id.clone())
|
2019-03-08 00:07:30 +00:00
|
|
|
};
|
2019-03-06 12:31:08 +00:00
|
|
|
|
2019-06-25 08:02:11 +00:00
|
|
|
// listen on the specified address
|
|
|
|
let listen_multiaddr = {
|
|
|
|
let mut m = Multiaddr::from(config.listen_address);
|
|
|
|
m.push(Protocol::Tcp(config.libp2p_port));
|
|
|
|
m
|
|
|
|
};
|
|
|
|
|
|
|
|
match Swarm::listen_on(&mut swarm, listen_multiaddr.clone()) {
|
|
|
|
Ok(_) => {
|
|
|
|
let mut log_address = listen_multiaddr;
|
|
|
|
log_address.push(Protocol::P2p(local_peer_id.clone().into()));
|
|
|
|
info!(log, "Listening on: {}", log_address);
|
|
|
|
}
|
|
|
|
Err(err) => warn!(
|
|
|
|
log,
|
|
|
|
"Cannot listen on: {} because: {:?}", listen_multiaddr, err
|
|
|
|
),
|
|
|
|
};
|
2019-03-06 12:31:08 +00:00
|
|
|
|
2019-03-13 04:37:44 +00:00
|
|
|
// subscribe to default gossipsub topics
|
2019-04-03 05:00:09 +00:00
|
|
|
let mut topics = vec![];
|
|
|
|
//TODO: Handle multiple shard attestations. For now we simply use a separate topic for
|
|
|
|
//attestations
|
2019-06-25 04:51:45 +00:00
|
|
|
topics.push(BEACON_ATTESTATION_TOPIC.to_string());
|
2019-04-03 05:33:12 +00:00
|
|
|
topics.push(BEACON_PUBSUB_TOPIC.to_string());
|
2019-04-03 05:00:09 +00:00
|
|
|
topics.append(&mut config.topics.clone());
|
|
|
|
|
2019-03-13 04:37:44 +00:00
|
|
|
let mut subscribed_topics = vec![];
|
2019-04-03 05:00:09 +00:00
|
|
|
for topic in topics {
|
|
|
|
let t = TopicBuilder::new(topic.clone()).build();
|
2019-03-19 12:20:39 +00:00
|
|
|
if swarm.subscribe(t) {
|
|
|
|
trace!(log, "Subscribed to topic: {:?}", topic);
|
|
|
|
subscribed_topics.push(topic);
|
|
|
|
} else {
|
|
|
|
warn!(log, "Could not subscribe to topic: {:?}", topic)
|
|
|
|
}
|
2019-03-13 04:37:44 +00:00
|
|
|
}
|
|
|
|
info!(log, "Subscribed to topics: {:?}", subscribed_topics);
|
|
|
|
|
2019-03-12 06:28:11 +00:00
|
|
|
Ok(Service {
|
2019-04-03 05:23:09 +00:00
|
|
|
_local_peer_id: local_peer_id,
|
2019-03-07 00:43:55 +00:00
|
|
|
swarm,
|
2019-03-12 06:28:11 +00:00
|
|
|
log,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Stream for Service {
|
|
|
|
type Item = Libp2pEvent;
|
|
|
|
type Error = crate::error::Error;
|
|
|
|
|
|
|
|
fn poll(&mut self) -> Poll<Option<Self::Item>, Self::Error> {
|
|
|
|
loop {
|
|
|
|
match self.swarm.poll() {
|
2019-03-21 01:57:41 +00:00
|
|
|
//Behaviour events
|
|
|
|
Ok(Async::Ready(Some(event))) => match event {
|
2019-03-12 06:28:11 +00:00
|
|
|
// TODO: Stub here for debugging
|
2019-03-25 12:02:51 +00:00
|
|
|
BehaviourEvent::GossipMessage {
|
|
|
|
source,
|
|
|
|
topics,
|
|
|
|
message,
|
|
|
|
} => {
|
2019-03-31 06:26:28 +00:00
|
|
|
trace!(self.log, "Pubsub message received: {:?}", message);
|
2019-03-25 12:02:51 +00:00
|
|
|
return Ok(Async::Ready(Some(Libp2pEvent::PubsubMessage {
|
|
|
|
source,
|
|
|
|
topics,
|
|
|
|
message,
|
|
|
|
})));
|
2019-03-21 01:57:41 +00:00
|
|
|
}
|
|
|
|
BehaviourEvent::RPC(peer_id, event) => {
|
|
|
|
return Ok(Async::Ready(Some(Libp2pEvent::RPC(peer_id, event))));
|
|
|
|
}
|
|
|
|
BehaviourEvent::PeerDialed(peer_id) => {
|
|
|
|
return Ok(Async::Ready(Some(Libp2pEvent::PeerDialed(peer_id))));
|
|
|
|
}
|
2019-07-16 12:32:37 +00:00
|
|
|
BehaviourEvent::PeerDisconnected(peer_id) => {
|
|
|
|
return Ok(Async::Ready(Some(Libp2pEvent::PeerDisconnected(peer_id))));
|
|
|
|
}
|
2019-03-21 01:57:41 +00:00
|
|
|
},
|
2019-03-12 06:28:11 +00:00
|
|
|
Ok(Async::Ready(None)) => unreachable!("Swarm stream shouldn't end"),
|
|
|
|
Ok(Async::NotReady) => break,
|
|
|
|
_ => break,
|
|
|
|
}
|
2019-03-07 00:43:55 +00:00
|
|
|
}
|
2019-03-12 06:28:11 +00:00
|
|
|
Ok(Async::NotReady)
|
2019-03-04 07:31:01 +00:00
|
|
|
}
|
|
|
|
}
|
2019-03-06 12:31:08 +00:00
|
|
|
|
|
|
|
/// The implementation supports TCP/IP, WebSockets over TCP/IP, secio as the encryption layer, and
|
|
|
|
/// mplex or yamux as the multiplexing layer.
|
2019-07-01 06:38:42 +00:00
|
|
|
fn build_transport(local_private_key: Keypair) -> Boxed<(PeerId, StreamMuxerBox), Error> {
|
2019-03-06 12:31:08 +00:00
|
|
|
// TODO: The Wire protocol currently doesn't specify encryption and this will need to be customised
|
|
|
|
// in the future.
|
2019-03-07 05:17:06 +00:00
|
|
|
let transport = libp2p::tcp::TcpConfig::new();
|
|
|
|
let transport = libp2p::dns::DnsConfig::new(transport);
|
|
|
|
#[cfg(feature = "libp2p-websocket")]
|
|
|
|
let transport = {
|
|
|
|
let trans_clone = transport.clone();
|
|
|
|
transport.or_transport(websocket::WsConfig::new(trans_clone))
|
|
|
|
};
|
|
|
|
transport
|
|
|
|
.with_upgrade(secio::SecioConfig::new(local_private_key))
|
|
|
|
.and_then(move |out, endpoint| {
|
|
|
|
let peer_id = out.remote_key.into_peer_id();
|
|
|
|
let peer_id2 = peer_id.clone();
|
|
|
|
let upgrade = core::upgrade::SelectUpgrade::new(
|
|
|
|
libp2p::yamux::Config::default(),
|
|
|
|
libp2p::mplex::MplexConfig::new(),
|
|
|
|
)
|
|
|
|
// TODO: use a single `.map` instead of two maps
|
|
|
|
.map_inbound(move |muxer| (peer_id, muxer))
|
|
|
|
.map_outbound(move |muxer| (peer_id2, muxer));
|
2019-03-07 00:43:55 +00:00
|
|
|
|
2019-03-07 05:17:06 +00:00
|
|
|
core::upgrade::apply(out.stream, upgrade, endpoint)
|
|
|
|
.map(|(id, muxer)| (id, core::muxing::StreamMuxerBox::new(muxer)))
|
|
|
|
})
|
|
|
|
.with_timeout(Duration::from_secs(20))
|
|
|
|
.map_err(|err| Error::new(ErrorKind::Other, err))
|
|
|
|
.boxed()
|
2019-03-07 00:43:55 +00:00
|
|
|
}
|
2019-03-12 06:28:11 +00:00
|
|
|
|
|
|
|
/// Events that can be obtained from polling the Libp2p Service.
|
|
|
|
pub enum Libp2pEvent {
|
2019-03-21 02:15:14 +00:00
|
|
|
/// An RPC response request has been received on the swarm.
|
2019-03-19 01:47:36 +00:00
|
|
|
RPC(PeerId, RPCEvent),
|
2019-03-21 02:15:14 +00:00
|
|
|
/// Initiated the connection to a new peer.
|
2019-03-17 12:14:28 +00:00
|
|
|
PeerDialed(PeerId),
|
2019-07-16 12:32:37 +00:00
|
|
|
/// A peer has disconnected.
|
|
|
|
PeerDisconnected(PeerId),
|
2019-03-25 12:02:51 +00:00
|
|
|
/// Received pubsub message.
|
|
|
|
PubsubMessage {
|
|
|
|
source: PeerId,
|
|
|
|
topics: Vec<TopicHash>,
|
2019-04-03 05:23:09 +00:00
|
|
|
message: Box<PubsubMessage>,
|
2019-03-25 12:02:51 +00:00
|
|
|
},
|
2019-03-12 06:28:11 +00:00
|
|
|
}
|
2019-07-01 06:38:42 +00:00
|
|
|
|
|
|
|
/// Loads a private key from disk. If this fails, a new key is
|
|
|
|
/// generated and is then saved to disk.
|
|
|
|
///
|
|
|
|
/// Currently only secp256k1 keys are allowed, as these are the only keys supported by discv5.
|
|
|
|
fn load_private_key(config: &NetworkConfig, log: &slog::Logger) -> Keypair {
|
|
|
|
// TODO: Currently using secp256k1 keypairs - currently required for discv5
|
|
|
|
// check for key from disk
|
|
|
|
let network_key_f = config.network_dir.join(NETWORK_KEY_FILENAME);
|
|
|
|
if let Ok(mut network_key_file) = File::open(network_key_f.clone()) {
|
|
|
|
let mut key_bytes: Vec<u8> = Vec::with_capacity(36);
|
|
|
|
match network_key_file.read_to_end(&mut key_bytes) {
|
|
|
|
Err(_) => debug!(log, "Could not read network key file"),
|
|
|
|
Ok(_) => {
|
|
|
|
// only accept secp256k1 keys for now
|
|
|
|
if let Ok(secret_key) =
|
|
|
|
libp2p::core::identity::secp256k1::SecretKey::from_bytes(&mut key_bytes)
|
|
|
|
{
|
|
|
|
let kp: libp2p::core::identity::secp256k1::Keypair = secret_key.into();
|
|
|
|
debug!(log, "Loaded network key from disk.");
|
|
|
|
return Keypair::Secp256k1(kp);
|
|
|
|
} else {
|
|
|
|
debug!(log, "Network key file is not a valid secp256k1 key");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if a key could not be loaded from disk, generate a new one and save it
|
|
|
|
let local_private_key = Keypair::generate_secp256k1();
|
|
|
|
if let Keypair::Secp256k1(key) = local_private_key.clone() {
|
|
|
|
let _ = std::fs::create_dir_all(&config.network_dir);
|
|
|
|
match File::create(network_key_f.clone())
|
|
|
|
.and_then(|mut f| f.write_all(&key.secret().to_bytes()))
|
|
|
|
{
|
|
|
|
Ok(_) => {
|
|
|
|
debug!(log, "New network key generated and written to disk");
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
warn!(
|
|
|
|
log,
|
|
|
|
"Could not write node key to file: {:?}. Error: {}", network_key_f, e
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
local_private_key
|
|
|
|
}
|