mirror of
https://github.com/sigp/lighthouse.git
synced 2026-05-08 09:16:00 +00:00
Progress with account_manager deposit tools
This commit is contained in:
@@ -24,3 +24,5 @@ hex = "0.4"
|
|||||||
validator_client = { path = "../validator_client" }
|
validator_client = { path = "../validator_client" }
|
||||||
rayon = "1.2.0"
|
rayon = "1.2.0"
|
||||||
eth2_testnet = { path = "../eth2/utils/eth2_testnet" }
|
eth2_testnet = { path = "../eth2/utils/eth2_testnet" }
|
||||||
|
web3 = "0.8.0"
|
||||||
|
futures = "0.1.25"
|
||||||
|
|||||||
@@ -10,6 +10,47 @@ pub fn cli_app<'a, 'b>() -> App<'a, 'b> {
|
|||||||
.subcommand(
|
.subcommand(
|
||||||
SubCommand::with_name("new")
|
SubCommand::with_name("new")
|
||||||
.about("Create a new Ethereum 2.0 validator.")
|
.about("Create a new Ethereum 2.0 validator.")
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("send-deposits")
|
||||||
|
.long("send-deposits")
|
||||||
|
.help("If present, submit validator deposits to an eth1 endpoint")
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("eth1-endpoint")
|
||||||
|
.short("e")
|
||||||
|
.value_name("HTTP_SERVER")
|
||||||
|
.takes_value(true)
|
||||||
|
.requires("send-deposits")
|
||||||
|
.default_value("http://localhost:8545")
|
||||||
|
.help("The URL to the eth1 JSON-RPC http API."),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("deposit-contract")
|
||||||
|
.short("c")
|
||||||
|
.value_name("ADDRESS")
|
||||||
|
.takes_value(true)
|
||||||
|
.requires("send-deposits")
|
||||||
|
.conflicts_with("testnet-dir")
|
||||||
|
.help("The deposit contract for submitting deposits."),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("account-index")
|
||||||
|
.short("i")
|
||||||
|
.value_name("INDEX")
|
||||||
|
.takes_value(true)
|
||||||
|
.requires("send-deposits")
|
||||||
|
.default_value("0")
|
||||||
|
.help("The eth1 accounts[] index which will send the transaction"),
|
||||||
|
)
|
||||||
|
.arg(
|
||||||
|
Arg::with_name("testnet-dir")
|
||||||
|
.long("testnet-dir")
|
||||||
|
.value_name("DIRECTORY")
|
||||||
|
.takes_value(true)
|
||||||
|
.requires("send-deposits")
|
||||||
|
.default_value("0")
|
||||||
|
.help("The directory from which to read the deposit contract address."),
|
||||||
|
)
|
||||||
.subcommand(
|
.subcommand(
|
||||||
SubCommand::with_name("insecure")
|
SubCommand::with_name("insecure")
|
||||||
.about("Produce insecure, ephemeral validators. DO NOT USE TO STORE VALUE.")
|
.about("Produce insecure, ephemeral validators. DO NOT USE TO STORE VALUE.")
|
||||||
|
|||||||
@@ -1,13 +1,21 @@
|
|||||||
mod cli;
|
mod cli;
|
||||||
|
|
||||||
use clap::ArgMatches;
|
use clap::ArgMatches;
|
||||||
|
use deposit_contract::DEPOSIT_GAS;
|
||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
|
use eth2_testnet::Eth2TestnetDir;
|
||||||
|
use futures::{stream::unfold, Future, IntoFuture, Stream};
|
||||||
use rayon::prelude::*;
|
use rayon::prelude::*;
|
||||||
use slog::{crit, info};
|
use slog::{crit, error, info};
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use types::{ChainSpec, EthSpec};
|
use types::{ChainSpec, EthSpec};
|
||||||
use validator_client::validator_directory::{ValidatorDirectory, ValidatorDirectoryBuilder};
|
use validator_client::validator_directory::{ValidatorDirectory, ValidatorDirectoryBuilder};
|
||||||
|
use web3::{
|
||||||
|
transports::Http,
|
||||||
|
types::{Address, TransactionRequest, U256},
|
||||||
|
Web3,
|
||||||
|
};
|
||||||
|
|
||||||
pub use cli::cli_app;
|
pub use cli::cli_app;
|
||||||
|
|
||||||
@@ -110,7 +118,48 @@ fn run_new_validator_subcommand<T: EthSpec>(
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
let validators = make_validators(datadir.clone(), &methods, context.eth2_config.spec)?;
|
let validators = make_validators(datadir.clone(), &methods, &context.eth2_config.spec)?;
|
||||||
|
|
||||||
|
if matches.is_present("send-deposits") {
|
||||||
|
let eth1_endpoint = matches
|
||||||
|
.value_of("eth1-endpoint")
|
||||||
|
.ok_or_else(|| "No eth1-endpoint".to_string())?;
|
||||||
|
let account_index = matches
|
||||||
|
.value_of("account-index")
|
||||||
|
.ok_or_else(|| "No account-index".to_string())?
|
||||||
|
.parse::<usize>()
|
||||||
|
.map_err(|e| format!("Unable to parse account-index: {}", e))?;
|
||||||
|
|
||||||
|
let deposit_contract = if let Some(testnet_dir_str) = matches.value_of("testnet-dir") {
|
||||||
|
let testnet_dir = testnet_dir_str
|
||||||
|
.parse::<PathBuf>()
|
||||||
|
.map_err(|e| format!("Unable to parse testnet-dir: {}", e))?;
|
||||||
|
|
||||||
|
let eth2_testnet_dir = Eth2TestnetDir::load(testnet_dir)
|
||||||
|
.map_err(|e| format!("Failed to load testnet dir: {}", e))?;
|
||||||
|
|
||||||
|
// Convert from `types::Address` to `web3::types::Address`.
|
||||||
|
Address::from_slice(
|
||||||
|
eth2_testnet_dir
|
||||||
|
.deposit_contract_address()?
|
||||||
|
.as_fixed_bytes(),
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
matches
|
||||||
|
.value_of("deposit-contract")
|
||||||
|
.ok_or_else(|| "No deposit-contract".to_string())?
|
||||||
|
.parse::<Address>()
|
||||||
|
.map_err(|e| format!("Unable to parse deposit-contract: {}", e))?
|
||||||
|
};
|
||||||
|
|
||||||
|
context.executor.spawn(deposit_validators(
|
||||||
|
context.clone(),
|
||||||
|
eth1_endpoint.to_string(),
|
||||||
|
deposit_contract,
|
||||||
|
validators.clone(),
|
||||||
|
account_index,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
info!(
|
info!(
|
||||||
log,
|
log,
|
||||||
@@ -122,11 +171,90 @@ fn run_new_validator_subcommand<T: EthSpec>(
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn deposit_validators<E: EthSpec>(
|
||||||
|
context: RuntimeContext<E>,
|
||||||
|
eth1_endpoint: String,
|
||||||
|
deposit_contract: Address,
|
||||||
|
validators: Vec<ValidatorDirectory>,
|
||||||
|
account_index: usize,
|
||||||
|
) -> impl Future<Item = (), Error = ()> {
|
||||||
|
let deposit_amount = context.eth2_config.spec.max_effective_balance;
|
||||||
|
let log = context.log.clone();
|
||||||
|
|
||||||
|
Http::new(ð1_endpoint)
|
||||||
|
.map_err(|e| format!("Failed to start web3 HTTP transport: {:?}", e))
|
||||||
|
.into_future()
|
||||||
|
.and_then(move |(_event_loop, transport)| {
|
||||||
|
let web3 = Web3::new(transport);
|
||||||
|
|
||||||
|
unfold(validators.into_iter(), move |mut validators| {
|
||||||
|
let web3 = web3.clone();
|
||||||
|
|
||||||
|
validators.next().map(move |validator| {
|
||||||
|
deposit_validator(
|
||||||
|
web3,
|
||||||
|
deposit_contract,
|
||||||
|
&validator,
|
||||||
|
deposit_amount,
|
||||||
|
account_index,
|
||||||
|
)
|
||||||
|
.map(|()| ((), validators))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
})
|
||||||
|
.map_err(move |e| error!(log, "Error whilst depositing validator"; "error" => e))
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deposit_validator(
|
||||||
|
web3: Web3<Http>,
|
||||||
|
deposit_contract: Address,
|
||||||
|
validator: &ValidatorDirectory,
|
||||||
|
deposit_amount: u64,
|
||||||
|
account_index: usize,
|
||||||
|
) -> impl Future<Item = (), Error = String> {
|
||||||
|
let web3_1 = web3.clone();
|
||||||
|
|
||||||
|
let deposit_data = validator
|
||||||
|
.deposit_data
|
||||||
|
.clone()
|
||||||
|
.expect("Validators must have a deposit data");
|
||||||
|
|
||||||
|
web3.eth()
|
||||||
|
.accounts()
|
||||||
|
.map_err(|e| format!("Failed to get accounts: {:?}", e))
|
||||||
|
.and_then(move |accounts| {
|
||||||
|
accounts
|
||||||
|
.get(account_index)
|
||||||
|
.cloned()
|
||||||
|
.ok_or_else(|| "Insufficient accounts for deposit".to_string())
|
||||||
|
})
|
||||||
|
.and_then(move |from| {
|
||||||
|
let tx_request = TransactionRequest {
|
||||||
|
from,
|
||||||
|
to: Some(deposit_contract),
|
||||||
|
gas: Some(U256::from(DEPOSIT_GAS)),
|
||||||
|
gas_price: None,
|
||||||
|
value: Some(U256::from(deposit_amount)),
|
||||||
|
data: Some(deposit_data.into()),
|
||||||
|
nonce: None,
|
||||||
|
condition: None,
|
||||||
|
};
|
||||||
|
|
||||||
|
web3_1
|
||||||
|
.eth()
|
||||||
|
.send_transaction(tx_request)
|
||||||
|
.map_err(|e| format!("Failed to call deposit fn: {:?}", e))
|
||||||
|
})
|
||||||
|
.map(|_| ())
|
||||||
|
}
|
||||||
|
|
||||||
/// Produces a validator directory for each of the key generation methods provided in `methods`.
|
/// Produces a validator directory for each of the key generation methods provided in `methods`.
|
||||||
fn make_validators(
|
fn make_validators(
|
||||||
datadir: PathBuf,
|
datadir: PathBuf,
|
||||||
methods: &[KeygenMethod],
|
methods: &[KeygenMethod],
|
||||||
spec: ChainSpec,
|
spec: &ChainSpec,
|
||||||
) -> Result<Vec<ValidatorDirectory>, String> {
|
) -> Result<Vec<ValidatorDirectory>, String> {
|
||||||
methods
|
methods
|
||||||
.par_iter()
|
.par_iter()
|
||||||
|
|||||||
@@ -275,7 +275,8 @@ fn process_testnet_subcommand(
|
|||||||
epoch: Epoch::new(0),
|
epoch: Epoch::new(0),
|
||||||
};
|
};
|
||||||
|
|
||||||
client_config.eth1.deposit_contract_address = eth2_testnet_dir.deposit_contract_address;
|
client_config.eth1.deposit_contract_address =
|
||||||
|
format!("{}", eth2_testnet_dir.deposit_contract_address()?);
|
||||||
client_config.eth1.deposit_contract_deploy_block =
|
client_config.eth1.deposit_contract_deploy_block =
|
||||||
eth2_testnet_dir.deposit_contract_deploy_block;
|
eth2_testnet_dir.deposit_contract_deploy_block;
|
||||||
client_config.eth1.follow_distance = 16;
|
client_config.eth1.follow_distance = 16;
|
||||||
|
|||||||
@@ -12,3 +12,4 @@ tempdir = "0.3"
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
serde = "1.0"
|
serde = "1.0"
|
||||||
serde_json = "^1.0"
|
serde_json = "^1.0"
|
||||||
|
types = { path = "../../types"}
|
||||||
|
|||||||
@@ -9,6 +9,7 @@
|
|||||||
|
|
||||||
use std::fs::{create_dir_all, File};
|
use std::fs::{create_dir_all, File};
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
use types::Address;
|
||||||
|
|
||||||
pub const ADDRESS_FILE: &str = "deposit_contract.txt";
|
pub const ADDRESS_FILE: &str = "deposit_contract.txt";
|
||||||
pub const DEPLOY_BLOCK_FILE: &str = "deploy_block.txt";
|
pub const DEPLOY_BLOCK_FILE: &str = "deploy_block.txt";
|
||||||
@@ -16,7 +17,7 @@ pub const MIN_GENESIS_TIME_FILE: &str = "min_genesis_time.txt";
|
|||||||
|
|
||||||
#[derive(Clone, PartialEq, Debug)]
|
#[derive(Clone, PartialEq, Debug)]
|
||||||
pub struct Eth2TestnetDir {
|
pub struct Eth2TestnetDir {
|
||||||
pub deposit_contract_address: String,
|
deposit_contract_address: String,
|
||||||
pub deposit_contract_deploy_block: u64,
|
pub deposit_contract_deploy_block: u64,
|
||||||
pub min_genesis_time: u64,
|
pub min_genesis_time: u64,
|
||||||
}
|
}
|
||||||
@@ -91,6 +92,16 @@ impl Eth2TestnetDir {
|
|||||||
min_genesis_time,
|
min_genesis_time,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn deposit_contract_address(&self) -> Result<Address, String> {
|
||||||
|
if self.deposit_contract_address.starts_with("0x") {
|
||||||
|
self.deposit_contract_address[2..]
|
||||||
|
.parse()
|
||||||
|
.map_err(|e| format!("Corrupted address, unable to parse: {:?}", e))
|
||||||
|
} else {
|
||||||
|
Err("Corrupted address, must start with 0x".to_string())
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
|
|||||||
@@ -136,10 +136,6 @@ fn run<E: EthSpec>(
|
|||||||
let runtime_context = environment.core_context();
|
let runtime_context = environment.core_context();
|
||||||
|
|
||||||
account_manager::run(sub_matches, runtime_context);
|
account_manager::run(sub_matches, runtime_context);
|
||||||
|
|
||||||
// Exit early if the account manager was run. It does not use the tokio executor, no need
|
|
||||||
// to wait for it to shutdown.
|
|
||||||
return Ok(());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
let beacon_node = if let Some(sub_matches) = matches.subcommand_matches("beacon_node") {
|
let beacon_node = if let Some(sub_matches) = matches.subcommand_matches("beacon_node") {
|
||||||
|
|||||||
Reference in New Issue
Block a user