2019-09-09 05:55:14 +00:00
|
|
|
use clap::ArgMatches;
|
|
|
|
use ssz::{Decode, Encode};
|
|
|
|
use state_processing::{per_block_processing, per_slot_processing, BlockSignatureStrategy};
|
|
|
|
use std::fs::File;
|
|
|
|
use std::io::prelude::*;
|
|
|
|
use std::path::PathBuf;
|
2019-12-09 06:23:56 +00:00
|
|
|
use types::{BeaconBlock, BeaconState, EthSpec};
|
2019-09-09 05:55:14 +00:00
|
|
|
|
2019-12-09 06:23:56 +00:00
|
|
|
pub fn run_transition_blocks<T: EthSpec>(matches: &ArgMatches) -> Result<(), String> {
|
2019-09-09 05:55:14 +00:00
|
|
|
let pre_state_path = matches
|
|
|
|
.value_of("pre-state")
|
|
|
|
.ok_or_else(|| "No pre-state file supplied".to_string())?
|
|
|
|
.parse::<PathBuf>()
|
|
|
|
.map_err(|e| format!("Failed to parse pre-state path: {}", e))?;
|
|
|
|
|
|
|
|
let block_path = matches
|
|
|
|
.value_of("block")
|
|
|
|
.ok_or_else(|| "No block file supplied".to_string())?
|
|
|
|
.parse::<PathBuf>()
|
|
|
|
.map_err(|e| format!("Failed to parse block path: {}", e))?;
|
|
|
|
|
|
|
|
let output_path = matches
|
|
|
|
.value_of("output")
|
|
|
|
.ok_or_else(|| "No output file supplied".to_string())?
|
|
|
|
.parse::<PathBuf>()
|
|
|
|
.map_err(|e| format!("Failed to parse output path: {}", e))?;
|
|
|
|
|
|
|
|
info!("Using minimal spec");
|
|
|
|
info!("Pre-state path: {:?}", pre_state_path);
|
|
|
|
info!("Block path: {:?}", block_path);
|
|
|
|
|
2019-12-09 06:23:56 +00:00
|
|
|
let pre_state: BeaconState<T> = load_from_ssz(pre_state_path)?;
|
|
|
|
let block: BeaconBlock<T> = load_from_ssz(block_path)?;
|
2019-09-09 05:55:14 +00:00
|
|
|
|
|
|
|
let post_state = do_transition(pre_state, block)?;
|
|
|
|
|
|
|
|
let mut output_file = File::create(output_path.clone())
|
|
|
|
.map_err(|e| format!("Unable to create output file: {:?}", e))?;
|
|
|
|
|
|
|
|
output_file
|
|
|
|
.write_all(&post_state.as_ssz_bytes())
|
|
|
|
.map_err(|e| format!("Unable to write to output file: {:?}", e))?;
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn do_transition<T: EthSpec>(
|
|
|
|
mut pre_state: BeaconState<T>,
|
|
|
|
block: BeaconBlock<T>,
|
|
|
|
) -> Result<BeaconState<T>, String> {
|
|
|
|
let spec = &T::default_spec();
|
|
|
|
|
|
|
|
pre_state
|
|
|
|
.build_all_caches(spec)
|
|
|
|
.map_err(|e| format!("Unable to build caches: {:?}", e))?;
|
|
|
|
|
|
|
|
// Transition the parent state to the block slot.
|
|
|
|
for i in pre_state.slot.as_u64()..block.slot.as_u64() {
|
|
|
|
per_slot_processing(&mut pre_state, spec)
|
|
|
|
.map_err(|e| format!("Failed to advance slot on iteration {}: {:?}", i, e))?;
|
|
|
|
}
|
|
|
|
|
|
|
|
pre_state
|
|
|
|
.build_all_caches(spec)
|
|
|
|
.map_err(|e| format!("Unable to build caches: {:?}", e))?;
|
|
|
|
|
|
|
|
per_block_processing(
|
|
|
|
&mut pre_state,
|
|
|
|
&block,
|
|
|
|
None,
|
|
|
|
BlockSignatureStrategy::VerifyIndividual,
|
|
|
|
spec,
|
|
|
|
)
|
|
|
|
.map_err(|e| format!("State transition failed: {:?}", e))?;
|
|
|
|
|
|
|
|
Ok(pre_state)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn load_from_ssz<T: Decode>(path: PathBuf) -> Result<T, String> {
|
|
|
|
let mut file =
|
|
|
|
File::open(path.clone()).map_err(|e| format!("Unable to open file {:?}: {:?}", path, e))?;
|
|
|
|
let mut bytes = vec![];
|
|
|
|
file.read_to_end(&mut bytes)
|
|
|
|
.map_err(|e| format!("Unable to read from file {:?}: {:?}", path, e))?;
|
|
|
|
T::from_ssz_bytes(&bytes).map_err(|e| format!("Ssz decode failed: {:?}", e))
|
|
|
|
}
|