2019-08-06 03:29:27 +00:00
|
|
|
#[macro_use]
|
|
|
|
extern crate log;
|
|
|
|
|
2019-09-09 16:21:41 +00:00
|
|
|
mod parse_hex;
|
2019-09-13 22:49:39 +00:00
|
|
|
mod pycli;
|
2019-09-09 05:55:14 +00:00
|
|
|
mod transition_blocks;
|
|
|
|
|
2019-08-06 03:29:27 +00:00
|
|
|
use clap::{App, Arg, SubCommand};
|
2019-09-09 16:21:41 +00:00
|
|
|
use parse_hex::run_parse_hex;
|
2019-09-13 22:49:39 +00:00
|
|
|
use pycli::run_pycli;
|
2019-08-06 03:29:27 +00:00
|
|
|
use std::fs::File;
|
|
|
|
use std::path::PathBuf;
|
|
|
|
use std::time::{SystemTime, UNIX_EPOCH};
|
2019-09-09 05:55:14 +00:00
|
|
|
use transition_blocks::run_transition_blocks;
|
2019-08-06 03:29:27 +00:00
|
|
|
use types::{test_utils::TestingBeaconStateBuilder, EthSpec, MainnetEthSpec, MinimalEthSpec};
|
|
|
|
|
2019-09-13 22:49:39 +00:00
|
|
|
type LocalEthSpec = MinimalEthSpec;
|
|
|
|
|
2019-08-06 03:29:27 +00:00
|
|
|
fn main() {
|
|
|
|
simple_logger::init().expect("logger should initialize");
|
|
|
|
|
2019-09-09 21:05:23 +00:00
|
|
|
let matches = App::new("Lighthouse CLI Tool")
|
2019-08-06 03:29:27 +00:00
|
|
|
.version("0.1.0")
|
|
|
|
.author("Paul Hauner <paul@sigmaprime.io>")
|
2019-09-09 21:05:23 +00:00
|
|
|
.about(
|
|
|
|
"Performs various testing-related tasks, modelled after zcli. \
|
|
|
|
by @protolambda.",
|
|
|
|
)
|
2019-08-06 03:29:27 +00:00
|
|
|
.subcommand(
|
|
|
|
SubCommand::with_name("genesis_yaml")
|
|
|
|
.about("Generates a genesis YAML file")
|
|
|
|
.version("0.1.0")
|
|
|
|
.author("Paul Hauner <paul@sigmaprime.io>")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("num_validators")
|
|
|
|
.short("n")
|
|
|
|
.value_name("INTEGER")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.help("Number of initial validators."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("genesis_time")
|
|
|
|
.short("g")
|
|
|
|
.value_name("INTEGER")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(false)
|
|
|
|
.help("Eth2 genesis time (seconds since UNIX epoch)."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("spec")
|
|
|
|
.short("s")
|
|
|
|
.value_name("STRING")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.possible_values(&["minimal", "mainnet"])
|
|
|
|
.default_value("minimal")
|
|
|
|
.help("Eth2 genesis time (seconds since UNIX epoch)."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("output_file")
|
|
|
|
.short("f")
|
|
|
|
.value_name("PATH")
|
|
|
|
.takes_value(true)
|
|
|
|
.default_value("./genesis_state.yaml")
|
|
|
|
.help("Output file for generated state."),
|
|
|
|
),
|
|
|
|
)
|
2019-09-09 05:55:14 +00:00
|
|
|
.subcommand(
|
|
|
|
SubCommand::with_name("transition-blocks")
|
|
|
|
.about("Performs a state transition given a pre-state and block")
|
|
|
|
.version("0.1.0")
|
|
|
|
.author("Paul Hauner <paul@sigmaprime.io>")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("pre-state")
|
|
|
|
.value_name("BEACON_STATE")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.help("Path to a SSZ file of the pre-state."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("block")
|
|
|
|
.value_name("BEACON_BLOCK")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.help("Path to a SSZ file of the block to apply to pre-state."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("output")
|
|
|
|
.value_name("SSZ_FILE")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.default_value("./output.ssz")
|
|
|
|
.help("Path to output a SSZ file."),
|
|
|
|
),
|
|
|
|
)
|
2019-09-09 16:21:41 +00:00
|
|
|
.subcommand(
|
|
|
|
SubCommand::with_name("pretty-hex")
|
2019-09-09 21:05:23 +00:00
|
|
|
.about("Parses SSZ encoded as ASCII 0x-prefixed hex")
|
2019-09-09 16:21:41 +00:00
|
|
|
.version("0.1.0")
|
|
|
|
.author("Paul Hauner <paul@sigmaprime.io>")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("type")
|
|
|
|
.value_name("TYPE")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.possible_values(&["block"])
|
|
|
|
.help("The schema of the supplied SSZ."),
|
|
|
|
)
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("hex_ssz")
|
|
|
|
.value_name("HEX")
|
|
|
|
.takes_value(true)
|
|
|
|
.required(true)
|
|
|
|
.help("SSZ encoded as 0x-prefixed hex"),
|
|
|
|
),
|
|
|
|
)
|
2019-09-13 22:49:39 +00:00
|
|
|
.subcommand(
|
|
|
|
SubCommand::with_name("pycli")
|
|
|
|
.about("TODO")
|
|
|
|
.version("0.1.0")
|
|
|
|
.author("Paul Hauner <paul@sigmaprime.io>")
|
|
|
|
.arg(
|
|
|
|
Arg::with_name("pycli-path")
|
|
|
|
.long("pycli-path")
|
|
|
|
.short("p")
|
|
|
|
.value_name("PATH")
|
|
|
|
.takes_value(true)
|
|
|
|
.default_value("../../pycli")
|
|
|
|
.help("Path to the pycli repository."),
|
|
|
|
),
|
|
|
|
)
|
2019-08-06 03:29:27 +00:00
|
|
|
.get_matches();
|
|
|
|
|
2019-09-09 05:55:14 +00:00
|
|
|
match matches.subcommand() {
|
|
|
|
("genesis_yaml", Some(matches)) => {
|
|
|
|
let num_validators = matches
|
|
|
|
.value_of("num_validators")
|
|
|
|
.expect("slog requires num_validators")
|
|
|
|
.parse::<usize>()
|
|
|
|
.expect("num_validators must be a valid integer");
|
|
|
|
|
|
|
|
let genesis_time = if let Some(string) = matches.value_of("genesis_time") {
|
|
|
|
string
|
|
|
|
.parse::<u64>()
|
|
|
|
.expect("genesis_time must be a valid integer")
|
|
|
|
} else {
|
|
|
|
warn!("No genesis time supplied via CLI, using the current time.");
|
|
|
|
SystemTime::now()
|
|
|
|
.duration_since(UNIX_EPOCH)
|
|
|
|
.expect("should obtain time since unix epoch")
|
|
|
|
.as_secs()
|
|
|
|
};
|
|
|
|
|
|
|
|
let file = matches
|
|
|
|
.value_of("output_file")
|
|
|
|
.expect("slog requires output file")
|
|
|
|
.parse::<PathBuf>()
|
|
|
|
.expect("output_file must be a valid path");
|
|
|
|
|
|
|
|
info!(
|
|
|
|
"Creating genesis state with {} validators and genesis time {}.",
|
|
|
|
num_validators, genesis_time
|
|
|
|
);
|
|
|
|
|
|
|
|
match matches.value_of("spec").expect("spec is required by slog") {
|
|
|
|
"minimal" => genesis_yaml::<MinimalEthSpec>(num_validators, genesis_time, file),
|
|
|
|
"mainnet" => genesis_yaml::<MainnetEthSpec>(num_validators, genesis_time, file),
|
|
|
|
_ => unreachable!("guarded by slog possible_values"),
|
|
|
|
};
|
|
|
|
|
|
|
|
info!("Genesis state YAML file created. Exiting successfully.");
|
|
|
|
}
|
|
|
|
("transition-blocks", Some(matches)) => run_transition_blocks(matches)
|
|
|
|
.unwrap_or_else(|e| error!("Failed to transition blocks: {}", e)),
|
2019-09-09 16:21:41 +00:00
|
|
|
("pretty-hex", Some(matches)) => {
|
|
|
|
run_parse_hex(matches).unwrap_or_else(|e| error!("Failed to pretty print hex: {}", e))
|
|
|
|
}
|
2019-09-13 22:49:39 +00:00
|
|
|
("pycli", Some(matches)) => run_pycli::<LocalEthSpec>(matches)
|
|
|
|
.unwrap_or_else(|e| error!("Failed to run pycli: {}", e)),
|
2019-09-09 21:05:23 +00:00
|
|
|
(other, _) => error!("Unknown subcommand {}. See --help.", other),
|
2019-08-06 03:29:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Creates a genesis state and writes it to a YAML file.
|
|
|
|
fn genesis_yaml<T: EthSpec>(validator_count: usize, genesis_time: u64, output: PathBuf) {
|
|
|
|
let spec = &T::default_spec();
|
|
|
|
|
|
|
|
let builder: TestingBeaconStateBuilder<T> =
|
|
|
|
TestingBeaconStateBuilder::from_default_keypairs_file_if_exists(validator_count, spec);
|
|
|
|
|
|
|
|
let (mut state, _keypairs) = builder.build();
|
|
|
|
state.genesis_time = genesis_time;
|
|
|
|
|
|
|
|
info!("Generated state root: {:?}", state.canonical_root());
|
|
|
|
|
|
|
|
info!("Writing genesis state to {:?}", output);
|
|
|
|
|
|
|
|
let file = File::create(output.clone())
|
|
|
|
.unwrap_or_else(|e| panic!("unable to create file: {:?}. Error: {:?}", output, e));
|
|
|
|
serde_yaml::to_writer(file, &state).expect("should be able to serialize BeaconState");
|
|
|
|
}
|