mirror of
https://github.com/paradigmxyz/reth.git
synced 2026-02-11 07:25:09 -05:00
Co-authored-by: Emilia Hane <elsaemiliaevahane@gmail.com> Co-authored-by: Bjerg <onbjerg@users.noreply.github.com> Co-authored-by: Matthias Seitz <matthias.seitz@outlook.de>
598 lines
21 KiB
Rust
598 lines
21 KiB
Rust
//! Network config support
|
|
|
|
use crate::{
|
|
error::NetworkError,
|
|
import::{BlockImport, ProofOfStakeBlockImport},
|
|
peers::PeersConfig,
|
|
session::SessionsConfig,
|
|
transactions::TransactionsManagerConfig,
|
|
NetworkHandle, NetworkManager,
|
|
};
|
|
use reth_discv4::{Discv4Config, Discv4ConfigBuilder, DEFAULT_DISCOVERY_ADDRESS};
|
|
use reth_dns_discovery::DnsDiscoveryConfig;
|
|
use reth_ecies::util::pk2id;
|
|
use reth_eth_wire::{HelloMessage, HelloMessageWithProtocols, Status};
|
|
use reth_primitives::{
|
|
mainnet_nodes, sepolia_nodes, ChainSpec, ForkFilter, Head, NodeRecord, PeerId, MAINNET,
|
|
};
|
|
use reth_provider::{BlockReader, HeaderProvider};
|
|
use reth_tasks::{TaskSpawner, TokioTaskExecutor};
|
|
use secp256k1::SECP256K1;
|
|
use std::{collections::HashSet, net::SocketAddr, sync::Arc};
|
|
// re-export for convenience
|
|
use crate::protocol::{IntoRlpxSubProtocol, RlpxSubProtocols};
|
|
pub use secp256k1::SecretKey;
|
|
|
|
/// Convenience function to create a new random [`SecretKey`]
|
|
pub fn rng_secret_key() -> SecretKey {
|
|
SecretKey::new(&mut rand::thread_rng())
|
|
}
|
|
|
|
/// All network related initialization settings.
|
|
#[derive(Debug)]
|
|
pub struct NetworkConfig<C> {
|
|
/// The client type that can interact with the chain.
|
|
///
|
|
/// This type is used to fetch the block number after we established a session and received the
|
|
/// [Status] block hash.
|
|
pub client: C,
|
|
/// The node's secret key, from which the node's identity is derived.
|
|
pub secret_key: SecretKey,
|
|
/// All boot nodes to start network discovery with.
|
|
pub boot_nodes: HashSet<NodeRecord>,
|
|
/// How to set up discovery over DNS.
|
|
pub dns_discovery_config: Option<DnsDiscoveryConfig>,
|
|
/// How to set up discovery.
|
|
pub discovery_v4_config: Option<Discv4Config>,
|
|
/// Address to use for discovery
|
|
pub discovery_addr: SocketAddr,
|
|
/// Address to listen for incoming connections
|
|
pub listener_addr: SocketAddr,
|
|
/// How to instantiate peer manager.
|
|
pub peers_config: PeersConfig,
|
|
/// How to configure the [SessionManager](crate::session::SessionManager).
|
|
pub sessions_config: SessionsConfig,
|
|
/// The chain spec
|
|
pub chain_spec: Arc<ChainSpec>,
|
|
/// The [`ForkFilter`] to use at launch for authenticating sessions.
|
|
///
|
|
/// See also <https://github.com/ethereum/EIPs/blob/master/EIPS/eip-2124.md#stale-software-examples>
|
|
///
|
|
/// For sync from block `0`, this should be the default chain [`ForkFilter`] beginning at the
|
|
/// first hardfork, `Frontier` for mainnet.
|
|
pub fork_filter: ForkFilter,
|
|
/// The block importer type.
|
|
pub block_import: Box<dyn BlockImport>,
|
|
/// The default mode of the network.
|
|
pub network_mode: NetworkMode,
|
|
/// The executor to use for spawning tasks.
|
|
pub executor: Box<dyn TaskSpawner>,
|
|
/// The `Status` message to send to peers at the beginning.
|
|
pub status: Status,
|
|
/// Sets the hello message for the p2p handshake in RLPx
|
|
pub hello_message: HelloMessageWithProtocols,
|
|
/// Additional protocols to announce and handle in RLPx
|
|
pub extra_protocols: RlpxSubProtocols,
|
|
/// Whether to disable transaction gossip
|
|
pub tx_gossip_disabled: bool,
|
|
/// How to instantiate transactions manager.
|
|
pub transactions_manager_config: TransactionsManagerConfig,
|
|
/// Optimism Network Config
|
|
#[cfg(feature = "optimism")]
|
|
pub optimism_network_config: OptimismNetworkConfig,
|
|
}
|
|
|
|
/// Optimism Network Config
|
|
#[cfg(feature = "optimism")]
|
|
#[derive(Debug, Clone, Default)]
|
|
pub struct OptimismNetworkConfig {
|
|
/// The sequencer HTTP endpoint, if provided via CLI flag
|
|
pub sequencer_endpoint: Option<String>,
|
|
}
|
|
|
|
// === impl NetworkConfig ===
|
|
|
|
impl NetworkConfig<()> {
|
|
/// Convenience method for creating the corresponding builder type
|
|
pub fn builder(secret_key: SecretKey) -> NetworkConfigBuilder {
|
|
NetworkConfigBuilder::new(secret_key)
|
|
}
|
|
}
|
|
|
|
impl<C> NetworkConfig<C> {
|
|
/// Create a new instance with all mandatory fields set, rest is field with defaults.
|
|
pub fn new(client: C, secret_key: SecretKey) -> Self {
|
|
NetworkConfig::builder(secret_key).build(client)
|
|
}
|
|
|
|
/// Sets the config to use for the discovery v4 protocol.
|
|
pub fn set_discovery_v4(mut self, discovery_config: Discv4Config) -> Self {
|
|
self.discovery_v4_config = Some(discovery_config);
|
|
self
|
|
}
|
|
|
|
/// Sets the address for the incoming connection listener.
|
|
pub fn set_listener_addr(mut self, listener_addr: SocketAddr) -> Self {
|
|
self.listener_addr = listener_addr;
|
|
self
|
|
}
|
|
}
|
|
|
|
impl<C> NetworkConfig<C>
|
|
where
|
|
C: BlockReader + HeaderProvider + Clone + Unpin + 'static,
|
|
{
|
|
/// Starts the networking stack given a [NetworkConfig] and returns a handle to the network.
|
|
pub async fn start_network(self) -> Result<NetworkHandle, NetworkError> {
|
|
let client = self.client.clone();
|
|
let (handle, network, _txpool, eth) =
|
|
NetworkManager::builder(self).await?.request_handler(client).split_with_handle();
|
|
|
|
tokio::task::spawn(network);
|
|
// TODO: tokio::task::spawn(txpool);
|
|
tokio::task::spawn(eth);
|
|
Ok(handle)
|
|
}
|
|
}
|
|
|
|
/// Builder for [`NetworkConfig`](struct.NetworkConfig.html).
|
|
#[derive(Debug)]
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
|
|
pub struct NetworkConfigBuilder {
|
|
/// The node's secret key, from which the node's identity is derived.
|
|
secret_key: SecretKey,
|
|
/// How to configure discovery over DNS.
|
|
dns_discovery_config: Option<DnsDiscoveryConfig>,
|
|
/// How to set up discovery.
|
|
discovery_v4_builder: Option<Discv4ConfigBuilder>,
|
|
/// All boot nodes to start network discovery with.
|
|
boot_nodes: HashSet<NodeRecord>,
|
|
/// Address to use for discovery
|
|
discovery_addr: Option<SocketAddr>,
|
|
/// Listener for incoming connections
|
|
listener_addr: Option<SocketAddr>,
|
|
/// How to instantiate peer manager.
|
|
peers_config: Option<PeersConfig>,
|
|
/// How to configure the sessions manager
|
|
sessions_config: Option<SessionsConfig>,
|
|
/// The network's chain spec
|
|
chain_spec: Arc<ChainSpec>,
|
|
/// The default mode of the network.
|
|
network_mode: NetworkMode,
|
|
/// The executor to use for spawning tasks.
|
|
#[serde(skip)]
|
|
executor: Option<Box<dyn TaskSpawner>>,
|
|
/// Sets the hello message for the p2p handshake in RLPx
|
|
hello_message: Option<HelloMessageWithProtocols>,
|
|
/// The executor to use for spawning tasks.
|
|
#[serde(skip)]
|
|
extra_protocols: RlpxSubProtocols,
|
|
/// Head used to start set for the fork filter and status.
|
|
head: Option<Head>,
|
|
/// Whether tx gossip is disabled
|
|
tx_gossip_disabled: bool,
|
|
/// The block importer type
|
|
#[serde(skip)]
|
|
block_import: Option<Box<dyn BlockImport>>,
|
|
/// How to instantiate transactions manager.
|
|
transactions_manager_config: TransactionsManagerConfig,
|
|
/// Optimism Network Config Builder
|
|
#[cfg(feature = "optimism")]
|
|
optimism_network_config: OptimismNetworkConfigBuilder,
|
|
}
|
|
|
|
/// Optimism Network Config Builder
|
|
#[cfg(feature = "optimism")]
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
|
|
#[derive(Debug, Clone, Default)]
|
|
pub struct OptimismNetworkConfigBuilder {
|
|
/// The sequencer HTTP endpoint, if provided via CLI flag
|
|
sequencer_endpoint: Option<String>,
|
|
}
|
|
|
|
// === impl NetworkConfigBuilder ===
|
|
|
|
#[allow(missing_docs)]
|
|
impl NetworkConfigBuilder {
|
|
pub fn new(secret_key: SecretKey) -> Self {
|
|
Self {
|
|
secret_key,
|
|
dns_discovery_config: Some(Default::default()),
|
|
discovery_v4_builder: Some(Default::default()),
|
|
boot_nodes: Default::default(),
|
|
discovery_addr: None,
|
|
listener_addr: None,
|
|
peers_config: None,
|
|
sessions_config: None,
|
|
chain_spec: MAINNET.clone(),
|
|
network_mode: Default::default(),
|
|
executor: None,
|
|
hello_message: None,
|
|
extra_protocols: Default::default(),
|
|
head: None,
|
|
tx_gossip_disabled: false,
|
|
block_import: None,
|
|
#[cfg(feature = "optimism")]
|
|
optimism_network_config: OptimismNetworkConfigBuilder::default(),
|
|
transactions_manager_config: Default::default(),
|
|
}
|
|
}
|
|
|
|
/// Returns the configured [`PeerId`]
|
|
pub fn get_peer_id(&self) -> PeerId {
|
|
pk2id(&self.secret_key.public_key(SECP256K1))
|
|
}
|
|
|
|
/// Sets the chain spec.
|
|
pub fn chain_spec(mut self, chain_spec: Arc<ChainSpec>) -> Self {
|
|
self.chain_spec = chain_spec;
|
|
self
|
|
}
|
|
|
|
/// Sets the [`NetworkMode`].
|
|
pub fn network_mode(mut self, network_mode: NetworkMode) -> Self {
|
|
self.network_mode = network_mode;
|
|
self
|
|
}
|
|
|
|
/// Sets the highest synced block.
|
|
///
|
|
/// This is used to construct the appropriate [`ForkFilter`] and [`Status`] message.
|
|
///
|
|
/// If not set, this defaults to the genesis specified by the current chain specification.
|
|
pub fn set_head(mut self, head: Head) -> Self {
|
|
self.head = Some(head);
|
|
self
|
|
}
|
|
|
|
/// Sets the `HelloMessage` to send when connecting to peers.
|
|
///
|
|
/// ```
|
|
/// # use reth_eth_wire::HelloMessage;
|
|
/// # use reth_network::NetworkConfigBuilder;
|
|
/// # fn builder(builder: NetworkConfigBuilder) {
|
|
/// let peer_id = builder.get_peer_id();
|
|
/// builder.hello_message(HelloMessage::builder(peer_id).build());
|
|
/// # }
|
|
/// ```
|
|
pub fn hello_message(mut self, hello_message: HelloMessageWithProtocols) -> Self {
|
|
self.hello_message = Some(hello_message);
|
|
self
|
|
}
|
|
|
|
/// Set a custom peer config for how peers are handled
|
|
pub fn peer_config(mut self, config: PeersConfig) -> Self {
|
|
self.peers_config = Some(config);
|
|
self
|
|
}
|
|
|
|
/// Sets the executor to use for spawning tasks.
|
|
///
|
|
/// If `None`, then [tokio::spawn] is used for spawning tasks.
|
|
pub fn with_task_executor(mut self, executor: Box<dyn TaskSpawner>) -> Self {
|
|
self.executor = Some(executor);
|
|
self
|
|
}
|
|
|
|
/// Sets a custom config for how sessions are handled.
|
|
pub fn sessions_config(mut self, config: SessionsConfig) -> Self {
|
|
self.sessions_config = Some(config);
|
|
self
|
|
}
|
|
|
|
pub fn transactions_manager_config(mut self, config: TransactionsManagerConfig) -> Self {
|
|
self.transactions_manager_config = config;
|
|
self
|
|
}
|
|
|
|
/// Sets the discovery and listener address
|
|
///
|
|
/// This is a convenience function for both [NetworkConfigBuilder::listener_addr] and
|
|
/// [NetworkConfigBuilder::discovery_addr].
|
|
///
|
|
/// By default, both are on the same port:
|
|
/// [DEFAULT_DISCOVERY_PORT](reth_discv4::DEFAULT_DISCOVERY_PORT)
|
|
pub fn set_addrs(self, addr: SocketAddr) -> Self {
|
|
self.listener_addr(addr).discovery_addr(addr)
|
|
}
|
|
|
|
/// Sets the socket address the network will listen on.
|
|
///
|
|
/// By default, this is [DEFAULT_DISCOVERY_ADDRESS]
|
|
pub fn listener_addr(mut self, listener_addr: SocketAddr) -> Self {
|
|
self.listener_addr = Some(listener_addr);
|
|
self
|
|
}
|
|
|
|
/// Sets the port of the address the network will listen on.
|
|
///
|
|
/// By default, this is [DEFAULT_DISCOVERY_PORT](reth_discv4::DEFAULT_DISCOVERY_PORT)
|
|
pub fn listener_port(mut self, port: u16) -> Self {
|
|
self.listener_addr.get_or_insert(DEFAULT_DISCOVERY_ADDRESS).set_port(port);
|
|
self
|
|
}
|
|
|
|
/// Sets the socket address the discovery network will listen on
|
|
pub fn discovery_addr(mut self, discovery_addr: SocketAddr) -> Self {
|
|
self.discovery_addr = Some(discovery_addr);
|
|
self
|
|
}
|
|
|
|
/// Sets the port of the address the discovery network will listen on.
|
|
///
|
|
/// By default, this is [DEFAULT_DISCOVERY_PORT](reth_discv4::DEFAULT_DISCOVERY_PORT)
|
|
pub fn discovery_port(mut self, port: u16) -> Self {
|
|
self.discovery_addr.get_or_insert(DEFAULT_DISCOVERY_ADDRESS).set_port(port);
|
|
self
|
|
}
|
|
|
|
/// Sets the discv4 config to use.
|
|
pub fn discovery(mut self, builder: Discv4ConfigBuilder) -> Self {
|
|
self.discovery_v4_builder = Some(builder);
|
|
self
|
|
}
|
|
|
|
/// Sets the dns discovery config to use.
|
|
pub fn dns_discovery(mut self, config: DnsDiscoveryConfig) -> Self {
|
|
self.dns_discovery_config = Some(config);
|
|
self
|
|
}
|
|
|
|
/// Convenience function for setting [Self::boot_nodes] to the mainnet boot nodes.
|
|
pub fn mainnet_boot_nodes(self) -> Self {
|
|
self.boot_nodes(mainnet_nodes())
|
|
}
|
|
|
|
/// Convenience function for setting [Self::boot_nodes] to the sepolia boot nodes.
|
|
pub fn sepolia_boot_nodes(self) -> Self {
|
|
self.boot_nodes(sepolia_nodes())
|
|
}
|
|
|
|
/// Sets the boot nodes.
|
|
pub fn boot_nodes(mut self, nodes: impl IntoIterator<Item = NodeRecord>) -> Self {
|
|
self.boot_nodes = nodes.into_iter().collect();
|
|
self
|
|
}
|
|
|
|
/// Disable the DNS discovery.
|
|
pub fn disable_dns_discovery(mut self) -> Self {
|
|
self.dns_discovery_config = None;
|
|
self
|
|
}
|
|
|
|
/// Disables all discovery.
|
|
pub fn disable_discovery(self) -> Self {
|
|
self.disable_discv4_discovery().disable_dns_discovery()
|
|
}
|
|
|
|
/// Disables all discovery if the given condition is true.
|
|
pub fn disable_discovery_if(self, disable: bool) -> Self {
|
|
if disable {
|
|
self.disable_discovery()
|
|
} else {
|
|
self
|
|
}
|
|
}
|
|
|
|
/// Disable the Discv4 discovery.
|
|
pub fn disable_discv4_discovery(mut self) -> Self {
|
|
self.discovery_v4_builder = None;
|
|
self
|
|
}
|
|
|
|
/// Disable the DNS discovery if the given condition is true.
|
|
pub fn disable_dns_discovery_if(self, disable: bool) -> Self {
|
|
if disable {
|
|
self.disable_dns_discovery()
|
|
} else {
|
|
self
|
|
}
|
|
}
|
|
|
|
/// Disable the Discv4 discovery if the given condition is true.
|
|
pub fn disable_discv4_discovery_if(self, disable: bool) -> Self {
|
|
if disable {
|
|
self.disable_discv4_discovery()
|
|
} else {
|
|
self
|
|
}
|
|
}
|
|
|
|
/// Adds a new additional protocol to the RLPx sub-protocol list.
|
|
pub fn add_rlpx_sub_protocol(mut self, protocol: impl IntoRlpxSubProtocol) -> Self {
|
|
self.extra_protocols.push(protocol);
|
|
self
|
|
}
|
|
|
|
/// Sets whether tx gossip is disabled.
|
|
pub fn disable_tx_gossip(mut self, disable_tx_gossip: bool) -> Self {
|
|
self.tx_gossip_disabled = disable_tx_gossip;
|
|
self
|
|
}
|
|
|
|
/// Sets the block import type.
|
|
pub fn block_import(mut self, block_import: Box<dyn BlockImport>) -> Self {
|
|
self.block_import = Some(block_import);
|
|
self
|
|
}
|
|
|
|
/// Sets the sequencer HTTP endpoint.
|
|
#[cfg(feature = "optimism")]
|
|
pub fn sequencer_endpoint(mut self, endpoint: Option<String>) -> Self {
|
|
self.optimism_network_config.sequencer_endpoint = endpoint;
|
|
self
|
|
}
|
|
|
|
/// Convenience function for creating a [NetworkConfig] with a noop provider that does nothing.
|
|
#[cfg(any(test, feature = "test-utils"))]
|
|
pub fn build_with_noop_provider(
|
|
self,
|
|
) -> NetworkConfig<reth_provider::test_utils::NoopProvider> {
|
|
self.build(reth_provider::test_utils::NoopProvider::default())
|
|
}
|
|
|
|
/// Consumes the type and creates the actual [`NetworkConfig`]
|
|
/// for the given client type that can interact with the chain.
|
|
///
|
|
/// The given client is to be used for interacting with the chain, for example fetching the
|
|
/// corresponding block for a given block hash we receive from a peer in the status message when
|
|
/// establishing a connection.
|
|
pub fn build<C>(self, client: C) -> NetworkConfig<C> {
|
|
let peer_id = self.get_peer_id();
|
|
let Self {
|
|
secret_key,
|
|
mut dns_discovery_config,
|
|
discovery_v4_builder,
|
|
boot_nodes,
|
|
discovery_addr,
|
|
listener_addr,
|
|
peers_config,
|
|
sessions_config,
|
|
chain_spec,
|
|
network_mode,
|
|
executor,
|
|
hello_message,
|
|
extra_protocols,
|
|
head,
|
|
tx_gossip_disabled,
|
|
block_import,
|
|
#[cfg(feature = "optimism")]
|
|
optimism_network_config: OptimismNetworkConfigBuilder { sequencer_endpoint },
|
|
transactions_manager_config,
|
|
} = self;
|
|
|
|
let listener_addr = listener_addr.unwrap_or(DEFAULT_DISCOVERY_ADDRESS);
|
|
|
|
let mut hello_message =
|
|
hello_message.unwrap_or_else(|| HelloMessage::builder(peer_id).build());
|
|
hello_message.port = listener_addr.port();
|
|
|
|
let head = head.unwrap_or(Head {
|
|
hash: chain_spec.genesis_hash(),
|
|
number: 0,
|
|
timestamp: chain_spec.genesis.timestamp,
|
|
difficulty: chain_spec.genesis.difficulty,
|
|
total_difficulty: chain_spec.genesis.difficulty,
|
|
});
|
|
|
|
// set the status
|
|
let status = Status::spec_builder(&chain_spec, &head).build();
|
|
|
|
// set a fork filter based on the chain spec and head
|
|
let fork_filter = chain_spec.fork_filter(head);
|
|
|
|
// If default DNS config is used then we add the known dns network to bootstrap from
|
|
if let Some(dns_networks) =
|
|
dns_discovery_config.as_mut().and_then(|c| c.bootstrap_dns_networks.as_mut())
|
|
{
|
|
if dns_networks.is_empty() {
|
|
if let Some(link) = chain_spec.chain().public_dns_network_protocol() {
|
|
dns_networks.insert(link.parse().expect("is valid DNS link entry"));
|
|
}
|
|
}
|
|
}
|
|
|
|
NetworkConfig {
|
|
client,
|
|
secret_key,
|
|
boot_nodes,
|
|
dns_discovery_config,
|
|
discovery_v4_config: discovery_v4_builder.map(|builder| builder.build()),
|
|
discovery_addr: discovery_addr.unwrap_or(DEFAULT_DISCOVERY_ADDRESS),
|
|
listener_addr,
|
|
peers_config: peers_config.unwrap_or_default(),
|
|
sessions_config: sessions_config.unwrap_or_default(),
|
|
chain_spec,
|
|
block_import: block_import.unwrap_or(Box::<ProofOfStakeBlockImport>::default()),
|
|
network_mode,
|
|
executor: executor.unwrap_or_else(|| Box::<TokioTaskExecutor>::default()),
|
|
status,
|
|
hello_message,
|
|
extra_protocols,
|
|
fork_filter,
|
|
tx_gossip_disabled,
|
|
#[cfg(feature = "optimism")]
|
|
optimism_network_config: OptimismNetworkConfig { sequencer_endpoint },
|
|
transactions_manager_config,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Describes the mode of the network wrt. POS or POW.
|
|
///
|
|
/// This affects block propagation in the `eth` sub-protocol [EIP-3675](https://eips.ethereum.org/EIPS/eip-3675#devp2p)
|
|
///
|
|
/// In POS `NewBlockHashes` and `NewBlock` messages become invalid.
|
|
#[derive(Debug, Clone, Copy, Eq, PartialEq, Default)]
|
|
#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))]
|
|
pub enum NetworkMode {
|
|
/// Network is in proof-of-work mode.
|
|
Work,
|
|
/// Network is in proof-of-stake mode
|
|
#[default]
|
|
Stake,
|
|
}
|
|
|
|
// === impl NetworkMode ===
|
|
|
|
impl NetworkMode {
|
|
/// Returns true if network has entered proof-of-stake
|
|
pub fn is_stake(&self) -> bool {
|
|
matches!(self, NetworkMode::Stake)
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use rand::thread_rng;
|
|
use reth_dns_discovery::tree::LinkEntry;
|
|
use reth_primitives::{Chain, ForkHash};
|
|
use reth_provider::test_utils::NoopProvider;
|
|
use std::collections::BTreeMap;
|
|
|
|
fn builder() -> NetworkConfigBuilder {
|
|
let secret_key = SecretKey::new(&mut thread_rng());
|
|
NetworkConfigBuilder::new(secret_key)
|
|
}
|
|
|
|
#[test]
|
|
fn test_network_dns_defaults() {
|
|
let config = builder().build(NoopProvider::default());
|
|
|
|
let dns = config.dns_discovery_config.unwrap();
|
|
let bootstrap_nodes = dns.bootstrap_dns_networks.unwrap();
|
|
let mainnet_dns: LinkEntry =
|
|
Chain::mainnet().public_dns_network_protocol().unwrap().parse().unwrap();
|
|
assert!(bootstrap_nodes.contains(&mainnet_dns));
|
|
assert_eq!(bootstrap_nodes.len(), 1);
|
|
}
|
|
|
|
#[test]
|
|
fn test_network_fork_filter_default() {
|
|
let mut chain_spec = Arc::clone(&MAINNET);
|
|
|
|
// remove any `next` fields we would have by removing all hardforks
|
|
Arc::make_mut(&mut chain_spec).hardforks = BTreeMap::new();
|
|
|
|
// check that the forkid is initialized with the genesis and no other forks
|
|
let genesis_fork_hash = ForkHash::from(chain_spec.genesis_hash());
|
|
|
|
// enforce that the fork_id set in the status is consistent with the generated fork filter
|
|
let config = builder().chain_spec(chain_spec).build(NoopProvider::default());
|
|
|
|
let status = config.status;
|
|
let fork_filter = config.fork_filter;
|
|
|
|
// assert that there are no other forks
|
|
assert_eq!(status.forkid.next, 0);
|
|
|
|
// assert the same thing for the fork_filter
|
|
assert_eq!(fork_filter.current().next, 0);
|
|
|
|
// check status and fork_filter forkhash
|
|
assert_eq!(status.forkid.hash, genesis_fork_hash);
|
|
assert_eq!(fork_filter.current().hash, genesis_fork_hash);
|
|
}
|
|
}
|