use crate::metrics; use crate::Config; use crate::{libp2p_event_handler::Libp2pEventHandler, peer_manager::PeerManager}; use chunk_pool::ChunkPoolMessage; use file_location_cache::FileLocationCache; use futures::{channel::mpsc::Sender, prelude::*}; use miner::MinerMessage; use network::{ BehaviourEvent, Keypair, Libp2pEvent, NetworkGlobals, NetworkMessage, RequestId, Service as LibP2PService, Swarm, }; use pruner::PrunerMessage; use std::sync::Arc; use storage::log_store::Store as LogStore; use storage_async::Store; use sync::{SyncMessage, SyncSender}; use task_executor::ShutdownReason; use tokio::sync::mpsc::UnboundedSender; use tokio::sync::{broadcast, mpsc, RwLock}; use tokio::time::interval; /// Service that handles communication between internal services and the libp2p service. pub struct RouterService { config: Config, /// The underlying libp2p service that drives all the network interactions. libp2p: LibP2PService, /// A collection of global variables, accessible outside of the network service. network_globals: Arc, /// The receiver channel for Zgs to communicate with the network service. network_recv: mpsc::UnboundedReceiver, /// The receiver channel for Zgs to communicate with the pruner service. pruner_recv: Option>, /// All connected peers. peers: Arc>, /// Handler for libp2p events. libp2p_event_handler: Libp2pEventHandler, /// Stores potentially created UPnP mappings to be removed on shutdown. (TCP port and UDP /// port). upnp_mappings: (Option, Option), } impl RouterService { #[allow(clippy::too_many_arguments)] pub fn spawn( executor: task_executor::TaskExecutor, libp2p: LibP2PService, network_globals: Arc, network_recv: mpsc::UnboundedReceiver, network_send: mpsc::UnboundedSender, sync_send: SyncSender, _miner_send: Option>, chunk_pool_send: UnboundedSender, pruner_recv: Option>, store: Arc, file_location_cache: Arc, local_keypair: Keypair, config: Config, ) { let store = Store::new(store, executor.clone()); let peers = Arc::new(RwLock::new(PeerManager::new(config.clone()))); // create the network service and spawn the task let router = RouterService { config: config.clone(), libp2p, network_globals: network_globals.clone(), network_recv, pruner_recv, peers: peers.clone(), libp2p_event_handler: Libp2pEventHandler::new( config, network_globals, network_send, sync_send, chunk_pool_send, local_keypair, store, file_location_cache, peers, ), upnp_mappings: (None, None), }; // spawn service let shutdown_sender = executor.shutdown_sender(); executor.spawn(router.main(shutdown_sender), "router"); } async fn main(mut self, mut shutdown_sender: Sender) { let mut heartbeat_service = interval(self.config.heartbeat_interval); let mut heartbeat_batcher = interval(self.config.batcher_timeout); loop { tokio::select! { // handle a message sent to the network Some(msg) = self.network_recv.recv() => self.on_network_msg(msg, &mut shutdown_sender).await, // handle event coming from the network event = self.libp2p.next_event() => self.on_libp2p_event(event, &mut shutdown_sender).await, Some(msg) = Self::try_recv(&mut self.pruner_recv) => self.on_pruner_msg(msg).await, // heartbeat for service _ = heartbeat_service.tick() => self.on_heartbeat().await, // heartbeat for expire file batcher _ = heartbeat_batcher.tick() => self.libp2p_event_handler.expire_batcher().await, } } } async fn try_recv(maybe_recv: &mut Option>) -> Option { match maybe_recv { None => None, Some(recv) => recv.recv().await, } } /// Handle an event received from the network. async fn on_libp2p_event( &mut self, ev: Libp2pEvent, shutdown_sender: &mut Sender, ) { trace!(?ev, "Received new event from libp2p"); match ev { Libp2pEvent::Behaviour(event) => match event { BehaviourEvent::PeerConnectedOutgoing(peer_id) => { self.libp2p_event_handler .on_peer_connected(peer_id, true) .await; } BehaviourEvent::PeerConnectedIncoming(peer_id) => { self.libp2p_event_handler .on_peer_connected(peer_id, false) .await; } BehaviourEvent::PeerBanned(_) | BehaviourEvent::PeerUnbanned(_) => { // No action required for these events. } BehaviourEvent::PeerDisconnected(peer_id) => { self.libp2p_event_handler .on_peer_disconnected(peer_id) .await; } BehaviourEvent::RequestReceived { peer_id, id, request, } => { if self.network_globals.peers.read().is_connected(&peer_id) { self.libp2p_event_handler .on_rpc_request(peer_id, id, request) .await; } else { debug!(%peer_id, ?request, "Dropping request of disconnected peer"); } } BehaviourEvent::ResponseReceived { peer_id, id, response, } => { self.libp2p_event_handler .on_rpc_response(peer_id, id, response) .await; } BehaviourEvent::RPCFailed { id, peer_id } => { self.libp2p_event_handler.on_rpc_error(peer_id, id).await; } BehaviourEvent::StatusPeer(peer_id) => { self.libp2p_event_handler.send_status(peer_id); } BehaviourEvent::PubsubMessage { id, propagation_source, source, message, .. } => { let result = self .libp2p_event_handler .on_pubsub_message(propagation_source, source, &id, message) .await; self.libp2p .swarm .behaviour_mut() .report_message_validation_result(&propagation_source, id, result); } }, Libp2pEvent::NewListenAddr(multiaddr) => { info!(?multiaddr, "New listen address"); self.network_globals .listen_multiaddrs .write() .push(multiaddr); } Libp2pEvent::ZeroListeners => { let _ = shutdown_sender .send(ShutdownReason::Failure( "All listeners are closed. Unable to listen", )) .await .map_err(|e| { warn!( error = %e, "failed to send a shutdown signal", ) }); } } } /// Handle a message sent to the network service. async fn on_network_msg( &mut self, msg: NetworkMessage, _shutdown_sender: &mut Sender, ) { trace!(?msg, "Received new message"); metrics::SERVICE_ROUTE_NETWORK_MESSAGE.mark(1); match msg { NetworkMessage::SendRequest { peer_id, request, request_id, } => { self.libp2p.send_request(peer_id, request_id, request); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_SEND_REQUEST.mark(1); } NetworkMessage::SendResponse { peer_id, response, id, } => { self.libp2p.send_response(peer_id, id, response); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_SEND_RESPONSE.mark(1); } NetworkMessage::SendErrorResponse { peer_id, error, id, reason, } => { self.libp2p.respond_with_error(peer_id, id, error, reason); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_SEND_ERROR_RESPONSE.mark(1); } NetworkMessage::Publish { messages } => { if self.libp2p.swarm.connected_peers().next().is_none() { // this is a boardcast message, when current node doesn't have any peers connected, try to connect any peer in config for multiaddr in &self.config.libp2p_nodes { match Swarm::dial(&mut self.libp2p.swarm, multiaddr.clone()) { Ok(()) => { debug!(address = %multiaddr, "Dialing libp2p peer"); break; } Err(err) => { debug!(address = %multiaddr, error = ?err, "Could not connect to peer"); } }; } } let mut topic_kinds = Vec::new(); for message in &messages { if !topic_kinds.contains(&message.kind()) { topic_kinds.push(message.kind()); } } debug!( count = messages.len(), topics = ?topic_kinds, "Sending pubsub messages", ); self.libp2p.swarm.behaviour_mut().publish(messages); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_PUBLISH.mark(1); } NetworkMessage::ReportPeer { peer_id, action, source, msg, } => { self.libp2p.report_peer(&peer_id, action, source, msg); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_REPORT_PEER.mark(1); } NetworkMessage::GoodbyePeer { peer_id, reason, source, } => { self.libp2p.goodbye_peer(&peer_id, reason, source); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_GOODBYE_PEER.mark(1); } NetworkMessage::DialPeer { address, peer_id } => { metrics::SERVICE_ROUTE_NETWORK_MESSAGE_DAIL_PEER.mark(1); if self.libp2p.swarm.is_connected(&peer_id) { self.libp2p_event_handler .send_to_sync(SyncMessage::PeerConnected { peer_id }); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_DAIL_PEER_ALREADY.mark(1); } else { match Swarm::dial(&mut self.libp2p.swarm, address.clone()) { Ok(()) => { debug!(%address, "Dialing libp2p peer"); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_DAIL_PEER_NEW_OK.mark(1); } Err(err) => { info!(%address, error = ?err, "Failed to dial peer"); self.libp2p_event_handler .send_to_sync(SyncMessage::DailFailed { peer_id, err }); metrics::SERVICE_ROUTE_NETWORK_MESSAGE_DAIL_PEER_NEW_FAIL.mark(1); } }; } } NetworkMessage::AnnounceLocalFile { tx_id } => { if self .libp2p_event_handler .publish_file(tx_id) .await .is_some() { metrics::SERVICE_ROUTE_NETWORK_MESSAGE_ANNOUNCE_LOCAL_FILE.mark(1); } } NetworkMessage::UPnPMappingEstablished { tcp_socket, udp_socket, } => { metrics::SERVICE_ROUTE_NETWORK_MESSAGE_UPNP.mark(1); self.upnp_mappings = (tcp_socket.map(|s| s.port()), udp_socket.map(|s| s.port())); // If there is an external TCP port update, modify our local ENR. if let Some(tcp_socket) = tcp_socket { if let Err(e) = self .libp2p .swarm .behaviour_mut() .discovery_mut() .update_enr_tcp_port(tcp_socket.port()) { warn!(error = %e, "Failed to update ENR"); } } if let Some(udp_socket) = udp_socket { if let Err(e) = self .libp2p .swarm .behaviour_mut() .discovery_mut() .update_enr_udp_socket(udp_socket) { warn!(error = %e, "Failed to update ENR"); } } } } } async fn on_pruner_msg(&mut self, msg: PrunerMessage) { match msg { PrunerMessage::ChangeShardConfig(shard_config) => { self.libp2p_event_handler .send_to_chunk_pool(ChunkPoolMessage::ChangeShardConfig(shard_config)); if let Some(msg) = self .libp2p_event_handler .construct_announce_shard_config_message(shard_config) .await { self.libp2p_event_handler.publish(msg) } } } } async fn on_heartbeat(&mut self) { let expired_peers = self.peers.write().await.expired_peers(); let num_expired_peers = expired_peers.len() as u64; metrics::SERVICE_EXPIRED_PEERS.update(num_expired_peers); if num_expired_peers > 0 { debug!(%num_expired_peers, "Heartbeat, remove expired peers") } let mut num_succeeded = 0; let mut num_failed = 0; for peer_id in expired_peers { // async operation, once peer disconnected, swarm event `PeerDisconnected` // will be polled to handle in advance. match self.libp2p.swarm.disconnect_peer_id(peer_id) { Ok(_) => { debug!(%peer_id, "Peer expired and disconnect it"); num_succeeded += 1; } Err(_) => { debug!(%peer_id, "Peer expired but failed to disconnect"); num_failed += 1; } } } metrics::SERVICE_EXPIRED_PEERS_DISCONNECT_OK.update(num_succeeded); metrics::SERVICE_EXPIRED_PEERS_DISCONNECT_FAIL.update(num_failed); } } impl Drop for RouterService { fn drop(&mut self) { info!("Router service shutdown"); // attempt to remove port mappings network::nat::remove_mappings(self.upnp_mappings.0, self.upnp_mappings.1); } }