mirror of
https://github.com/sigp/lighthouse.git
synced 2026-05-07 16:55:46 +00:00
Port account_manager to stable futures (#1121)
* Port account_manager to stable futures * Run async fns in tokio environment
This commit is contained in:
@@ -25,6 +25,7 @@ validator_client = { path = "../validator_client" }
|
|||||||
rayon = "1.3.0"
|
rayon = "1.3.0"
|
||||||
eth2_testnet_config = { path = "../eth2/utils/eth2_testnet_config" }
|
eth2_testnet_config = { path = "../eth2/utils/eth2_testnet_config" }
|
||||||
web3 = "0.10.0"
|
web3 = "0.10.0"
|
||||||
futures = "0.3.4"
|
futures = {version = "0.3.4", features = ["compat"]}
|
||||||
clap_utils = { path = "../eth2/utils/clap_utils" }
|
clap_utils = { path = "../eth2/utils/clap_utils" }
|
||||||
tokio = "0.2.20"
|
# reduce feature set
|
||||||
|
tokio = {version = "0.2.20", features = ["full"]}
|
||||||
|
|||||||
@@ -1,15 +1,11 @@
|
|||||||
use clap::{App, Arg, ArgMatches};
|
use clap::{App, Arg, ArgMatches};
|
||||||
use clap_utils;
|
use clap_utils;
|
||||||
use environment::Environment;
|
use environment::Environment;
|
||||||
use futures::{
|
use futures::compat::Future01CompatExt;
|
||||||
future::{self, Loop},
|
|
||||||
Future,
|
|
||||||
};
|
|
||||||
use slog::{info, Logger};
|
use slog::{info, Logger};
|
||||||
use std::fs;
|
use std::fs;
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{delay_until, Duration, Instant};
|
||||||
use tokio::time::Delay;
|
|
||||||
use types::EthSpec;
|
use types::EthSpec;
|
||||||
use validator_client::validator_directory::ValidatorDirectoryBuilder;
|
use validator_client::validator_directory::ValidatorDirectoryBuilder;
|
||||||
use web3::{
|
use web3::{
|
||||||
@@ -80,7 +76,10 @@ pub fn cli_app<'a, 'b>() -> App<'a, 'b> {
|
|||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn cli_run<T: EthSpec>(matches: &ArgMatches, mut env: Environment<T>) -> Result<(), String> {
|
pub fn cli_run<T: EthSpec>(
|
||||||
|
matches: &ArgMatches<'_>,
|
||||||
|
mut env: Environment<T>,
|
||||||
|
) -> Result<(), String> {
|
||||||
let spec = env.core_context().eth2_config.spec;
|
let spec = env.core_context().eth2_config.spec;
|
||||||
let log = env.core_context().log;
|
let log = env.core_context().log;
|
||||||
|
|
||||||
@@ -138,12 +137,13 @@ pub fn cli_run<T: EthSpec>(matches: &ArgMatches, mut env: Environment<T>) -> Res
|
|||||||
let tx_hash_log = log.clone();
|
let tx_hash_log = log.clone();
|
||||||
|
|
||||||
env.runtime()
|
env.runtime()
|
||||||
.block_on(
|
.block_on(async {
|
||||||
ValidatorDirectoryBuilder::default()
|
ValidatorDirectoryBuilder::default()
|
||||||
.spec(spec.clone())
|
.spec(spec.clone())
|
||||||
.custom_deposit_amount(deposit_gwei)
|
.custom_deposit_amount(deposit_gwei)
|
||||||
.thread_random_keypairs()
|
.thread_random_keypairs()
|
||||||
.submit_eth1_deposit(web3.clone(), from_address, deposit_contract)
|
.submit_eth1_deposit(web3.clone(), from_address, deposit_contract)
|
||||||
|
.await
|
||||||
.map(move |(builder, tx_hash)| {
|
.map(move |(builder, tx_hash)| {
|
||||||
info!(
|
info!(
|
||||||
tx_hash_log,
|
tx_hash_log,
|
||||||
@@ -152,8 +152,8 @@ pub fn cli_run<T: EthSpec>(matches: &ArgMatches, mut env: Environment<T>) -> Res
|
|||||||
"index" => format!("{}/{}", i + 1, n),
|
"index" => format!("{}/{}", i + 1, n),
|
||||||
);
|
);
|
||||||
builder
|
builder
|
||||||
}),
|
})
|
||||||
)?
|
})?
|
||||||
.create_directory(validator_dir.clone())?
|
.create_directory(validator_dir.clone())?
|
||||||
.write_keypair_files()?
|
.write_keypair_files()?
|
||||||
.write_eth1_data_file()?
|
.write_eth1_data_file()?
|
||||||
@@ -183,17 +183,19 @@ fn existing_validator_count(validator_dir: &PathBuf) -> Result<usize, String> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Run a poll on the `eth_syncing` endpoint, blocking until the node is synced.
|
/// Run a poll on the `eth_syncing` endpoint, blocking until the node is synced.
|
||||||
fn poll_until_synced<T>(web3: Web3<T>, log: Logger) -> impl Future<Item = (), Error = String> + Send
|
async fn poll_until_synced<T>(web3: Web3<T>, log: Logger) -> Result<(), String>
|
||||||
where
|
where
|
||||||
T: Transport + Send + 'static,
|
T: Transport + Send + 'static,
|
||||||
<T as Transport>::Out: Send,
|
<T as Transport>::Out: Send,
|
||||||
{
|
{
|
||||||
loop_fn((web3.clone(), log.clone()), move |(web3, log)| {
|
loop {
|
||||||
web3.clone()
|
let sync_state = web3
|
||||||
|
.clone()
|
||||||
.eth()
|
.eth()
|
||||||
.syncing()
|
.syncing()
|
||||||
.map_err(|e| format!("Unable to read syncing state from eth1 node: {:?}", e))
|
.compat()
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(move |sync_state| {
|
.await
|
||||||
|
.map_err(|e| format!("Unable to read syncing state from eth1 node: {:?}", e))?;
|
||||||
match sync_state {
|
match sync_state {
|
||||||
SyncState::Syncing(SyncInfo {
|
SyncState::Syncing(SyncInfo {
|
||||||
current_block,
|
current_block,
|
||||||
@@ -207,49 +209,33 @@ where
|
|||||||
"current_block" => format!("{}", current_block),
|
"current_block" => format!("{}", current_block),
|
||||||
);
|
);
|
||||||
|
|
||||||
Box::new(
|
delay_until(Instant::now() + SYNCING_STATE_RETRY_DELAY).await;
|
||||||
Delay::new(Instant::now() + SYNCING_STATE_RETRY_DELAY)
|
|
||||||
.map_err(|e| format!("Failed to trigger delay: {:?}", e))
|
|
||||||
.and_then(|_| future::ok(Loop::Continue((web3, log)))),
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
SyncState::NotSyncing => Box::new(
|
SyncState::NotSyncing => {
|
||||||
web3.clone()
|
let block_number = web3
|
||||||
|
.clone()
|
||||||
.eth()
|
.eth()
|
||||||
.block_number()
|
.block_number()
|
||||||
.map_err(|e| {
|
.compat()
|
||||||
format!("Unable to read block number from eth1 node: {:?}", e)
|
.await
|
||||||
})
|
.map_err(|e| format!("Unable to read block number from eth1 node: {:?}", e))?;
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(
|
|
||||||
|block_number| {
|
|
||||||
if block_number > 0.into() {
|
if block_number > 0.into() {
|
||||||
info!(
|
info!(
|
||||||
log,
|
log,
|
||||||
"Eth1 node is synced";
|
"Eth1 node is synced";
|
||||||
"head_block" => format!("{}", block_number),
|
"head_block" => format!("{}", block_number),
|
||||||
);
|
);
|
||||||
Box::new(future::ok(Loop::Break((web3, log))))
|
break;
|
||||||
} else {
|
} else {
|
||||||
Box::new(
|
delay_until(Instant::now() + SYNCING_STATE_RETRY_DELAY).await;
|
||||||
Delay::new(Instant::now() + SYNCING_STATE_RETRY_DELAY)
|
|
||||||
.map_err(|e| {
|
|
||||||
format!("Failed to trigger delay: {:?}", e)
|
|
||||||
})
|
|
||||||
.and_then(|_| {
|
|
||||||
info!(
|
info!(
|
||||||
log,
|
log,
|
||||||
"Waiting for eth1 node to sync";
|
"Waiting for eth1 node to sync";
|
||||||
"current_block" => 0,
|
"current_block" => 0,
|
||||||
);
|
);
|
||||||
future::ok(Loop::Continue((web3, log)))
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
})
|
}
|
||||||
.map(|_| ())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,8 @@ use clap::ArgMatches;
|
|||||||
use deposit_contract::DEPOSIT_GAS;
|
use deposit_contract::DEPOSIT_GAS;
|
||||||
use environment::{Environment, RuntimeContext};
|
use environment::{Environment, RuntimeContext};
|
||||||
use eth2_testnet_config::Eth2TestnetConfig;
|
use eth2_testnet_config::Eth2TestnetConfig;
|
||||||
use futures::{future, Future, IntoFuture, Stream};
|
use futures::compat::Future01CompatExt;
|
||||||
|
use futures::{FutureExt, StreamExt};
|
||||||
use rayon::prelude::*;
|
use rayon::prelude::*;
|
||||||
use slog::{error, info, Logger};
|
use slog::{error, info, Logger};
|
||||||
use std::fs;
|
use std::fs;
|
||||||
@@ -23,7 +24,10 @@ use web3::{
|
|||||||
pub use cli::cli_app;
|
pub use cli::cli_app;
|
||||||
|
|
||||||
/// Run the account manager, returning an error if the operation did not succeed.
|
/// Run the account manager, returning an error if the operation did not succeed.
|
||||||
pub fn run<T: EthSpec>(matches: &ArgMatches, mut env: Environment<T>) -> Result<(), String> {
|
pub fn run<T: EthSpec>(
|
||||||
|
matches: &ArgMatches<'_>,
|
||||||
|
mut env: Environment<T>,
|
||||||
|
) -> Result<(), String> {
|
||||||
let context = env.core_context();
|
let context = env.core_context();
|
||||||
let log = context.log.clone();
|
let log = context.log.clone();
|
||||||
|
|
||||||
@@ -292,7 +296,7 @@ fn make_validators(
|
|||||||
///
|
///
|
||||||
/// Returns success as soon as the eth1 endpoint accepts the transaction (i.e., does not wait for
|
/// Returns success as soon as the eth1 endpoint accepts the transaction (i.e., does not wait for
|
||||||
/// transaction success/revert).
|
/// transaction success/revert).
|
||||||
fn deposit_validators<E: EthSpec>(
|
async fn deposit_validators<E: EthSpec>(
|
||||||
context: RuntimeContext<E>,
|
context: RuntimeContext<E>,
|
||||||
eth1_endpoint: String,
|
eth1_endpoint: String,
|
||||||
deposit_contract: Address,
|
deposit_contract: Address,
|
||||||
@@ -300,123 +304,117 @@ fn deposit_validators<E: EthSpec>(
|
|||||||
account_index: usize,
|
account_index: usize,
|
||||||
deposit_value: u64,
|
deposit_value: u64,
|
||||||
password: Option<String>,
|
password: Option<String>,
|
||||||
) -> impl Future<Item = (), Error = ()> {
|
) -> Result<(), ()> {
|
||||||
let log_1 = context.log.clone();
|
let log_1 = context.log.clone();
|
||||||
let log_2 = context.log.clone();
|
let log_2 = context.log.clone();
|
||||||
|
|
||||||
Http::new(ð1_endpoint)
|
let (event_loop, transport) = Http::new(ð1_endpoint).map_err(move |e| {
|
||||||
.map_err(move |e| {
|
|
||||||
error!(
|
error!(
|
||||||
log_1,
|
log_1,
|
||||||
"Failed to start web3 HTTP transport";
|
"Failed to start web3 HTTP transport";
|
||||||
"error" => format!("{:?}", e)
|
"error" => format!("{:?}", e)
|
||||||
)
|
)
|
||||||
})
|
})?;
|
||||||
.into_future()
|
|
||||||
/*
|
/*
|
||||||
* Loop through the validator directories and submit the deposits.
|
* Loop through the validator directories and submit the deposits.
|
||||||
*/
|
*/
|
||||||
.and_then(move |(event_loop, transport)| {
|
|
||||||
let web3 = Web3::new(transport);
|
let web3 = Web3::new(transport);
|
||||||
|
|
||||||
futures::stream::iter_ok(validators)
|
let _ = futures::stream::iter(validators)
|
||||||
.for_each(move |validator| {
|
.for_each(|validator| async {
|
||||||
let web3 = web3.clone();
|
let web3 = web3.clone();
|
||||||
let log = log_2.clone();
|
let log = log_2.clone();
|
||||||
let password = password.clone();
|
let password = password.clone();
|
||||||
|
|
||||||
deposit_validator(
|
let _ = deposit_validator(
|
||||||
web3,
|
web3,
|
||||||
deposit_contract,
|
deposit_contract,
|
||||||
&validator,
|
validator,
|
||||||
deposit_value,
|
deposit_value,
|
||||||
account_index,
|
account_index,
|
||||||
password,
|
password,
|
||||||
log,
|
log,
|
||||||
)
|
)
|
||||||
|
.await;
|
||||||
})
|
})
|
||||||
.map(|_| event_loop)
|
.map(|_| event_loop)
|
||||||
})
|
// // Web3 gives errors if the event loop is dropped whilst performing requests.
|
||||||
// Web3 gives errors if the event loop is dropped whilst performing requests.
|
.map(drop);
|
||||||
.map(drop)
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// For the given `ValidatorDirectory`, submit a deposit transaction to the `web3` node.
|
/// For the given `ValidatorDirectory`, submit a deposit transaction to the `web3` node.
|
||||||
///
|
///
|
||||||
/// Returns success as soon as the eth1 endpoint accepts the transaction (i.e., does not wait for
|
/// Returns success as soon as the eth1 endpoint accepts the transaction (i.e., does not wait for
|
||||||
/// transaction success/revert).
|
/// transaction success/revert).
|
||||||
fn deposit_validator(
|
async fn deposit_validator(
|
||||||
web3: Web3<Http>,
|
web3: Web3<Http>,
|
||||||
deposit_contract: Address,
|
deposit_contract: Address,
|
||||||
validator: &ValidatorDirectory,
|
validator: ValidatorDirectory,
|
||||||
deposit_amount: u64,
|
deposit_amount: u64,
|
||||||
account_index: usize,
|
account_index: usize,
|
||||||
password_opt: Option<String>,
|
password_opt: Option<String>,
|
||||||
log: Logger,
|
log: Logger,
|
||||||
) -> impl Future<Item = (), Error = ()> {
|
) -> Result<(), ()> {
|
||||||
validator
|
let voting_keypair = validator
|
||||||
.voting_keypair
|
.voting_keypair
|
||||||
.clone()
|
.clone()
|
||||||
.ok_or_else(|| error!(log, "Validator does not have voting keypair"))
|
.ok_or_else(|| error!(log, "Validator does not have voting keypair"))?;
|
||||||
.and_then(|voting_keypair| {
|
|
||||||
validator
|
let deposit_data = validator
|
||||||
.deposit_data
|
.deposit_data
|
||||||
.clone()
|
.clone()
|
||||||
.ok_or_else(|| error!(log, "Validator does not have deposit data"))
|
.ok_or_else(|| error!(log, "Validator does not have deposit data"))?;
|
||||||
.map(|deposit_data| (voting_keypair, deposit_data))
|
|
||||||
})
|
|
||||||
.into_future()
|
|
||||||
.and_then(move |(voting_keypair, deposit_data)| {
|
|
||||||
let pubkey_1 = voting_keypair.pk.clone();
|
let pubkey_1 = voting_keypair.pk.clone();
|
||||||
let pubkey_2 = voting_keypair.pk;
|
let pubkey_2 = voting_keypair.pk;
|
||||||
|
|
||||||
let web3_1 = web3.clone();
|
|
||||||
let web3_2 = web3.clone();
|
|
||||||
|
|
||||||
let log_1 = log.clone();
|
let log_1 = log.clone();
|
||||||
let log_2 = log.clone();
|
let log_2 = log.clone();
|
||||||
|
|
||||||
web3.eth()
|
// TODO: creating a future to extract the Error type
|
||||||
|
// check if there's a better way
|
||||||
|
let future = async move {
|
||||||
|
let accounts = web3
|
||||||
|
.eth()
|
||||||
.accounts()
|
.accounts()
|
||||||
.map_err(|e| format!("Failed to get accounts: {:?}", e))
|
.compat()
|
||||||
.and_then(move |accounts| {
|
.await
|
||||||
accounts
|
.map_err(|e| format!("Failed to get accounts: {:?}", e))?;
|
||||||
|
|
||||||
|
let from_address = accounts
|
||||||
.get(account_index)
|
.get(account_index)
|
||||||
.cloned()
|
.cloned()
|
||||||
.ok_or_else(|| "Insufficient accounts for deposit".to_string())
|
.ok_or_else(|| "Insufficient accounts for deposit".to_string())?;
|
||||||
})
|
|
||||||
/*
|
/*
|
||||||
* If a password was supplied, unlock the account.
|
* If a password was supplied, unlock the account.
|
||||||
*/
|
*/
|
||||||
.and_then(move |from_address| {
|
let from = if let Some(password) = password_opt {
|
||||||
let future: Box<dyn Future<Item = Address, Error = String> + Send> =
|
|
||||||
if let Some(password) = password_opt {
|
|
||||||
// Unlock for only a single transaction.
|
// Unlock for only a single transaction.
|
||||||
let duration = None;
|
let duration = None;
|
||||||
|
|
||||||
let future = web3_1
|
let result = web3
|
||||||
.personal()
|
.personal()
|
||||||
.unlock_account(from_address, &password, duration)
|
.unlock_account(from_address, &password, duration)
|
||||||
.then(move |result| match result {
|
.compat()
|
||||||
Ok(true) => Ok(from_address),
|
.await;
|
||||||
|
match result {
|
||||||
|
Ok(true) => from_address,
|
||||||
Ok(false) => {
|
Ok(false) => {
|
||||||
Err("Eth1 node refused to unlock account. Check password."
|
return Err::<(), String>(
|
||||||
.to_string())
|
"Eth1 node refused to unlock account. Check password.".to_string(),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
Err(e) => return Err::<(), String>(format!("Eth1 unlock request failed: {:?}", e)),
|
||||||
}
|
}
|
||||||
Err(e) => Err(format!("Eth1 unlock request failed: {:?}", e)),
|
|
||||||
});
|
|
||||||
|
|
||||||
Box::new(future)
|
|
||||||
} else {
|
} else {
|
||||||
Box::new(future::ok(from_address))
|
from_address
|
||||||
};
|
};
|
||||||
|
|
||||||
future
|
|
||||||
})
|
|
||||||
/*
|
/*
|
||||||
* Submit the deposit transaction.
|
* Submit the deposit transaction.
|
||||||
*/
|
*/
|
||||||
.and_then(move |from| {
|
|
||||||
let tx_request = TransactionRequest {
|
let tx_request = TransactionRequest {
|
||||||
from,
|
from,
|
||||||
to: Some(deposit_contract),
|
to: Some(deposit_contract),
|
||||||
@@ -428,28 +426,30 @@ fn deposit_validator(
|
|||||||
condition: None,
|
condition: None,
|
||||||
};
|
};
|
||||||
|
|
||||||
web3_2
|
let tx = web3
|
||||||
.eth()
|
.eth()
|
||||||
.send_transaction(tx_request)
|
.send_transaction(tx_request)
|
||||||
.map_err(|e| format!("Failed to call deposit fn: {:?}", e))
|
.compat()
|
||||||
})
|
.await
|
||||||
.map(move |tx| {
|
.map_err(|e| format!("Failed to call deposit fn: {:?}", e))?;
|
||||||
info!(
|
info!(
|
||||||
log_1,
|
log_1,
|
||||||
"Validator deposit successful";
|
"Validator deposit successful";
|
||||||
"eth1_tx_hash" => format!("{:?}", tx),
|
"eth1_tx_hash" => format!("{:?}", tx),
|
||||||
"validator_voting_pubkey" => format!("{:?}", pubkey_1)
|
"validator_voting_pubkey" => format!("{:?}", pubkey_1)
|
||||||
)
|
);
|
||||||
})
|
Ok(())
|
||||||
.map_err(move |e| {
|
};
|
||||||
|
|
||||||
|
future.await.map_err(move |e| {
|
||||||
error!(
|
error!(
|
||||||
log_2,
|
log_2,
|
||||||
"Validator deposit_failed";
|
"Validator deposit_failed";
|
||||||
"error" => e,
|
"error" => e,
|
||||||
"validator_voting_pubkey" => format!("{:?}", pubkey_2)
|
"validator_voting_pubkey" => format!("{:?}", pubkey_2)
|
||||||
)
|
);
|
||||||
})
|
})?;
|
||||||
})
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Converts gwei to wei.
|
/// Converts gwei to wei.
|
||||||
|
|||||||
@@ -304,11 +304,11 @@ impl ValidatorDirectoryBuilder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub async fn submit_eth1_deposit<T: Transport>(
|
pub async fn submit_eth1_deposit<T: Transport>(
|
||||||
&self,
|
self,
|
||||||
web3: Web3<T>,
|
web3: Web3<T>,
|
||||||
from: Address,
|
from: Address,
|
||||||
deposit_contract: Address,
|
deposit_contract: Address,
|
||||||
) -> Result<Hash256, String> {
|
) -> Result<(Self, Hash256), String> {
|
||||||
let (deposit_data, deposit_amount) = self.get_deposit_data()?;
|
let (deposit_data, deposit_amount) = self.get_deposit_data()?;
|
||||||
web3.eth()
|
web3.eth()
|
||||||
.send_transaction(TransactionRequest {
|
.send_transaction(TransactionRequest {
|
||||||
@@ -324,6 +324,7 @@ impl ValidatorDirectoryBuilder {
|
|||||||
.compat()
|
.compat()
|
||||||
.await
|
.await
|
||||||
.map_err(|e| format!("Failed to send transaction: {:?}", e))
|
.map_err(|e| format!("Failed to send transaction: {:?}", e))
|
||||||
|
.map(|tx| (self, tx))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn build(self) -> Result<ValidatorDirectory, String> {
|
pub fn build(self) -> Result<ValidatorDirectory, String> {
|
||||||
|
|||||||
Reference in New Issue
Block a user