//! This crate should eventually represent the structure at this repo: //! //! https://github.com/eth2-clients/eth2-testnets/tree/master/nimbus/testnet1 //! //! It is not accurate at the moment, we include extra files and we also don't support a few //! others. We are unable to conform to the repo until we have the following PR merged: //! //! https://github.com/sigp/lighthouse/pull/605 //! use eth2_config::{testnets_dir, *}; use enr::{CombinedKey, Enr}; use ssz::Decode; use std::fs::{create_dir_all, File}; use std::io::{Read, Write}; use std::path::PathBuf; use types::{BeaconState, EthSpec, EthSpecId, YamlConfig}; pub const ADDRESS_FILE: &str = "deposit_contract.txt"; pub const DEPLOY_BLOCK_FILE: &str = "deploy_block.txt"; pub const BOOT_ENR_FILE: &str = "boot_enr.yaml"; pub const GENESIS_STATE_FILE: &str = "genesis.ssz"; pub const YAML_CONFIG_FILE: &str = "config.yaml"; #[derive(Copy, Clone, Debug, PartialEq)] pub struct HardcodedNet { pub name: &'static str, pub genesis_is_known: bool, pub yaml_config: &'static [u8], pub deploy_block: &'static [u8], pub boot_enr: &'static [u8], pub genesis_state_bytes: &'static [u8], } macro_rules! define_net { ($mod: ident, $include_file: tt) => {{ use eth2_config::$mod::ETH2_NET_DIR; HardcodedNet { name: ETH2_NET_DIR.name, genesis_is_known: ETH2_NET_DIR.genesis_is_known, yaml_config: $include_file!("../", "config.yaml"), deploy_block: $include_file!("../", "deploy_block.txt"), boot_enr: $include_file!("../", "boot_enr.yaml"), genesis_state_bytes: $include_file!("../", "genesis.ssz"), } }}; } const ALTONA: HardcodedNet = define_net!(altona, include_altona_file); const MEDALLA: HardcodedNet = define_net!(medalla, include_medalla_file); const SPADINA: HardcodedNet = define_net!(spadina, include_spadina_file); const PYRMONT: HardcodedNet = define_net!(pyrmont, include_pyrmont_file); const MAINNET: HardcodedNet = define_net!(mainnet, include_mainnet_file); const TOLEDO: HardcodedNet = define_net!(toledo, include_toledo_file); const HARDCODED_NETS: &[HardcodedNet] = &[ALTONA, MEDALLA, SPADINA, PYRMONT, MAINNET, TOLEDO]; pub const DEFAULT_HARDCODED_TESTNET: &str = "mainnet"; /// Specifies an Eth2 testnet. /// /// See the crate-level documentation for more details. #[derive(Clone, PartialEq, Debug)] pub struct Eth2TestnetConfig { /// Note: instead of the block where the contract is deployed, it is acceptable to set this /// value to be the block number where the first deposit occurs. pub deposit_contract_deploy_block: u64, pub boot_enr: Option>>, pub genesis_state_bytes: Option>, pub yaml_config: Option, } impl Eth2TestnetConfig { /// When Lighthouse is built it includes zero or more "hardcoded" network specifications. This /// function allows for instantiating one of these nets by name. pub fn constant(name: &str) -> Result, String> { HARDCODED_NETS .iter() .find(|net| net.name == name) .map(Self::from_hardcoded_net) .transpose() } /// Instantiates `Self` from a `HardcodedNet`. fn from_hardcoded_net(net: &HardcodedNet) -> Result { Ok(Self { deposit_contract_deploy_block: serde_yaml::from_reader(net.deploy_block) .map_err(|e| format!("Unable to parse deploy block: {:?}", e))?, boot_enr: Some( serde_yaml::from_reader(net.boot_enr) .map_err(|e| format!("Unable to parse boot enr: {:?}", e))?, ), genesis_state_bytes: Some(net.genesis_state_bytes.to_vec()) .filter(|bytes| !bytes.is_empty()), yaml_config: Some( serde_yaml::from_reader(net.yaml_config) .map_err(|e| format!("Unable to parse yaml config: {:?}", e))?, ), }) } /// Returns an identifier that should be used for selecting an `EthSpec` instance for this /// testnet. pub fn eth_spec_id(&self) -> Result { self.yaml_config .as_ref() .ok_or_else(|| "YAML specification file missing".to_string()) .and_then(|config| { config .eth_spec_id() .ok_or_else(|| format!("Unknown CONFIG_NAME: {}", config.config_name)) }) } /// Returns `true` if this configuration contains a `BeaconState`. pub fn beacon_state_is_known(&self) -> bool { self.genesis_state_bytes.is_some() } /// Attempts to deserialize `self.beacon_state`, returning an error if it's missing or invalid. pub fn beacon_state(&self) -> Result, String> { let genesis_state_bytes = self .genesis_state_bytes .as_ref() .ok_or("Genesis state is unknown")?; BeaconState::from_ssz_bytes(genesis_state_bytes) .map_err(|e| format!("Genesis state SSZ bytes are invalid: {:?}", e)) } /// Write the files to the directory. /// /// Overwrites files if specified to do so. pub fn write_to_file(&self, base_dir: PathBuf, overwrite: bool) -> Result<(), String> { if base_dir.exists() && !overwrite { return Err("Testnet directory already exists".to_string()); } self.force_write_to_file(base_dir) } /// Write the files to the directory, even if the directory already exists. pub fn force_write_to_file(&self, base_dir: PathBuf) -> Result<(), String> { create_dir_all(&base_dir) .map_err(|e| format!("Unable to create testnet directory: {:?}", e))?; macro_rules! write_to_yaml_file { ($file: ident, $variable: expr) => { File::create(base_dir.join($file)) .map_err(|e| format!("Unable to create {}: {:?}", $file, e)) .and_then(|mut file| { let yaml = serde_yaml::to_string(&$variable) .map_err(|e| format!("Unable to YAML encode {}: {:?}", $file, e))?; // Remove the doc header from the YAML file. // // This allows us to play nice with other clients that are expecting // plain-text, not YAML. let no_doc_header = if let Some(stripped) = yaml.strip_prefix("---\n") { stripped } else { &yaml }; file.write_all(no_doc_header.as_bytes()) .map_err(|e| format!("Unable to write {}: {:?}", $file, e)) })?; }; } write_to_yaml_file!(DEPLOY_BLOCK_FILE, self.deposit_contract_deploy_block); if let Some(boot_enr) = &self.boot_enr { write_to_yaml_file!(BOOT_ENR_FILE, boot_enr); } if let Some(yaml_config) = &self.yaml_config { write_to_yaml_file!(YAML_CONFIG_FILE, yaml_config); } // The genesis state is a special case because it uses SSZ, not YAML. if let Some(genesis_state_bytes) = &self.genesis_state_bytes { let file = base_dir.join(GENESIS_STATE_FILE); File::create(&file) .map_err(|e| format!("Unable to create {:?}: {:?}", file, e)) .and_then(|mut file| { file.write_all(genesis_state_bytes) .map_err(|e| format!("Unable to write {:?}: {:?}", file, e)) })?; } Ok(()) } pub fn load(base_dir: PathBuf) -> Result { macro_rules! load_from_file { ($file: ident) => { File::open(base_dir.join($file)) .map_err(|e| format!("Unable to open {}: {:?}", $file, e)) .and_then(|file| { serde_yaml::from_reader(file) .map_err(|e| format!("Unable to parse {}: {:?}", $file, e)) })?; }; } macro_rules! optional_load_from_file { ($file: ident) => { if base_dir.join($file).exists() { Some(load_from_file!($file)) } else { None } }; } let deposit_contract_deploy_block = load_from_file!(DEPLOY_BLOCK_FILE); let boot_enr = optional_load_from_file!(BOOT_ENR_FILE); let yaml_config = optional_load_from_file!(YAML_CONFIG_FILE); // The genesis state is a special case because it uses SSZ, not YAML. let genesis_file_path = base_dir.join(GENESIS_STATE_FILE); let genesis_state_bytes = if genesis_file_path.exists() { let mut bytes = vec![]; File::open(&genesis_file_path) .map_err(|e| format!("Unable to open {:?}: {:?}", genesis_file_path, e)) .and_then(|mut file| { file.read_to_end(&mut bytes) .map_err(|e| format!("Unable to read {:?}: {:?}", file, e)) })?; Some(bytes).filter(|bytes| !bytes.is_empty()) } else { None }; Ok(Self { deposit_contract_deploy_block, boot_enr, genesis_state_bytes, yaml_config, }) } } #[cfg(test)] mod tests { use super::*; use ssz::Encode; use tempdir::TempDir; use types::{Eth1Data, Hash256, MainnetEthSpec, V012LegacyEthSpec, YamlConfig}; type E = V012LegacyEthSpec; #[test] fn hard_coded_nets_work() { for net in HARDCODED_NETS { let config = Eth2TestnetConfig::from_hardcoded_net(net).expect(&format!("{:?}", net.name)); if net.name == "mainnet" || net.name == "toledo" || net.name == "pyrmont" { // Ensure we can parse the YAML config to a chain spec. config .yaml_config .as_ref() .unwrap() .apply_to_chain_spec::(&E::default_spec()) .unwrap(); } else { // Ensure we can parse the YAML config to a chain spec. config .yaml_config .as_ref() .unwrap() .apply_to_chain_spec::(&E::default_spec()) .unwrap(); } assert_eq!( config.genesis_state_bytes.is_some(), net.genesis_is_known, "{:?}", net.name ); } } #[test] fn round_trip() { let spec = &E::default_spec(); let eth1_data = Eth1Data { deposit_root: Hash256::zero(), deposit_count: 0, block_hash: Hash256::zero(), }; // TODO: figure out how to generate ENR and add some here. let boot_enr = None; let genesis_state = Some(BeaconState::new(42, eth1_data, spec)); let yaml_config = Some(YamlConfig::from_spec::(spec)); do_test::(boot_enr, genesis_state, yaml_config); do_test::(None, None, None); } fn do_test( boot_enr: Option>>, genesis_state: Option>, yaml_config: Option, ) { let temp_dir = TempDir::new("eth2_testnet_test").expect("should create temp dir"); let base_dir = temp_dir.path().join("my_testnet"); let deposit_contract_deploy_block = 42; let testnet: Eth2TestnetConfig = Eth2TestnetConfig { deposit_contract_deploy_block, boot_enr, genesis_state_bytes: genesis_state.as_ref().map(Encode::as_ssz_bytes), yaml_config, }; testnet .write_to_file(base_dir.clone(), false) .expect("should write to file"); let decoded = Eth2TestnetConfig::load(base_dir).expect("should load struct"); assert_eq!(testnet, decoded, "should decode as encoded"); } }