1#![doc(
10 html_logo_url = "https://raw.githubusercontent.com/paradigmxyz/reth/main/assets/reth-docs.png",
11 html_favicon_url = "https://avatars0.githubusercontent.com/u/97369466?s=256",
12 issue_tracker_base_url = "https://github.com/SeismicSystems/seismic-reth/issues/"
13)]
14#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))]
15
16pub mod downloaders;
17pub mod error;
19pub mod events;
20pub mod noop;
22pub mod test_utils;
23
24pub use alloy_rpc_types_admin::EthProtocolInfo;
25use reth_network_p2p::sync::NetworkSyncUpdater;
26pub use reth_network_p2p::BlockClient;
27pub use reth_network_types::{PeerKind, Reputation, ReputationChangeKind};
28
29pub use downloaders::BlockDownloaderProvider;
30pub use error::NetworkError;
31pub use events::{
32 DiscoveredEvent, DiscoveryEvent, NetworkEvent, NetworkEventListenerProvider, PeerRequest,
33 PeerRequestSender,
34};
35
36use std::{future::Future, net::SocketAddr, sync::Arc, time::Instant};
37
38use reth_eth_wire_types::{capability::Capabilities, DisconnectReason, EthVersion, Status};
39use reth_network_peers::NodeRecord;
40
41pub type PeerId = alloy_primitives::B512;
43
44pub trait FullNetwork:
46 BlockDownloaderProvider
47 + NetworkSyncUpdater
48 + NetworkInfo
49 + NetworkEventListenerProvider
50 + PeersInfo
51 + Peers
52 + Clone
53 + 'static
54{
55}
56
57impl<T> FullNetwork for T where
58 T: BlockDownloaderProvider
59 + NetworkSyncUpdater
60 + NetworkInfo
61 + NetworkEventListenerProvider
62 + PeersInfo
63 + Peers
64 + Clone
65 + 'static
66{
67}
68
69#[auto_impl::auto_impl(&, Arc)]
71pub trait NetworkInfo: Send + Sync {
72 fn local_addr(&self) -> SocketAddr;
74
75 fn network_status(&self) -> impl Future<Output = Result<NetworkStatus, NetworkError>> + Send;
77
78 fn chain_id(&self) -> u64;
80
81 fn is_syncing(&self) -> bool;
83
84 fn is_initially_syncing(&self) -> bool;
86}
87
88#[auto_impl::auto_impl(&, Arc)]
90pub trait PeersInfo: Send + Sync {
91 fn num_connected_peers(&self) -> usize;
95
96 fn local_node_record(&self) -> NodeRecord;
98
99 fn local_enr(&self) -> enr::Enr<enr::secp256k1::SecretKey>;
101}
102
103#[auto_impl::auto_impl(&, Arc)]
105pub trait Peers: PeersInfo {
106 fn add_peer(&self, peer: PeerId, tcp_addr: SocketAddr) {
108 self.add_peer_kind(peer, PeerKind::Static, tcp_addr, None);
109 }
110
111 fn add_peer_with_udp(&self, peer: PeerId, tcp_addr: SocketAddr, udp_addr: SocketAddr) {
113 self.add_peer_kind(peer, PeerKind::Static, tcp_addr, Some(udp_addr));
114 }
115
116 fn add_trusted_peer_id(&self, peer: PeerId);
120
121 fn add_trusted_peer(&self, peer: PeerId, tcp_addr: SocketAddr) {
123 self.add_peer_kind(peer, PeerKind::Trusted, tcp_addr, None);
124 }
125
126 fn add_trusted_peer_with_udp(&self, peer: PeerId, tcp_addr: SocketAddr, udp_addr: SocketAddr) {
128 self.add_peer_kind(peer, PeerKind::Trusted, tcp_addr, Some(udp_addr));
129 }
130
131 fn add_peer_kind(
133 &self,
134 peer: PeerId,
135 kind: PeerKind,
136 tcp_addr: SocketAddr,
137 udp_addr: Option<SocketAddr>,
138 );
139
140 fn get_trusted_peers(
142 &self,
143 ) -> impl Future<Output = Result<Vec<PeerInfo>, NetworkError>> + Send {
144 self.get_peers_by_kind(PeerKind::Trusted)
145 }
146
147 fn get_basic_peers(&self) -> impl Future<Output = Result<Vec<PeerInfo>, NetworkError>> + Send {
149 self.get_peers_by_kind(PeerKind::Basic)
150 }
151
152 fn get_peers_by_kind(
154 &self,
155 kind: PeerKind,
156 ) -> impl Future<Output = Result<Vec<PeerInfo>, NetworkError>> + Send;
157
158 fn get_all_peers(&self) -> impl Future<Output = Result<Vec<PeerInfo>, NetworkError>> + Send;
160
161 fn get_peer_by_id(
165 &self,
166 peer_id: PeerId,
167 ) -> impl Future<Output = Result<Option<PeerInfo>, NetworkError>> + Send;
168
169 fn get_peers_by_id(
174 &self,
175 peer_ids: Vec<PeerId>,
176 ) -> impl Future<Output = Result<Vec<PeerInfo>, NetworkError>> + Send;
177
178 fn remove_peer(&self, peer: PeerId, kind: PeerKind);
180
181 fn disconnect_peer(&self, peer: PeerId);
183
184 fn disconnect_peer_with_reason(&self, peer: PeerId, reason: DisconnectReason);
186
187 fn connect_peer(&self, peer: PeerId, tcp_addr: SocketAddr) {
190 self.connect_peer_kind(peer, PeerKind::Static, tcp_addr, None)
191 }
192
193 fn connect_peer_kind(
195 &self,
196 peer: PeerId,
197 kind: PeerKind,
198 tcp_addr: SocketAddr,
199 udp_addr: Option<SocketAddr>,
200 );
201
202 fn reputation_change(&self, peer_id: PeerId, kind: ReputationChangeKind);
204
205 fn reputation_by_id(
207 &self,
208 peer_id: PeerId,
209 ) -> impl Future<Output = Result<Option<Reputation>, NetworkError>> + Send;
210}
211
212#[derive(Debug, Clone)]
214pub struct PeerInfo {
215 pub capabilities: Arc<Capabilities>,
217 pub remote_id: PeerId,
219 pub client_version: Arc<str>,
221 pub enode: String,
223 pub enr: Option<String>,
225 pub remote_addr: SocketAddr,
227 pub local_addr: Option<SocketAddr>,
229 pub direction: Direction,
231 pub eth_version: EthVersion,
233 pub status: Arc<Status>,
235 pub session_established: Instant,
237 pub kind: PeerKind,
239}
240
241#[derive(Debug, Copy, Clone, PartialEq, Eq)]
243pub enum Direction {
244 Incoming,
246 Outgoing(PeerId),
248}
249
250impl Direction {
251 pub const fn is_incoming(&self) -> bool {
253 matches!(self, Self::Incoming)
254 }
255
256 pub const fn is_outgoing(&self) -> bool {
258 matches!(self, Self::Outgoing(_))
259 }
260}
261
262impl std::fmt::Display for Direction {
263 fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
264 match self {
265 Self::Incoming => write!(f, "incoming"),
266 Self::Outgoing(_) => write!(f, "outgoing"),
267 }
268 }
269}
270
271#[derive(Clone, Debug)]
273#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
274pub struct NetworkStatus {
275 pub client_version: String,
277 pub protocol_version: u64,
279 pub eth_protocol_info: EthProtocolInfo,
281}