mirror of
https://github.com/sigp/lighthouse.git
synced 2026-05-08 17:26:04 +00:00
Port validator_client to stable futures (#1114)
* Add PH & MS slot clock changes * Account for genesis time * Add progress on duties refactor * Add simple is_aggregator bool to val subscription * Start work on attestation_verification.rs * Add progress on ObservedAttestations * Progress with ObservedAttestations * Fix tests * Add observed attestations to the beacon chain * Add attestation observation to processing code * Add progress on attestation verification * Add first draft of ObservedAttesters * Add more tests * Add observed attesters to beacon chain * Add observers to attestation processing * Add more attestation verification * Create ObservedAggregators map * Remove commented-out code * Add observed aggregators into chain * Add progress * Finish adding features to attestation verification * Ensure beacon chain compiles * Link attn verification into chain * Integrate new attn verification in chain * Remove old attestation processing code * Start trying to fix beacon_chain tests * Split adding into pools into two functions * Add aggregation to harness * Get test harness working again * Adjust the number of aggregators for test harness * Fix edge-case in harness * Integrate new attn processing in network * Fix compile bug in validator_client * Update validator API endpoints * Fix aggreagation in test harness * Fix enum thing * Fix attestation observation bug: * Patch failing API tests * Start adding comments to attestation verification * Remove unused attestation field * Unify "is block known" logic * Update comments * Supress fork choice errors for network processing * Add todos * Tidy * Add gossip attn tests * Disallow test harness to produce old attns * Comment out in-progress tests * Partially address pruning tests * Fix failing store test * Add aggregate tests * Add comments about which spec conditions we check * Dont re-aggregate * Split apart test harness attn production * Fix compile error in network * Make progress on commented-out test * Fix skipping attestation test * Add fork choice verification tests * Tidy attn tests, remove dead code * Remove some accidentally added code * Fix clippy lint * Rename test file * Add block tests, add cheap block proposer check * Rename block testing file * Add observed_block_producers * Tidy * Switch around block signature verification * Finish block testing * Remove gossip from signature tests * First pass of self review * Fix deviation in spec * Update test spec tags * Start moving over to hashset * Finish moving observed attesters to hashmap * Move aggregation pool over to hashmap * Make fc attn borrow again * Fix rest_api compile error * Fix missing comments * Fix monster test * Uncomment increasing slots test * Address remaining comments * Remove unsafe, use cfg test * Remove cfg test flag * Fix dodgy comment * Revert "Update hashmap hashset to stable futures" This reverts commitd432378a3c. * Revert "Adds panic test to hashset delay" This reverts commit281502396f. * Ported attestation_service * Ported duties_service * Ported fork_service * More ports * Port block_service * Minor fixes * VC compiles * Update TODOS * Borrow self where possible * Ignore aggregates that are already known. * Unify aggregator modulo logic * Fix typo in logs * Refactor validator subscription logic * Avoid reproducing selection proof * Skip HTTP call if no subscriptions * Rename DutyAndState -> DutyAndProof * Tidy logs * Print root as dbg * Fix compile errors in tests * Fix compile error in test * Re-Fix attestation and duties service * Minor fixes Co-authored-by: Paul Hauner <paul@paulhauner.com>
This commit is contained in:
1
Cargo.lock
generated
1
Cargo.lock
generated
@@ -5253,7 +5253,6 @@ dependencies = [
|
|||||||
"slot_clock",
|
"slot_clock",
|
||||||
"tempdir",
|
"tempdir",
|
||||||
"tokio 0.2.20",
|
"tokio 0.2.20",
|
||||||
"tokio-timer 0.2.13",
|
|
||||||
"tree_hash",
|
"tree_hash",
|
||||||
"types",
|
"types",
|
||||||
"web3",
|
"web3",
|
||||||
|
|||||||
@@ -23,11 +23,10 @@ serde_json = "1.0.52"
|
|||||||
slog = { version = "2.5.2", features = ["max_level_trace", "release_max_level_trace"] }
|
slog = { version = "2.5.2", features = ["max_level_trace", "release_max_level_trace"] }
|
||||||
slog-async = "2.5.0"
|
slog-async = "2.5.0"
|
||||||
slog-term = "2.5.0"
|
slog-term = "2.5.0"
|
||||||
tokio = "0.2.20"
|
tokio = {version = "0.2.20", features = ["time"]}
|
||||||
tokio-timer = "0.2.13"
|
|
||||||
error-chain = "0.12.2"
|
error-chain = "0.12.2"
|
||||||
bincode = "1.2.1"
|
bincode = "1.2.1"
|
||||||
futures = "0.3.4"
|
futures = {version ="0.3.4", features = ["compat"]}
|
||||||
dirs = "2.0.2"
|
dirs = "2.0.2"
|
||||||
logging = { path = "../eth2/utils/logging" }
|
logging = { path = "../eth2/utils/logging" }
|
||||||
environment = { path = "../lighthouse/environment" }
|
environment = { path = "../lighthouse/environment" }
|
||||||
|
|||||||
@@ -4,15 +4,14 @@ use crate::{
|
|||||||
};
|
};
|
||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use futures::{future, Future, Stream};
|
use futures::{FutureExt, StreamExt};
|
||||||
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
||||||
use slog::{crit, debug, info, trace};
|
use slog::{crit, debug, info, trace};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{delay_until, interval_at, Duration, Instant};
|
||||||
use tokio::timer::{Delay, Interval};
|
|
||||||
use types::{Attestation, ChainSpec, CommitteeIndex, EthSpec, Slot};
|
use types::{Attestation, ChainSpec, CommitteeIndex, EthSpec, Slot};
|
||||||
|
|
||||||
/// Builds an `AttestationService`.
|
/// Builds an `AttestationService`.
|
||||||
@@ -130,7 +129,8 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
.ok_or_else(|| "Unable to determine duration to next slot".to_string())?;
|
.ok_or_else(|| "Unable to determine duration to next slot".to_string())?;
|
||||||
|
|
||||||
let interval = {
|
let interval = {
|
||||||
Interval::new(
|
// Note: `interval_at` panics if `slot_duration` is 0
|
||||||
|
interval_at(
|
||||||
Instant::now() + duration_to_next_slot + slot_duration / 3,
|
Instant::now() + duration_to_next_slot + slot_duration / 3,
|
||||||
slot_duration,
|
slot_duration,
|
||||||
)
|
)
|
||||||
@@ -140,38 +140,28 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
let service = self.clone();
|
let service = self.clone();
|
||||||
let log_1 = log.clone();
|
let log_1 = log.clone();
|
||||||
let log_2 = log.clone();
|
let log_2 = log.clone();
|
||||||
let log_3 = log.clone();
|
|
||||||
|
|
||||||
context.executor.spawn(
|
let interval_fut = interval.for_each(move |_| {
|
||||||
exit_fut
|
if let Err(e) = service.spawn_attestation_tasks(slot_duration) {
|
||||||
.until(
|
crit!(
|
||||||
interval
|
log_1,
|
||||||
.map_err(move |e| {
|
"Failed to spawn attestation tasks";
|
||||||
crit! {
|
"error" => e
|
||||||
log_1,
|
|
||||||
"Timer thread failed";
|
|
||||||
"error" => format!("{}", e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.for_each(move |_| {
|
|
||||||
if let Err(e) = service.spawn_attestation_tasks(slot_duration) {
|
|
||||||
crit!(
|
|
||||||
log_2,
|
|
||||||
"Failed to spawn attestation tasks";
|
|
||||||
"error" => e
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
trace!(
|
|
||||||
log_2,
|
|
||||||
"Spawned attestation tasks";
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}),
|
|
||||||
)
|
)
|
||||||
.map(move |_| info!(log_3, "Shutdown complete")),
|
} else {
|
||||||
|
trace!(
|
||||||
|
log_1,
|
||||||
|
"Spawned attestation tasks";
|
||||||
|
)
|
||||||
|
}
|
||||||
|
futures::future::ready(())
|
||||||
|
});
|
||||||
|
|
||||||
|
let future = futures::future::select(
|
||||||
|
interval_fut,
|
||||||
|
exit_fut.map(move |_| info!(log_2, "Shutdown complete")),
|
||||||
);
|
);
|
||||||
|
tokio::task::spawn(future);
|
||||||
|
|
||||||
Ok(exit_signal)
|
Ok(exit_signal)
|
||||||
}
|
}
|
||||||
@@ -181,11 +171,11 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
fn spawn_attestation_tasks(&self, slot_duration: Duration) -> Result<(), String> {
|
fn spawn_attestation_tasks(&self, slot_duration: Duration) -> Result<(), String> {
|
||||||
let service = self.clone();
|
let service = self.clone();
|
||||||
|
|
||||||
let slot = service
|
let slot = self
|
||||||
.slot_clock
|
.slot_clock
|
||||||
.now()
|
.now()
|
||||||
.ok_or_else(|| "Failed to read slot clock".to_string())?;
|
.ok_or_else(|| "Failed to read slot clock".to_string())?;
|
||||||
let duration_to_next_slot = service
|
let duration_to_next_slot = self
|
||||||
.slot_clock
|
.slot_clock
|
||||||
.duration_to_next_slot()
|
.duration_to_next_slot()
|
||||||
.ok_or_else(|| "Unable to determine duration to next slot".to_string())?;
|
.ok_or_else(|| "Unable to determine duration to next slot".to_string())?;
|
||||||
@@ -197,7 +187,7 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
.checked_sub(slot_duration / 3)
|
.checked_sub(slot_duration / 3)
|
||||||
.unwrap_or_else(|| Duration::from_secs(0));
|
.unwrap_or_else(|| Duration::from_secs(0));
|
||||||
|
|
||||||
let duties_by_committee_index: HashMap<CommitteeIndex, Vec<DutyAndProof>> = service
|
let duties_by_committee_index: HashMap<CommitteeIndex, Vec<DutyAndProof>> = self
|
||||||
.duties_service
|
.duties_service
|
||||||
.attesters(slot)
|
.attesters(slot)
|
||||||
.into_iter()
|
.into_iter()
|
||||||
@@ -219,15 +209,12 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
.into_iter()
|
.into_iter()
|
||||||
.for_each(|(committee_index, validator_duties)| {
|
.for_each(|(committee_index, validator_duties)| {
|
||||||
// Spawn a separate task for each attestation.
|
// Spawn a separate task for each attestation.
|
||||||
service
|
tokio::task::spawn(service.clone().publish_attestations_and_aggregates(
|
||||||
.context
|
slot,
|
||||||
.executor
|
committee_index,
|
||||||
.spawn(self.clone().publish_attestations_and_aggregates(
|
validator_duties,
|
||||||
slot,
|
aggregate_production_instant,
|
||||||
committee_index,
|
));
|
||||||
validator_duties,
|
|
||||||
aggregate_production_instant,
|
|
||||||
));
|
|
||||||
});
|
});
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@@ -242,75 +229,68 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
///
|
///
|
||||||
/// The given `validator_duties` should already be filtered to only contain those that match
|
/// The given `validator_duties` should already be filtered to only contain those that match
|
||||||
/// `slot` and `committee_index`. Critical errors will be logged if this is not the case.
|
/// `slot` and `committee_index`. Critical errors will be logged if this is not the case.
|
||||||
fn publish_attestations_and_aggregates(
|
async fn publish_attestations_and_aggregates(
|
||||||
&self,
|
self,
|
||||||
slot: Slot,
|
slot: Slot,
|
||||||
committee_index: CommitteeIndex,
|
committee_index: CommitteeIndex,
|
||||||
validator_duties: Vec<DutyAndProof>,
|
validator_duties: Vec<DutyAndProof>,
|
||||||
aggregate_production_instant: Instant,
|
aggregate_production_instant: Instant,
|
||||||
) -> Box<dyn Future<Item = (), Error = ()> + Send> {
|
) -> Result<(), ()> {
|
||||||
// There's not need to produce `Attestation` or `SignedAggregateAndProof` if we do not have
|
// There's not need to produce `Attestation` or `SignedAggregateAndProof` if we do not have
|
||||||
// any validators for the given `slot` and `committee_index`.
|
// any validators for the given `slot` and `committee_index`.
|
||||||
if validator_duties.is_empty() {
|
if validator_duties.is_empty() {
|
||||||
return Box::new(future::ok(()));
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
let service_1 = self.clone();
|
|
||||||
let log_1 = self.context.log.clone();
|
let log_1 = self.context.log.clone();
|
||||||
|
let log_2 = self.context.log.clone();
|
||||||
let validator_duties_1 = Arc::new(validator_duties);
|
let validator_duties_1 = Arc::new(validator_duties);
|
||||||
let validator_duties_2 = validator_duties_1.clone();
|
let validator_duties_2 = validator_duties_1.clone();
|
||||||
|
|
||||||
Box::new(
|
// Step 1.
|
||||||
// Step 1.
|
//
|
||||||
//
|
// Download, sign and publish an `Attestation` for each validator.
|
||||||
// Download, sign and publish an `Attestation` for each validator.
|
let attestation_opt = self
|
||||||
self.produce_and_publish_attestations(slot, committee_index, validator_duties_1)
|
.produce_and_publish_attestations(slot, committee_index, validator_duties_1)
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(
|
.await
|
||||||
move |attestation_opt| {
|
.map_err(move |e| {
|
||||||
if let Some(attestation) = attestation_opt {
|
crit!(
|
||||||
Box::new(
|
log_1,
|
||||||
// Step 2. (Only if step 1 produced an attestation)
|
"Error during attestation routine";
|
||||||
//
|
"error" => format!("{:?}", e),
|
||||||
// First, wait until the `aggregation_production_instant` (2/3rds
|
"committee_index" => committee_index,
|
||||||
// of the way though the slot). As verified in the
|
"slot" => slot.as_u64(),
|
||||||
// `delay_triggers_when_in_the_past` test, this code will still run
|
|
||||||
// even if the instant has already elapsed.
|
|
||||||
//
|
|
||||||
// Then download, sign and publish a `SignedAggregateAndProof` for each
|
|
||||||
// validator that is elected to aggregate for this `slot` and
|
|
||||||
// `committee_index`.
|
|
||||||
Delay::new(aggregate_production_instant)
|
|
||||||
.map_err(|e| {
|
|
||||||
format!(
|
|
||||||
"Unable to create aggregate production delay: {:?}",
|
|
||||||
e
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.and_then(move |()| {
|
|
||||||
service_1.produce_and_publish_aggregates(
|
|
||||||
attestation,
|
|
||||||
validator_duties_2,
|
|
||||||
)
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
// If `produce_and_publish_attestations` did not download any
|
|
||||||
// attestations then there is no need to produce any
|
|
||||||
// `SignedAggregateAndProof`.
|
|
||||||
Box::new(future::ok(()))
|
|
||||||
}
|
|
||||||
},
|
|
||||||
)
|
)
|
||||||
.map_err(move |e| {
|
})?;
|
||||||
crit!(
|
if let Some(attestation) = attestation_opt {
|
||||||
log_1,
|
// Step 2. (Only if step 1 produced an attestation)
|
||||||
"Error during attestation routine";
|
//
|
||||||
"error" => format!("{:?}", e),
|
// First, wait until the `aggregation_production_instant` (2/3rds
|
||||||
"committee_index" => committee_index,
|
// of the way though the slot). As verified in the
|
||||||
"slot" => slot.as_u64(),
|
// `delay_triggers_when_in_the_past` test, this code will still run
|
||||||
)
|
// even if the instant has already elapsed.
|
||||||
}),
|
//
|
||||||
)
|
// Then download, sign and publish a `SignedAggregateAndProof` for each
|
||||||
|
// validator that is elected to aggregate for this `slot` and
|
||||||
|
// `committee_index`.
|
||||||
|
delay_until(aggregate_production_instant).await;
|
||||||
|
self.produce_and_publish_aggregates(attestation, validator_duties_2)
|
||||||
|
.await
|
||||||
|
} else {
|
||||||
|
// If `produce_and_publish_attestations` did not download any
|
||||||
|
// attestations then there is no need to produce any
|
||||||
|
// `SignedAggregateAndProof`.
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
.map_err(move |e| {
|
||||||
|
crit!(
|
||||||
|
log_2,
|
||||||
|
"Error during attestation routine";
|
||||||
|
"error" => format!("{:?}", e),
|
||||||
|
"committee_index" => committee_index,
|
||||||
|
"slot" => slot.as_u64(),
|
||||||
|
)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Performs the first step of the attesting process: downloading `Attestation` objects,
|
/// Performs the first step of the attesting process: downloading `Attestation` objects,
|
||||||
@@ -325,139 +305,132 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
///
|
///
|
||||||
/// Only one `Attestation` is downloaded from the BN. It is then cloned and signed by each
|
/// Only one `Attestation` is downloaded from the BN. It is then cloned and signed by each
|
||||||
/// validator and the list of individually-signed `Attestation` objects is returned to the BN.
|
/// validator and the list of individually-signed `Attestation` objects is returned to the BN.
|
||||||
fn produce_and_publish_attestations(
|
async fn produce_and_publish_attestations(
|
||||||
&self,
|
&self,
|
||||||
slot: Slot,
|
slot: Slot,
|
||||||
committee_index: CommitteeIndex,
|
committee_index: CommitteeIndex,
|
||||||
validator_duties: Arc<Vec<DutyAndProof>>,
|
validator_duties: Arc<Vec<DutyAndProof>>,
|
||||||
) -> Box<dyn Future<Item = Option<Attestation<E>>, Error = String> + Send> {
|
) -> Result<Option<Attestation<E>>, String> {
|
||||||
if validator_duties.is_empty() {
|
if validator_duties.is_empty() {
|
||||||
return Box::new(future::ok(None));
|
return Ok(None);
|
||||||
}
|
}
|
||||||
|
|
||||||
let service = self.clone();
|
let attestation = self
|
||||||
|
.beacon_node
|
||||||
|
.http
|
||||||
|
.validator()
|
||||||
|
.produce_attestation(slot, committee_index)
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("Failed to produce attestation: {:?}", e))?;
|
||||||
|
|
||||||
|
let log = self.context.log.clone();
|
||||||
|
|
||||||
|
// For each validator in `validator_duties`, clone the `attestation` and add
|
||||||
|
// their signature.
|
||||||
|
//
|
||||||
|
// If any validator is unable to sign, they are simply skipped.
|
||||||
|
let signed_attestations = validator_duties
|
||||||
|
.iter()
|
||||||
|
.filter_map(|duty| {
|
||||||
|
let log = self.context.log.clone();
|
||||||
|
// Ensure that all required fields are present in the validator duty.
|
||||||
|
let (duty_slot, duty_committee_index, validator_committee_position, _) =
|
||||||
|
if let Some(tuple) = duty.attestation_duties() {
|
||||||
|
tuple
|
||||||
|
} else {
|
||||||
|
crit!(
|
||||||
|
log,
|
||||||
|
"Missing validator duties when signing";
|
||||||
|
"duties" => format!("{:?}", duty)
|
||||||
|
);
|
||||||
|
return None;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Ensure that the attestation matches the duties.
|
||||||
|
if duty_slot != attestation.data.slot
|
||||||
|
|| duty_committee_index != attestation.data.index
|
||||||
|
{
|
||||||
|
crit!(
|
||||||
|
log,
|
||||||
|
"Inconsistent validator duties during signing";
|
||||||
|
"validator" => format!("{:?}", duty.validator_pubkey()),
|
||||||
|
"duty_slot" => duty_slot,
|
||||||
|
"attestation_slot" => attestation.data.slot,
|
||||||
|
"duty_index" => duty_committee_index,
|
||||||
|
"attestation_index" => attestation.data.index,
|
||||||
|
);
|
||||||
|
return None;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut attestation = attestation.clone();
|
||||||
|
|
||||||
|
if self
|
||||||
|
.validator_store
|
||||||
|
.sign_attestation(
|
||||||
|
duty.validator_pubkey(),
|
||||||
|
validator_committee_position,
|
||||||
|
&mut attestation,
|
||||||
|
)
|
||||||
|
.is_none()
|
||||||
|
{
|
||||||
|
crit!(
|
||||||
|
log,
|
||||||
|
"Attestation signing refused";
|
||||||
|
"validator" => format!("{:?}", duty.validator_pubkey()),
|
||||||
|
"slot" => attestation.data.slot,
|
||||||
|
"index" => attestation.data.index,
|
||||||
|
);
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(attestation)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
// If there are any signed attestations, publish them to the BN. Otherwise,
|
||||||
|
// just return early.
|
||||||
|
if let Some(attestation) = signed_attestations.first().cloned() {
|
||||||
|
let num_attestations = signed_attestations.len();
|
||||||
|
let beacon_block_root = attestation.data.beacon_block_root;
|
||||||
|
|
||||||
Box::new(
|
|
||||||
self.beacon_node
|
self.beacon_node
|
||||||
.http
|
.http
|
||||||
.validator()
|
.validator()
|
||||||
.produce_attestation(slot, committee_index)
|
.publish_attestations(signed_attestations)
|
||||||
.map_err(|e| format!("Failed to produce attestation: {:?}", e))
|
.await
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(move |attestation| {
|
.map_err(|e| format!("Failed to publish attestation: {:?}", e))
|
||||||
let log = service.context.log.clone();
|
.map(move |publish_status| match publish_status {
|
||||||
|
PublishStatus::Valid => info!(
|
||||||
// For each validator in `validator_duties`, clone the `attestation` and add
|
log,
|
||||||
// their signature.
|
"Successfully published attestations";
|
||||||
//
|
"count" => num_attestations,
|
||||||
// If any validator is unable to sign, they are simply skipped.
|
"head_block" => format!("{:?}", beacon_block_root),
|
||||||
let signed_attestations = validator_duties
|
"committee_index" => committee_index,
|
||||||
.iter()
|
"slot" => slot.as_u64(),
|
||||||
.filter_map(|duty| {
|
"type" => "unaggregated",
|
||||||
let log = service.context.log.clone();
|
),
|
||||||
|
PublishStatus::Invalid(msg) => crit!(
|
||||||
// Ensure that all required fields are present in the validator duty.
|
log,
|
||||||
let (duty_slot, duty_committee_index, validator_committee_position, _) =
|
"Published attestation was invalid";
|
||||||
if let Some(tuple) = duty.attestation_duties() {
|
"message" => msg,
|
||||||
tuple
|
"committee_index" => committee_index,
|
||||||
} else {
|
"slot" => slot.as_u64(),
|
||||||
crit!(
|
"type" => "unaggregated",
|
||||||
log,
|
),
|
||||||
"Missing validator duties when signing";
|
PublishStatus::Unknown => {
|
||||||
"duties" => format!("{:?}", duty)
|
crit!(log, "Unknown condition when publishing unagg. attestation")
|
||||||
);
|
|
||||||
return None;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Ensure that the attestation matches the duties.
|
|
||||||
if duty_slot != attestation.data.slot
|
|
||||||
|| duty_committee_index != attestation.data.index
|
|
||||||
{
|
|
||||||
crit!(
|
|
||||||
log,
|
|
||||||
"Inconsistent validator duties during signing";
|
|
||||||
"validator" => format!("{:?}", duty.validator_pubkey()),
|
|
||||||
"duty_slot" => duty_slot,
|
|
||||||
"attestation_slot" => attestation.data.slot,
|
|
||||||
"duty_index" => duty_committee_index,
|
|
||||||
"attestation_index" => attestation.data.index,
|
|
||||||
);
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
|
|
||||||
let mut attestation = attestation.clone();
|
|
||||||
|
|
||||||
if service
|
|
||||||
.validator_store
|
|
||||||
.sign_attestation(
|
|
||||||
duty.validator_pubkey(),
|
|
||||||
validator_committee_position,
|
|
||||||
&mut attestation,
|
|
||||||
)
|
|
||||||
.is_none()
|
|
||||||
{
|
|
||||||
crit!(
|
|
||||||
log,
|
|
||||||
"Attestation signing refused";
|
|
||||||
"validator" => format!("{:?}", duty.validator_pubkey()),
|
|
||||||
"slot" => attestation.data.slot,
|
|
||||||
"index" => attestation.data.index,
|
|
||||||
);
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
Some(attestation)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
// If there are any signed attestations, publish them to the BN. Otherwise,
|
|
||||||
// just return early.
|
|
||||||
if let Some(attestation) = signed_attestations.first().cloned() {
|
|
||||||
let num_attestations = signed_attestations.len();
|
|
||||||
let beacon_block_root = attestation.data.beacon_block_root;
|
|
||||||
|
|
||||||
Box::new(
|
|
||||||
service
|
|
||||||
.beacon_node
|
|
||||||
.http
|
|
||||||
.validator()
|
|
||||||
.publish_attestations(signed_attestations)
|
|
||||||
.map_err(|e| format!("Failed to publish attestation: {:?}", e))
|
|
||||||
.map(move |publish_status| match publish_status {
|
|
||||||
PublishStatus::Valid => info!(
|
|
||||||
log,
|
|
||||||
"Successfully published attestations";
|
|
||||||
"count" => num_attestations,
|
|
||||||
"head_block" => format!("{:?}", beacon_block_root),
|
|
||||||
"committee_index" => committee_index,
|
|
||||||
"slot" => slot.as_u64(),
|
|
||||||
"type" => "unaggregated",
|
|
||||||
),
|
|
||||||
PublishStatus::Invalid(msg) => crit!(
|
|
||||||
log,
|
|
||||||
"Published attestation was invalid";
|
|
||||||
"message" => msg,
|
|
||||||
"committee_index" => committee_index,
|
|
||||||
"slot" => slot.as_u64(),
|
|
||||||
"type" => "unaggregated",
|
|
||||||
),
|
|
||||||
PublishStatus::Unknown => crit!(
|
|
||||||
log,
|
|
||||||
"Unknown condition when publishing unagg. attestation"
|
|
||||||
),
|
|
||||||
})
|
|
||||||
.map(|()| Some(attestation)),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
debug!(
|
|
||||||
log,
|
|
||||||
"No attestations to publish";
|
|
||||||
"committee_index" => committee_index,
|
|
||||||
"slot" => slot.as_u64(),
|
|
||||||
);
|
|
||||||
Box::new(future::ok(None))
|
|
||||||
}
|
}
|
||||||
}),
|
})
|
||||||
)
|
.map(|()| Some(attestation))
|
||||||
|
} else {
|
||||||
|
debug!(
|
||||||
|
log,
|
||||||
|
"No attestations to publish";
|
||||||
|
"committee_index" => committee_index,
|
||||||
|
"slot" => slot.as_u64(),
|
||||||
|
);
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Performs the second step of the attesting process: downloading an aggregated `Attestation`,
|
/// Performs the second step of the attesting process: downloading an aggregated `Attestation`,
|
||||||
@@ -473,107 +446,105 @@ impl<T: SlotClock + 'static, E: EthSpec> AttestationService<T, E> {
|
|||||||
/// Only one aggregated `Attestation` is downloaded from the BN. It is then cloned and signed
|
/// Only one aggregated `Attestation` is downloaded from the BN. It is then cloned and signed
|
||||||
/// by each validator and the list of individually-signed `SignedAggregateAndProof` objects is
|
/// by each validator and the list of individually-signed `SignedAggregateAndProof` objects is
|
||||||
/// returned to the BN.
|
/// returned to the BN.
|
||||||
fn produce_and_publish_aggregates(
|
async fn produce_and_publish_aggregates(
|
||||||
&self,
|
&self,
|
||||||
attestation: Attestation<E>,
|
attestation: Attestation<E>,
|
||||||
validator_duties: Arc<Vec<DutyAndProof>>,
|
validator_duties: Arc<Vec<DutyAndProof>>,
|
||||||
) -> impl Future<Item = (), Error = String> {
|
) -> Result<(), String> {
|
||||||
let service_1 = self.clone();
|
|
||||||
let log_1 = self.context.log.clone();
|
let log_1 = self.context.log.clone();
|
||||||
|
|
||||||
self.beacon_node
|
let aggregated_attestation = self
|
||||||
|
.beacon_node
|
||||||
.http
|
.http
|
||||||
.validator()
|
.validator()
|
||||||
.produce_aggregate_attestation(&attestation.data)
|
.produce_aggregate_attestation(&attestation.data)
|
||||||
.map_err(|e| format!("Failed to produce an aggregate attestation: {:?}", e))
|
.await
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(
|
.map_err(|e| format!("Failed to produce an aggregate attestation: {:?}", e))?;
|
||||||
move |aggregated_attestation| {
|
|
||||||
// For each validator, clone the `aggregated_attestation` and convert it into
|
|
||||||
// a `SignedAggregateAndProof`
|
|
||||||
let signed_aggregate_and_proofs = validator_duties
|
|
||||||
.iter()
|
|
||||||
.filter_map(|duty_and_proof| {
|
|
||||||
// Do not produce a signed aggregator for validators that are not
|
|
||||||
// subscribed aggregators.
|
|
||||||
let selection_proof = duty_and_proof.selection_proof.as_ref()?.clone();
|
|
||||||
|
|
||||||
let (duty_slot, duty_committee_index, _, validator_index) =
|
// For each validator, clone the `aggregated_attestation` and convert it into
|
||||||
duty_and_proof.attestation_duties().or_else(|| {
|
// a `SignedAggregateAndProof`
|
||||||
crit!(log_1, "Missing duties when signing aggregate");
|
let signed_aggregate_and_proofs = validator_duties
|
||||||
None
|
.iter()
|
||||||
})?;
|
.filter_map(|duty_and_proof| {
|
||||||
|
// Do not produce a signed aggregator for validators that are not
|
||||||
|
// subscribed aggregators.
|
||||||
|
let selection_proof = duty_and_proof.selection_proof.as_ref()?.clone();
|
||||||
|
|
||||||
let pubkey = &duty_and_proof.duty.validator_pubkey;
|
let (duty_slot, duty_committee_index, _, validator_index) =
|
||||||
let slot = attestation.data.slot;
|
duty_and_proof.attestation_duties().or_else(|| {
|
||||||
let committee_index = attestation.data.index;
|
crit!(log_1, "Missing duties when signing aggregate");
|
||||||
|
None
|
||||||
|
})?;
|
||||||
|
|
||||||
if duty_slot != slot || duty_committee_index != committee_index {
|
let pubkey = &duty_and_proof.duty.validator_pubkey;
|
||||||
crit!(log_1, "Inconsistent validator duties during signing");
|
let slot = attestation.data.slot;
|
||||||
return None;
|
let committee_index = attestation.data.index;
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(signed_aggregate_and_proof) = service_1
|
if duty_slot != slot || duty_committee_index != committee_index {
|
||||||
.validator_store
|
crit!(log_1, "Inconsistent validator duties during signing");
|
||||||
.produce_signed_aggregate_and_proof(
|
return None;
|
||||||
pubkey,
|
}
|
||||||
validator_index,
|
|
||||||
aggregated_attestation.clone(),
|
|
||||||
selection_proof,
|
|
||||||
)
|
|
||||||
{
|
|
||||||
Some(signed_aggregate_and_proof)
|
|
||||||
} else {
|
|
||||||
crit!(log_1, "Failed to sign attestation");
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
|
|
||||||
// If there any signed aggregates and proofs were produced, publish them to the
|
if let Some(signed_aggregate_and_proof) =
|
||||||
// BN.
|
self.validator_store.produce_signed_aggregate_and_proof(
|
||||||
if let Some(first) = signed_aggregate_and_proofs.first().cloned() {
|
pubkey,
|
||||||
let attestation = first.message.aggregate;
|
validator_index,
|
||||||
|
aggregated_attestation.clone(),
|
||||||
|
selection_proof,
|
||||||
|
)
|
||||||
|
{
|
||||||
|
Some(signed_aggregate_and_proof)
|
||||||
|
} else {
|
||||||
|
crit!(log_1, "Failed to sign attestation");
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
Box::new(service_1
|
// If there any signed aggregates and proofs were produced, publish them to the
|
||||||
.beacon_node
|
// BN.
|
||||||
.http
|
if let Some(first) = signed_aggregate_and_proofs.first().cloned() {
|
||||||
.validator()
|
let attestation = first.message.aggregate;
|
||||||
.publish_aggregate_and_proof(signed_aggregate_and_proofs)
|
|
||||||
.map(|publish_status| (attestation, publish_status))
|
let publish_status = self
|
||||||
.map_err(|e| format!("Failed to publish aggregate and proofs: {:?}", e))
|
.beacon_node
|
||||||
.map(move |(attestation, publish_status)| match publish_status {
|
.http
|
||||||
PublishStatus::Valid => info!(
|
.validator()
|
||||||
log_1,
|
.publish_aggregate_and_proof(signed_aggregate_and_proofs)
|
||||||
"Successfully published attestations";
|
.await
|
||||||
"signatures" => attestation.aggregation_bits.num_set_bits(),
|
.map_err(|e| format!("Failed to publish aggregate and proofs: {:?}", e))?;
|
||||||
"head_block" => format!("{:?}", attestation.data.beacon_block_root),
|
match publish_status {
|
||||||
"committee_index" => attestation.data.index,
|
PublishStatus::Valid => info!(
|
||||||
"slot" => attestation.data.slot.as_u64(),
|
log_1,
|
||||||
"type" => "aggregated",
|
"Successfully published attestations";
|
||||||
),
|
"signatures" => attestation.aggregation_bits.num_set_bits(),
|
||||||
PublishStatus::Invalid(msg) => crit!(
|
"head_block" => format!("{:?}", attestation.data.beacon_block_root),
|
||||||
log_1,
|
"committee_index" => attestation.data.index,
|
||||||
"Published attestation was invalid";
|
"slot" => attestation.data.slot.as_u64(),
|
||||||
"message" => msg,
|
"type" => "aggregated",
|
||||||
"committee_index" => attestation.data.index,
|
),
|
||||||
"slot" => attestation.data.slot.as_u64(),
|
PublishStatus::Invalid(msg) => crit!(
|
||||||
"type" => "aggregated",
|
log_1,
|
||||||
),
|
"Published attestation was invalid";
|
||||||
PublishStatus::Unknown => {
|
"message" => msg,
|
||||||
crit!(log_1, "Unknown condition when publishing agg. attestation")
|
"committee_index" => attestation.data.index,
|
||||||
}
|
"slot" => attestation.data.slot.as_u64(),
|
||||||
}))
|
"type" => "aggregated",
|
||||||
} else {
|
),
|
||||||
debug!(
|
PublishStatus::Unknown => {
|
||||||
log_1,
|
crit!(log_1, "Unknown condition when publishing agg. attestation")
|
||||||
"No signed aggregates to publish";
|
}
|
||||||
"committee_index" => attestation.data.index,
|
};
|
||||||
"slot" => attestation.data.slot.as_u64(),
|
Ok(())
|
||||||
);
|
} else {
|
||||||
Box::new(future::ok(()))
|
debug!(
|
||||||
}
|
log_1,
|
||||||
},
|
"No signed aggregates to publish";
|
||||||
)
|
"committee_index" => attestation.data.index,
|
||||||
|
"slot" => attestation.data.slot.as_u64(),
|
||||||
|
);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -591,16 +562,14 @@ mod tests {
|
|||||||
let state_1 = Arc::new(RwLock::new(in_the_past));
|
let state_1 = Arc::new(RwLock::new(in_the_past));
|
||||||
let state_2 = state_1.clone();
|
let state_2 = state_1.clone();
|
||||||
|
|
||||||
let future = Delay::new(in_the_past)
|
let future = delay_until(in_the_past).map(move |()| *state_1.write() = Instant::now());
|
||||||
.map_err(|_| panic!("Failed to create duration"))
|
|
||||||
.map(move |()| *state_1.write() = Instant::now());
|
|
||||||
|
|
||||||
let mut runtime = RuntimeBuilder::new()
|
let mut runtime = RuntimeBuilder::new()
|
||||||
.core_threads(1)
|
.core_threads(1)
|
||||||
.build()
|
.build()
|
||||||
.expect("failed to start runtime");
|
.expect("failed to start runtime");
|
||||||
|
|
||||||
runtime.block_on(future).expect("failed to complete future");
|
runtime.block_on(future);
|
||||||
|
|
||||||
assert!(
|
assert!(
|
||||||
*state_2.read() > in_the_past,
|
*state_2.read() > in_the_past,
|
||||||
|
|||||||
@@ -1,15 +1,14 @@
|
|||||||
use crate::{duties_service::DutiesService, validator_store::ValidatorStore};
|
use crate::{duties_service::DutiesService, validator_store::ValidatorStore};
|
||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use futures::{stream, Future, IntoFuture, Stream};
|
use futures::{FutureExt, StreamExt};
|
||||||
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
||||||
use slog::{crit, error, info, trace};
|
use slog::{crit, error, info, trace};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{interval_at, Duration, Instant};
|
||||||
use tokio::timer::Interval;
|
use types::{ChainSpec, EthSpec, PublicKey, Slot};
|
||||||
use types::{ChainSpec, EthSpec};
|
|
||||||
|
|
||||||
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
||||||
const TIME_DELAY_FROM_SLOT: Duration = Duration::from_millis(100);
|
const TIME_DELAY_FROM_SLOT: Duration = Duration::from_millis(100);
|
||||||
@@ -124,7 +123,8 @@ impl<T: SlotClock + 'static, E: EthSpec> BlockService<T, E> {
|
|||||||
|
|
||||||
let interval = {
|
let interval = {
|
||||||
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
||||||
Interval::new(
|
// Note: interval_at panics if slot_duration = 0
|
||||||
|
interval_at(
|
||||||
Instant::now() + duration_to_next_slot + TIME_DELAY_FROM_SLOT,
|
Instant::now() + duration_to_next_slot + TIME_DELAY_FROM_SLOT,
|
||||||
slot_duration,
|
slot_duration,
|
||||||
)
|
)
|
||||||
@@ -132,135 +132,102 @@ impl<T: SlotClock + 'static, E: EthSpec> BlockService<T, E> {
|
|||||||
|
|
||||||
let (exit_signal, exit_fut) = exit_future::signal();
|
let (exit_signal, exit_fut) = exit_future::signal();
|
||||||
let service = self.clone();
|
let service = self.clone();
|
||||||
let log_1 = log.clone();
|
let interval_fut = interval.for_each(move |_| {
|
||||||
let log_2 = log.clone();
|
let _ = service.clone().do_update();
|
||||||
|
futures::future::ready(())
|
||||||
|
});
|
||||||
|
|
||||||
self.context.executor.spawn(
|
let future = futures::future::select(
|
||||||
exit_fut
|
interval_fut,
|
||||||
.until(
|
exit_fut.map(move |_| info!(log, "Shutdown complete")),
|
||||||
interval
|
|
||||||
.map_err(move |e| {
|
|
||||||
crit! {
|
|
||||||
log_1,
|
|
||||||
"Timer thread failed";
|
|
||||||
"error" => format!("{}", e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.for_each(move |_| service.clone().do_update().then(|_| Ok(()))),
|
|
||||||
)
|
|
||||||
.map(move |_| info!(log_2, "Shutdown complete")),
|
|
||||||
);
|
);
|
||||||
|
tokio::task::spawn(future);
|
||||||
|
|
||||||
Ok(exit_signal)
|
Ok(exit_signal)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Attempt to produce a block for any block producers in the `ValidatorStore`.
|
/// Attempt to produce a block for any block producers in the `ValidatorStore`.
|
||||||
fn do_update(self) -> impl Future<Item = (), Error = ()> {
|
fn do_update(&self) -> Result<(), ()> {
|
||||||
let service = self.clone();
|
|
||||||
let log_1 = self.context.log.clone();
|
let log_1 = self.context.log.clone();
|
||||||
let log_2 = self.context.log.clone();
|
let log_2 = self.context.log.clone();
|
||||||
|
|
||||||
self.slot_clock
|
let slot = self.slot_clock.now().ok_or_else(move || {
|
||||||
.now()
|
crit!(log_1, "Duties manager failed to read slot clock");
|
||||||
.ok_or_else(move || {
|
})?;
|
||||||
crit!(log_1, "Duties manager failed to read slot clock");
|
let iter = self.duties_service.block_producers(slot).into_iter();
|
||||||
})
|
|
||||||
.into_future()
|
|
||||||
.and_then(move |slot| {
|
|
||||||
let iter = service.duties_service.block_producers(slot).into_iter();
|
|
||||||
|
|
||||||
if iter.len() == 0 {
|
if iter.len() == 0 {
|
||||||
trace!(
|
trace!(
|
||||||
log_2,
|
log_2,
|
||||||
"No local block proposers for this slot";
|
"No local block proposers for this slot";
|
||||||
"slot" => slot.as_u64()
|
"slot" => slot.as_u64()
|
||||||
)
|
)
|
||||||
} else if iter.len() > 1 {
|
} else if iter.len() > 1 {
|
||||||
error!(
|
error!(
|
||||||
log_2,
|
log_2,
|
||||||
"Multiple block proposers for this slot";
|
"Multiple block proposers for this slot";
|
||||||
"action" => "producing blocks for all proposers",
|
"action" => "producing blocks for all proposers",
|
||||||
"num_proposers" => iter.len(),
|
"num_proposers" => iter.len(),
|
||||||
"slot" => slot.as_u64(),
|
"slot" => slot.as_u64(),
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
stream::unfold(iter, move |mut block_producers| {
|
// TODO: check if the logic is same as stream::unfold version
|
||||||
let log_1 = service.context.log.clone();
|
let _ = futures::stream::iter(iter).for_each(|validator_pubkey| async {
|
||||||
let log_2 = service.context.log.clone();
|
match self.publish_block(slot, validator_pubkey).await {
|
||||||
let service_1 = service.clone();
|
Ok(()) => (),
|
||||||
let service_2 = service.clone();
|
Err(e) => crit!(
|
||||||
let service_3 = service.clone();
|
log_2,
|
||||||
|
"Error whilst producing block";
|
||||||
|
"message" => e
|
||||||
|
),
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
block_producers.next().map(move |validator_pubkey| {
|
Ok(())
|
||||||
service_1
|
}
|
||||||
.validator_store
|
|
||||||
.randao_reveal(&validator_pubkey, slot.epoch(E::slots_per_epoch()))
|
/// Produce a block at the given slot for validator_pubkey
|
||||||
.ok_or_else(|| "Unable to produce randao reveal".to_string())
|
async fn publish_block(&self, slot: Slot, validator_pubkey: PublicKey) -> Result<(), String> {
|
||||||
.into_future()
|
let log_1 = self.context.log.clone();
|
||||||
.and_then(move |randao_reveal| {
|
let randao_reveal = self
|
||||||
service_1
|
.validator_store
|
||||||
.beacon_node
|
.randao_reveal(&validator_pubkey, slot.epoch(E::slots_per_epoch()))
|
||||||
.http
|
.ok_or_else(|| "Unable to produce randao reveal".to_string())?;
|
||||||
.validator()
|
let block = self
|
||||||
.produce_block(slot, randao_reveal)
|
.beacon_node
|
||||||
.map_err(|e| {
|
.http
|
||||||
format!(
|
.validator()
|
||||||
"Error from beacon node when producing block: {:?}",
|
.produce_block(slot, randao_reveal)
|
||||||
e
|
.await
|
||||||
)
|
.map_err(|e| format!("Error from beacon node when producing block: {:?}", e))?;
|
||||||
})
|
let signed_block = self
|
||||||
})
|
.validator_store
|
||||||
.and_then(move |block| {
|
.sign_block(&validator_pubkey, block)
|
||||||
service_2
|
.ok_or_else(|| "Unable to sign block".to_string())?;
|
||||||
.validator_store
|
let publish_status = self
|
||||||
.sign_block(&validator_pubkey, block)
|
.beacon_node
|
||||||
.ok_or_else(|| "Unable to sign block".to_string())
|
.http
|
||||||
})
|
.validator()
|
||||||
.and_then(move |block| {
|
.publish_block(signed_block.clone())
|
||||||
service_3
|
.await
|
||||||
.beacon_node
|
.map_err(|e| format!("Error from beacon node when publishing block: {:?}", e))?;
|
||||||
.http
|
match publish_status {
|
||||||
.validator()
|
PublishStatus::Valid => info!(
|
||||||
.publish_block(block.clone())
|
log_1,
|
||||||
.map(|publish_status| (block, publish_status))
|
"Successfully published block";
|
||||||
.map_err(|e| {
|
"deposits" => signed_block.message.body.deposits.len(),
|
||||||
format!(
|
"attestations" => signed_block.message.body.attestations.len(),
|
||||||
"Error from beacon node when publishing block: {:?}",
|
"slot" => signed_block.slot().as_u64(),
|
||||||
e
|
),
|
||||||
)
|
PublishStatus::Invalid(msg) => crit!(
|
||||||
})
|
log_1,
|
||||||
})
|
"Published block was invalid";
|
||||||
.map(move |(block, publish_status)| match publish_status {
|
"message" => msg,
|
||||||
PublishStatus::Valid => info!(
|
"slot" => signed_block.slot().as_u64(),
|
||||||
log_1,
|
),
|
||||||
"Successfully published block";
|
PublishStatus::Unknown => crit!(log_1, "Unknown condition when publishing block"),
|
||||||
"deposits" => block.message.body.deposits.len(),
|
}
|
||||||
"attestations" => block.message.body.attestations.len(),
|
Ok(())
|
||||||
"slot" => block.slot().as_u64(),
|
|
||||||
),
|
|
||||||
PublishStatus::Invalid(msg) => crit!(
|
|
||||||
log_1,
|
|
||||||
"Published block was invalid";
|
|
||||||
"message" => msg,
|
|
||||||
"slot" => block.slot().as_u64(),
|
|
||||||
),
|
|
||||||
PublishStatus::Unknown => {
|
|
||||||
crit!(log_1, "Unknown condition when publishing block")
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.map_err(move |e| {
|
|
||||||
crit!(
|
|
||||||
log_2,
|
|
||||||
"Error whilst producing block";
|
|
||||||
"message" => e
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.then(|_| Ok(((), block_producers)))
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
.map(|_| ())
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,17 @@
|
|||||||
use crate::validator_store::ValidatorStore;
|
use crate::validator_store::ValidatorStore;
|
||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use futures::{future, Future, IntoFuture, Stream};
|
use futures::{FutureExt, StreamExt};
|
||||||
use parking_lot::RwLock;
|
use parking_lot::RwLock;
|
||||||
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
use remote_beacon_node::{PublishStatus, RemoteBeaconNode};
|
||||||
use rest_types::{ValidatorDuty, ValidatorDutyBytes, ValidatorSubscription};
|
use rest_types::{ValidatorDuty, ValidatorDutyBytes, ValidatorSubscription};
|
||||||
use slog::{crit, debug, error, info, trace, warn};
|
use slog::{debug, error, info, trace, warn};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use std::convert::TryInto;
|
use std::convert::TryInto;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{interval_at, Duration, Instant};
|
||||||
use tokio::timer::Interval;
|
|
||||||
use types::{ChainSpec, CommitteeIndex, Epoch, EthSpec, PublicKey, SelectionProof, Slot};
|
use types::{ChainSpec, CommitteeIndex, Epoch, EthSpec, PublicKey, SelectionProof, Slot};
|
||||||
|
|
||||||
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
||||||
@@ -440,54 +439,53 @@ impl<T: SlotClock + 'static, E: EthSpec> DutiesService<T, E> {
|
|||||||
|
|
||||||
let interval = {
|
let interval = {
|
||||||
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
||||||
Interval::new(
|
// Note: `interval_at` panics if `slot_duration` is 0
|
||||||
|
interval_at(
|
||||||
Instant::now() + duration_to_next_slot + TIME_DELAY_FROM_SLOT,
|
Instant::now() + duration_to_next_slot + TIME_DELAY_FROM_SLOT,
|
||||||
slot_duration,
|
slot_duration,
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
let (exit_signal, exit_fut) = exit_future::signal();
|
let (exit_signal, exit_fut) = exit_future::signal();
|
||||||
let service = self.clone();
|
let service_1 = self.clone();
|
||||||
let log_1 = log.clone();
|
let service_2 = self.clone();
|
||||||
let log_2 = log.clone();
|
|
||||||
|
|
||||||
// Run an immediate update before starting the updater service.
|
// Run an immediate update before starting the updater service.
|
||||||
self.context.executor.spawn(service.clone().do_update());
|
tokio::task::spawn(service_1.do_update());
|
||||||
|
|
||||||
self.context.executor.spawn(
|
let interval_fut = interval.for_each(move |_| {
|
||||||
exit_fut
|
let _ = service_2.clone().do_update();
|
||||||
.until(
|
futures::future::ready(())
|
||||||
interval
|
});
|
||||||
.map_err(move |e| {
|
|
||||||
crit! {
|
let future = futures::future::select(
|
||||||
log_1,
|
interval_fut,
|
||||||
"Timer thread failed";
|
exit_fut.map(move |_| info!(log, "Shutdown complete")),
|
||||||
"error" => format!("{}", e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.for_each(move |_| service.clone().do_update().then(|_| Ok(()))),
|
|
||||||
)
|
|
||||||
.map(move |_| info!(log_2, "Shutdown complete")),
|
|
||||||
);
|
);
|
||||||
|
tokio::task::spawn(future);
|
||||||
|
|
||||||
Ok(exit_signal)
|
Ok(exit_signal)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Attempt to download the duties of all managed validators for this epoch and the next.
|
/// Attempt to download the duties of all managed validators for this epoch and the next.
|
||||||
fn do_update(&self) -> impl Future<Item = (), Error = ()> {
|
async fn do_update(self) -> Result<(), ()> {
|
||||||
|
let log_1 = self.context.log.clone();
|
||||||
|
let log_2 = self.context.log.clone();
|
||||||
|
let log_3 = self.context.log.clone();
|
||||||
|
let log = self.context.log.clone();
|
||||||
|
|
||||||
let service_1 = self.clone();
|
let service_1 = self.clone();
|
||||||
let service_2 = self.clone();
|
let service_2 = self.clone();
|
||||||
let service_3 = self.clone();
|
let service_3 = self.clone();
|
||||||
let service_4 = self.clone();
|
let service_4 = self.clone();
|
||||||
let log_1 = self.context.log.clone();
|
let service_5 = self.clone();
|
||||||
let log_2 = self.context.log.clone();
|
|
||||||
|
|
||||||
self.slot_clock
|
let current_epoch = service_1
|
||||||
|
.slot_clock
|
||||||
.now()
|
.now()
|
||||||
.ok_or_else(move || {
|
.ok_or_else(move || {
|
||||||
error!(log_1, "Duties manager failed to read slot clock");
|
error!(log_1, "Duties manager failed to read slot clock");
|
||||||
})
|
})
|
||||||
.into_future()
|
|
||||||
.map(move |slot| {
|
.map(move |slot| {
|
||||||
let epoch = slot.epoch(E::slots_per_epoch());
|
let epoch = slot.epoch(E::slots_per_epoch());
|
||||||
|
|
||||||
@@ -501,218 +499,203 @@ impl<T: SlotClock + 'static, E: EthSpec> DutiesService<T, E> {
|
|||||||
"current_epoch" => epoch.as_u64(),
|
"current_epoch" => epoch.as_u64(),
|
||||||
);
|
);
|
||||||
|
|
||||||
service_1.store.prune(prune_below);
|
self.store.prune(prune_below);
|
||||||
}
|
}
|
||||||
|
|
||||||
epoch
|
epoch
|
||||||
})
|
})?;
|
||||||
.and_then(move |epoch| {
|
|
||||||
let log = service_2.context.log.clone();
|
|
||||||
|
|
||||||
service_2
|
let beacon_head_epoch = service_2
|
||||||
.beacon_node
|
.beacon_node
|
||||||
.http
|
.http
|
||||||
.beacon()
|
.beacon()
|
||||||
.get_head()
|
.get_head()
|
||||||
.map(move |head| (epoch, head.slot.epoch(E::slots_per_epoch())))
|
.await
|
||||||
.map_err(move |e| {
|
.map(|head| head.slot.epoch(E::slots_per_epoch()))
|
||||||
error!(
|
.map_err(move |e| {
|
||||||
log,
|
error!(
|
||||||
"Failed to contact beacon node";
|
log_3,
|
||||||
"error" => format!("{:?}", e)
|
"Failed to contact beacon node";
|
||||||
)
|
"error" => format!("{:?}", e)
|
||||||
})
|
)
|
||||||
})
|
})?;
|
||||||
.and_then(move |(current_epoch, beacon_head_epoch)| {
|
|
||||||
let log = service_3.context.log.clone();
|
|
||||||
|
|
||||||
let future: Box<dyn Future<Item = (), Error = ()> + Send> = if beacon_head_epoch + 1
|
if beacon_head_epoch + 1 < current_epoch && !service_3.allow_unsynced_beacon_node {
|
||||||
< current_epoch
|
error!(
|
||||||
&& !service_3.allow_unsynced_beacon_node
|
log,
|
||||||
{
|
"Beacon node is not synced";
|
||||||
|
"node_head_epoch" => format!("{}", beacon_head_epoch),
|
||||||
|
"current_epoch" => format!("{}", current_epoch),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
let result = service_4.clone().update_epoch(current_epoch).await;
|
||||||
|
if let Err(e) = result {
|
||||||
|
error!(
|
||||||
|
log,
|
||||||
|
"Failed to get current epoch duties";
|
||||||
|
"http_error" => format!("{:?}", e)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
service_5
|
||||||
|
.clone()
|
||||||
|
.update_epoch(current_epoch + 1)
|
||||||
|
.await
|
||||||
|
.map_err(move |e| {
|
||||||
error!(
|
error!(
|
||||||
log,
|
log,
|
||||||
"Beacon node is not synced";
|
"Failed to get next epoch duties";
|
||||||
"node_head_epoch" => format!("{}", beacon_head_epoch),
|
"http_error" => format!("{:?}", e)
|
||||||
"current_epoch" => format!("{}", current_epoch),
|
|
||||||
);
|
);
|
||||||
|
})?;
|
||||||
Box::new(future::ok(()))
|
};
|
||||||
} else {
|
Ok(())
|
||||||
Box::new(service_3.update_epoch(current_epoch).then(move |result| {
|
|
||||||
if let Err(e) = result {
|
|
||||||
error!(
|
|
||||||
log,
|
|
||||||
"Failed to get current epoch duties";
|
|
||||||
"http_error" => format!("{:?}", e)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
let log = service_4.context.log.clone();
|
|
||||||
service_4.update_epoch(current_epoch + 1).map_err(move |e| {
|
|
||||||
error!(
|
|
||||||
log,
|
|
||||||
"Failed to get next epoch duties";
|
|
||||||
"http_error" => format!("{:?}", e)
|
|
||||||
);
|
|
||||||
})
|
|
||||||
}))
|
|
||||||
};
|
|
||||||
|
|
||||||
future
|
|
||||||
})
|
|
||||||
.map(|_| ())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Attempt to download the duties of all managed validators for the given `epoch`.
|
/// Attempt to download the duties of all managed validators for the given `epoch`.
|
||||||
fn update_epoch(self, epoch: Epoch) -> impl Future<Item = (), Error = String> {
|
async fn update_epoch(self, epoch: Epoch) -> Result<(), String> {
|
||||||
let service_1 = self.clone();
|
let pubkeys = self.validator_store.voting_pubkeys();
|
||||||
let service_2 = self.clone();
|
let all_duties = self
|
||||||
let service_3 = self;
|
|
||||||
|
|
||||||
let pubkeys = service_1.validator_store.voting_pubkeys();
|
|
||||||
service_1
|
|
||||||
.beacon_node
|
.beacon_node
|
||||||
.http
|
.http
|
||||||
.validator()
|
.validator()
|
||||||
.get_duties(epoch, pubkeys.as_slice())
|
.get_duties(epoch, pubkeys.as_slice())
|
||||||
.map(move |all_duties| (epoch, all_duties))
|
.await
|
||||||
.map_err(move |e| format!("Failed to get duties for epoch {}: {:?}", epoch, e))
|
.map_err(move |e| format!("Failed to get duties for epoch {}: {:?}", epoch, e))?;
|
||||||
.and_then(move |(epoch, all_duties)| {
|
|
||||||
let log = service_2.context.log.clone();
|
|
||||||
|
|
||||||
let mut new_validator = 0;
|
let log = self.context.log.clone();
|
||||||
let mut new_epoch = 0;
|
|
||||||
let mut identical = 0;
|
|
||||||
let mut replaced = 0;
|
|
||||||
let mut invalid = 0;
|
|
||||||
|
|
||||||
// For each of the duties, attempt to insert them into our local store and build a
|
let mut new_validator = 0;
|
||||||
// list of new or changed selections proofs for any aggregating validators.
|
let mut new_epoch = 0;
|
||||||
let validator_subscriptions = all_duties.into_iter().filter_map(|remote_duties| {
|
let mut identical = 0;
|
||||||
// Convert the remote duties into our local representation.
|
let mut replaced = 0;
|
||||||
let duties: DutyAndProof = remote_duties
|
let mut invalid = 0;
|
||||||
.try_into()
|
|
||||||
.map_err(|e| error!(
|
// For each of the duties, attempt to insert them into our local store and build a
|
||||||
|
// list of new or changed selections proofs for any aggregating validators.
|
||||||
|
let validator_subscriptions = all_duties
|
||||||
|
.into_iter()
|
||||||
|
.filter_map(|remote_duties| {
|
||||||
|
// Convert the remote duties into our local representation.
|
||||||
|
let duties: DutyAndProof = remote_duties
|
||||||
|
.try_into()
|
||||||
|
.map_err(|e| {
|
||||||
|
error!(
|
||||||
log,
|
log,
|
||||||
"Unable to convert remote duties";
|
"Unable to convert remote duties";
|
||||||
"error" => e
|
"error" => e
|
||||||
))
|
)
|
||||||
.ok()?;
|
})
|
||||||
|
.ok()?;
|
||||||
|
|
||||||
// Attempt to update our local store.
|
// Attempt to update our local store.
|
||||||
let outcome = service_2
|
let outcome = self
|
||||||
.store
|
.store
|
||||||
.insert(epoch, duties.clone(), E::slots_per_epoch(), &service_2.validator_store)
|
.insert(
|
||||||
.map_err(|e| error!(
|
epoch,
|
||||||
|
duties.clone(),
|
||||||
|
E::slots_per_epoch(),
|
||||||
|
&self.validator_store,
|
||||||
|
)
|
||||||
|
.map_err(|e| {
|
||||||
|
error!(
|
||||||
log,
|
log,
|
||||||
"Unable to store duties";
|
"Unable to store duties";
|
||||||
"error" => e
|
"error" => e
|
||||||
))
|
)
|
||||||
.ok()?;
|
})
|
||||||
|
.ok()?;
|
||||||
|
|
||||||
match &outcome {
|
match &outcome {
|
||||||
InsertOutcome::NewValidator => {
|
InsertOutcome::NewValidator => {
|
||||||
debug!(
|
debug!(
|
||||||
log,
|
log,
|
||||||
"First duty assignment for validator";
|
"First duty assignment for validator";
|
||||||
"proposal_slots" => format!("{:?}", &duties.duty.block_proposal_slots),
|
"proposal_slots" => format!("{:?}", &duties.duty.block_proposal_slots),
|
||||||
"attestation_slot" => format!("{:?}", &duties.duty.attestation_slot),
|
"attestation_slot" => format!("{:?}", &duties.duty.attestation_slot),
|
||||||
"validator" => format!("{:?}", &duties.duty.validator_pubkey)
|
"validator" => format!("{:?}", &duties.duty.validator_pubkey)
|
||||||
);
|
);
|
||||||
new_validator += 1;
|
new_validator += 1;
|
||||||
}
|
|
||||||
InsertOutcome::NewEpoch => new_epoch += 1,
|
|
||||||
InsertOutcome::Identical => identical += 1,
|
|
||||||
InsertOutcome::Replaced { .. } => replaced += 1,
|
|
||||||
InsertOutcome::Invalid => invalid += 1,
|
|
||||||
};
|
|
||||||
|
|
||||||
if outcome.is_subscription_candidate() {
|
|
||||||
Some(ValidatorSubscription {
|
|
||||||
validator_index: duties.duty.validator_index?,
|
|
||||||
attestation_committee_index: duties.duty.attestation_committee_index?,
|
|
||||||
slot: duties.duty.attestation_slot?,
|
|
||||||
is_aggregator: duties.selection_proof.is_some(),
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
}
|
||||||
}).collect::<Vec<_>>();
|
InsertOutcome::NewEpoch => new_epoch += 1,
|
||||||
|
InsertOutcome::Identical => identical += 1,
|
||||||
|
InsertOutcome::Replaced { .. } => replaced += 1,
|
||||||
|
InsertOutcome::Invalid => invalid += 1,
|
||||||
|
};
|
||||||
|
|
||||||
if invalid > 0 {
|
if outcome.is_subscription_candidate() {
|
||||||
error!(
|
Some(ValidatorSubscription {
|
||||||
log,
|
validator_index: duties.duty.validator_index?,
|
||||||
"Received invalid duties from beacon node";
|
attestation_committee_index: duties.duty.attestation_committee_index?,
|
||||||
"bad_duty_count" => invalid,
|
slot: duties.duty.attestation_slot?,
|
||||||
"info" => "Duties are from wrong epoch."
|
is_aggregator: duties.selection_proof.is_some(),
|
||||||
)
|
})
|
||||||
}
|
|
||||||
|
|
||||||
trace!(
|
|
||||||
log,
|
|
||||||
"Performed duties update";
|
|
||||||
"identical" => identical,
|
|
||||||
"new_epoch" => new_epoch,
|
|
||||||
"new_validator" => new_validator,
|
|
||||||
"replaced" => replaced,
|
|
||||||
"epoch" => format!("{}", epoch)
|
|
||||||
);
|
|
||||||
|
|
||||||
if replaced > 0 {
|
|
||||||
warn!(
|
|
||||||
log,
|
|
||||||
"Duties changed during routine update";
|
|
||||||
"info" => "Chain re-org likely occurred."
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(validator_subscriptions)
|
|
||||||
})
|
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(move |validator_subscriptions| {
|
|
||||||
let log = service_3.context.log.clone();
|
|
||||||
let count = validator_subscriptions.len();
|
|
||||||
|
|
||||||
if count == 0 {
|
|
||||||
debug!(
|
|
||||||
log,
|
|
||||||
"No new subscriptions required"
|
|
||||||
);
|
|
||||||
|
|
||||||
Box::new(future::ok(()))
|
|
||||||
} else {
|
} else {
|
||||||
Box::new(service_3.beacon_node
|
None
|
||||||
.http
|
|
||||||
.validator()
|
|
||||||
.subscribe(validator_subscriptions)
|
|
||||||
.map_err(|e| format!("Failed to subscribe validators: {:?}", e))
|
|
||||||
.map(move |status| {
|
|
||||||
match status {
|
|
||||||
PublishStatus::Valid => {
|
|
||||||
debug!(
|
|
||||||
log,
|
|
||||||
"Successfully subscribed validators";
|
|
||||||
"count" => count
|
|
||||||
)
|
|
||||||
},
|
|
||||||
PublishStatus::Unknown => {
|
|
||||||
error!(
|
|
||||||
log,
|
|
||||||
"Unknown response from subscription";
|
|
||||||
)
|
|
||||||
},
|
|
||||||
PublishStatus::Invalid(e) => {
|
|
||||||
error!(
|
|
||||||
log,
|
|
||||||
"Failed to subscribe validator";
|
|
||||||
"error" => e
|
|
||||||
)
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
})
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
|
if invalid > 0 {
|
||||||
|
error!(
|
||||||
|
log,
|
||||||
|
"Received invalid duties from beacon node";
|
||||||
|
"bad_duty_count" => invalid,
|
||||||
|
"info" => "Duties are from wrong epoch."
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
trace!(
|
||||||
|
log,
|
||||||
|
"Performed duties update";
|
||||||
|
"identical" => identical,
|
||||||
|
"new_epoch" => new_epoch,
|
||||||
|
"new_validator" => new_validator,
|
||||||
|
"replaced" => replaced,
|
||||||
|
"epoch" => format!("{}", epoch)
|
||||||
|
);
|
||||||
|
|
||||||
|
if replaced > 0 {
|
||||||
|
warn!(
|
||||||
|
log,
|
||||||
|
"Duties changed during routine update";
|
||||||
|
"info" => "Chain re-org likely occurred."
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
let log = self.context.log.clone();
|
||||||
|
let count = validator_subscriptions.len();
|
||||||
|
|
||||||
|
if count == 0 {
|
||||||
|
debug!(log, "No new subscriptions required");
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
self.beacon_node
|
||||||
|
.http
|
||||||
|
.validator()
|
||||||
|
.subscribe(validator_subscriptions)
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("Failed to subscribe validators: {:?}", e))
|
||||||
|
.map(move |status| {
|
||||||
|
match status {
|
||||||
|
PublishStatus::Valid => debug!(
|
||||||
|
log,
|
||||||
|
"Successfully subscribed validators";
|
||||||
|
"count" => count
|
||||||
|
),
|
||||||
|
PublishStatus::Unknown => error!(
|
||||||
|
log,
|
||||||
|
"Unknown response from subscription";
|
||||||
|
),
|
||||||
|
PublishStatus::Invalid(e) => error!(
|
||||||
|
log,
|
||||||
|
"Failed to subscribe validator";
|
||||||
|
"error" => e
|
||||||
|
),
|
||||||
|
};
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use futures::{Future, Stream};
|
use futures::{FutureExt, StreamExt};
|
||||||
use parking_lot::RwLock;
|
use parking_lot::RwLock;
|
||||||
use remote_beacon_node::RemoteBeaconNode;
|
use remote_beacon_node::RemoteBeaconNode;
|
||||||
use slog::{crit, info, trace};
|
use slog::{info, trace};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use std::ops::Deref;
|
use std::ops::Deref;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{interval_at, Duration, Instant};
|
||||||
use tokio::timer::Interval;
|
|
||||||
use types::{ChainSpec, EthSpec, Fork};
|
use types::{ChainSpec, EthSpec, Fork};
|
||||||
|
|
||||||
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
/// Delay this period of time after the slot starts. This allows the node to process the new slot.
|
||||||
@@ -111,51 +110,46 @@ impl<T: SlotClock + 'static, E: EthSpec> ForkService<T, E> {
|
|||||||
|
|
||||||
let interval = {
|
let interval = {
|
||||||
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
let slot_duration = Duration::from_millis(spec.milliseconds_per_slot);
|
||||||
Interval::new(
|
// Note: interval_at panics if `slot_duration * E::slots_per_epoch()` = 0
|
||||||
|
interval_at(
|
||||||
Instant::now() + duration_to_next_epoch + TIME_DELAY_FROM_SLOT,
|
Instant::now() + duration_to_next_epoch + TIME_DELAY_FROM_SLOT,
|
||||||
slot_duration * E::slots_per_epoch() as u32,
|
slot_duration * E::slots_per_epoch() as u32,
|
||||||
)
|
)
|
||||||
};
|
};
|
||||||
|
|
||||||
let (exit_signal, exit_fut) = exit_future::signal();
|
let (exit_signal, exit_fut) = exit_future::signal();
|
||||||
let service = self.clone();
|
|
||||||
let log_1 = log.clone();
|
|
||||||
let log_2 = log.clone();
|
|
||||||
|
|
||||||
// Run an immediate update before starting the updater service.
|
// Run an immediate update before starting the updater service.
|
||||||
self.context.executor.spawn(service.clone().do_update());
|
let service_1 = self.clone();
|
||||||
|
let service_2 = self.clone();
|
||||||
|
tokio::task::spawn(service_1.do_update());
|
||||||
|
|
||||||
self.context.executor.spawn(
|
let interval_fut = interval.for_each(move |_| {
|
||||||
exit_fut
|
let _ = service_2.clone().do_update();
|
||||||
.until(
|
futures::future::ready(())
|
||||||
interval
|
});
|
||||||
.map_err(move |e| {
|
|
||||||
crit! {
|
let future = futures::future::select(
|
||||||
log_1,
|
interval_fut,
|
||||||
"Timer thread failed";
|
exit_fut.map(move |_| info!(log, "Shutdown complete")),
|
||||||
"error" => format!("{}", e)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.for_each(move |_| service.do_update().then(|_| Ok(()))),
|
|
||||||
)
|
|
||||||
.map(move |_| info!(log_2, "Shutdown complete")),
|
|
||||||
);
|
);
|
||||||
|
tokio::task::spawn(future);
|
||||||
|
|
||||||
Ok(exit_signal)
|
Ok(exit_signal)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Attempts to download the `Fork` from the server.
|
/// Attempts to download the `Fork` from the server.
|
||||||
fn do_update(&self) -> impl Future<Item = (), Error = ()> {
|
async fn do_update(self) -> Result<(), ()> {
|
||||||
let service_1 = self.clone();
|
let log_1 = self.context.log.clone();
|
||||||
let log_1 = service_1.context.log.clone();
|
let log_2 = self.context.log.clone();
|
||||||
let log_2 = service_1.context.log.clone();
|
let _ = self
|
||||||
|
.inner
|
||||||
self.inner
|
|
||||||
.beacon_node
|
.beacon_node
|
||||||
.http
|
.http
|
||||||
.beacon()
|
.beacon()
|
||||||
.get_fork()
|
.get_fork()
|
||||||
.map(move |fork| *(service_1.fork.write()) = Some(fork))
|
.await
|
||||||
|
.map(move |fork| *(self.fork.write()) = Some(fork))
|
||||||
.map(move |_| trace!(log_1, "Fork update success"))
|
.map(move |_| trace!(log_1, "Fork update success"))
|
||||||
.map_err(move |e| {
|
.map_err(move |e| {
|
||||||
trace!(
|
trace!(
|
||||||
@@ -163,9 +157,9 @@ impl<T: SlotClock + 'static, E: EthSpec> ForkService<T, E> {
|
|||||||
"Fork update failed";
|
"Fork update failed";
|
||||||
"error" => format!("Error retrieving fork: {:?}", e)
|
"error" => format!("Error retrieving fork: {:?}", e)
|
||||||
)
|
)
|
||||||
})
|
});
|
||||||
// Returning an error will stop the interval. This is not desired, a single failure
|
// Returning an error will stop the interval. This is not desired, a single failure
|
||||||
// should not stop all future attempts.
|
// should not stop all future attempts.
|
||||||
.then(|_| Ok(()))
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -19,18 +19,13 @@ use duties_service::{DutiesService, DutiesServiceBuilder};
|
|||||||
use environment::RuntimeContext;
|
use environment::RuntimeContext;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use fork_service::{ForkService, ForkServiceBuilder};
|
use fork_service::{ForkService, ForkServiceBuilder};
|
||||||
use futures::{
|
|
||||||
future::{self, loop_fn, Loop},
|
|
||||||
Future, IntoFuture,
|
|
||||||
};
|
|
||||||
use notifier::spawn_notifier;
|
use notifier::spawn_notifier;
|
||||||
use remote_beacon_node::RemoteBeaconNode;
|
use remote_beacon_node::RemoteBeaconNode;
|
||||||
use slog::{error, info, Logger};
|
use slog::{error, info, Logger};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use slot_clock::SystemTimeSlotClock;
|
use slot_clock::SystemTimeSlotClock;
|
||||||
use std::time::{Duration, Instant};
|
|
||||||
use std::time::{SystemTime, UNIX_EPOCH};
|
use std::time::{SystemTime, UNIX_EPOCH};
|
||||||
use tokio::timer::Delay;
|
use tokio::time::{delay_for, Duration};
|
||||||
use types::EthSpec;
|
use types::EthSpec;
|
||||||
use validator_store::ValidatorStore;
|
use validator_store::ValidatorStore;
|
||||||
|
|
||||||
@@ -52,22 +47,18 @@ pub struct ProductionValidatorClient<T: EthSpec> {
|
|||||||
impl<T: EthSpec> ProductionValidatorClient<T> {
|
impl<T: EthSpec> ProductionValidatorClient<T> {
|
||||||
/// Instantiates the validator client, _without_ starting the timers to trigger block
|
/// Instantiates the validator client, _without_ starting the timers to trigger block
|
||||||
/// and attestation production.
|
/// and attestation production.
|
||||||
pub fn new_from_cli(
|
pub async fn new_from_cli(
|
||||||
context: RuntimeContext<T>,
|
context: RuntimeContext<T>,
|
||||||
cli_args: &ArgMatches,
|
cli_args: &ArgMatches<'_>,
|
||||||
) -> impl Future<Item = Self, Error = String> {
|
) -> Result<Self, String> {
|
||||||
Config::from_cli(&cli_args)
|
let config = Config::from_cli(&cli_args)
|
||||||
.into_future()
|
.map_err(|e| format!("Unable to initialize config: {}", e))?;
|
||||||
.map_err(|e| format!("Unable to initialize config: {}", e))
|
Self::new(context, config).await
|
||||||
.and_then(|config| Self::new(context, config))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Instantiates the validator client, _without_ starting the timers to trigger block
|
/// Instantiates the validator client, _without_ starting the timers to trigger block
|
||||||
/// and attestation production.
|
/// and attestation production.
|
||||||
pub fn new(
|
pub async fn new(mut context: RuntimeContext<T>, config: Config) -> Result<Self, String> {
|
||||||
mut context: RuntimeContext<T>,
|
|
||||||
config: Config,
|
|
||||||
) -> impl Future<Item = Self, Error = String> {
|
|
||||||
let log_1 = context.log.clone();
|
let log_1 = context.log.clone();
|
||||||
let log_2 = context.log.clone();
|
let log_2 = context.log.clone();
|
||||||
let log_3 = context.log.clone();
|
let log_3 = context.log.clone();
|
||||||
@@ -80,184 +71,153 @@ impl<T: EthSpec> ProductionValidatorClient<T> {
|
|||||||
"datadir" => format!("{:?}", config.data_dir),
|
"datadir" => format!("{:?}", config.data_dir),
|
||||||
);
|
);
|
||||||
|
|
||||||
RemoteBeaconNode::new_with_timeout(config.http_server.clone(), HTTP_TIMEOUT)
|
let beacon_node =
|
||||||
.map_err(|e| format!("Unable to init beacon node http client: {}", e))
|
RemoteBeaconNode::new_with_timeout(config.http_server.clone(), HTTP_TIMEOUT)
|
||||||
.into_future()
|
.map_err(|e| format!("Unable to init beacon node http client: {}", e))?;
|
||||||
.and_then(move |beacon_node| wait_for_node(beacon_node, log_2))
|
|
||||||
.and_then(|beacon_node| {
|
|
||||||
beacon_node
|
|
||||||
.http
|
|
||||||
.spec()
|
|
||||||
.get_eth2_config()
|
|
||||||
.map(|eth2_config| (beacon_node, eth2_config))
|
|
||||||
.map_err(|e| format!("Unable to read eth2 config from beacon node: {:?}", e))
|
|
||||||
})
|
|
||||||
.and_then(|(beacon_node, eth2_config)| {
|
|
||||||
beacon_node
|
|
||||||
.http
|
|
||||||
.beacon()
|
|
||||||
.get_genesis_time()
|
|
||||||
.map(|genesis_time| (beacon_node, eth2_config, genesis_time))
|
|
||||||
.map_err(|e| format!("Unable to read genesis time from beacon node: {:?}", e))
|
|
||||||
})
|
|
||||||
.and_then(move |(beacon_node, remote_eth2_config, genesis_time)| {
|
|
||||||
SystemTime::now()
|
|
||||||
.duration_since(UNIX_EPOCH)
|
|
||||||
.into_future()
|
|
||||||
.map_err(|e| format!("Unable to read system time: {:?}", e))
|
|
||||||
.and_then::<_, Box<dyn Future<Item = _, Error = _> + Send>>(move |now| {
|
|
||||||
let log = log_3.clone();
|
|
||||||
let genesis = Duration::from_secs(genesis_time);
|
|
||||||
|
|
||||||
// If the time now is less than (prior to) genesis, then delay until the
|
// TODO: check if all logs in wait_for_node are produed while awaiting
|
||||||
// genesis instant.
|
let beacon_node = wait_for_node(beacon_node, log_2).await?;
|
||||||
//
|
let eth2_config = beacon_node
|
||||||
// If the validator client starts before genesis, it will get errors from
|
.http
|
||||||
// the slot clock.
|
.spec()
|
||||||
if now < genesis {
|
.get_eth2_config()
|
||||||
info!(
|
.await
|
||||||
log,
|
.map_err(|e| format!("Unable to read eth2 config from beacon node: {:?}", e))?;
|
||||||
"Starting node prior to genesis";
|
let genesis_time = beacon_node
|
||||||
"seconds_to_wait" => (genesis - now).as_secs()
|
.http
|
||||||
);
|
.beacon()
|
||||||
|
.get_genesis_time()
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("Unable to read genesis time from beacon node: {:?}", e))?;
|
||||||
|
let now = SystemTime::now()
|
||||||
|
.duration_since(UNIX_EPOCH)
|
||||||
|
.map_err(|e| format!("Unable to read system time: {:?}", e))?;
|
||||||
|
let log = log_3.clone();
|
||||||
|
let genesis = Duration::from_secs(genesis_time);
|
||||||
|
|
||||||
Box::new(
|
// If the time now is less than (prior to) genesis, then delay until the
|
||||||
Delay::new(Instant::now() + (genesis - now))
|
// genesis instant.
|
||||||
.map_err(|e| {
|
//
|
||||||
format!("Unable to create genesis wait delay: {:?}", e)
|
// If the validator client starts before genesis, it will get errors from
|
||||||
})
|
// the slot clock.
|
||||||
.map(move |_| (beacon_node, remote_eth2_config, genesis_time)),
|
if now < genesis {
|
||||||
)
|
info!(
|
||||||
} else {
|
log,
|
||||||
info!(
|
"Starting node prior to genesis";
|
||||||
log,
|
"seconds_to_wait" => (genesis - now).as_secs()
|
||||||
"Genesis has already occurred";
|
);
|
||||||
"seconds_ago" => (now - genesis).as_secs()
|
|
||||||
);
|
|
||||||
|
|
||||||
Box::new(future::ok((beacon_node, remote_eth2_config, genesis_time)))
|
delay_for(genesis - now).await
|
||||||
}
|
} else {
|
||||||
})
|
info!(
|
||||||
})
|
log,
|
||||||
.and_then(|(beacon_node, eth2_config, genesis_time)| {
|
"Genesis has already occurred";
|
||||||
beacon_node
|
"seconds_ago" => (now - genesis).as_secs()
|
||||||
.http
|
);
|
||||||
.beacon()
|
}
|
||||||
.get_genesis_validators_root()
|
let genesis_validators_root = beacon_node
|
||||||
.map(move |genesis_validators_root| {
|
.http
|
||||||
(
|
.beacon()
|
||||||
beacon_node,
|
.get_genesis_validators_root()
|
||||||
eth2_config,
|
.await
|
||||||
genesis_time,
|
.map_err(|e| {
|
||||||
genesis_validators_root,
|
format!(
|
||||||
)
|
"Unable to read genesis validators root from beacon node: {:?}",
|
||||||
})
|
e
|
||||||
.map_err(|e| {
|
)
|
||||||
format!(
|
})?;
|
||||||
"Unable to read genesis validators root from beacon node: {:?}",
|
let log = log_4.clone();
|
||||||
e
|
|
||||||
)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
.and_then(
|
|
||||||
move |(beacon_node, remote_eth2_config, genesis_time, genesis_validators_root)| {
|
|
||||||
let log = log_4.clone();
|
|
||||||
|
|
||||||
// Do not permit a connection to a beacon node using different spec constants.
|
// Do not permit a connection to a beacon node using different spec constants.
|
||||||
if context.eth2_config.spec_constants != remote_eth2_config.spec_constants {
|
if context.eth2_config.spec_constants != eth2_config.spec_constants {
|
||||||
return Err(format!(
|
return Err(format!(
|
||||||
"Beacon node is using an incompatible spec. Got {}, expected {}",
|
"Beacon node is using an incompatible spec. Got {}, expected {}",
|
||||||
remote_eth2_config.spec_constants, context.eth2_config.spec_constants
|
eth2_config.spec_constants, context.eth2_config.spec_constants
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note: here we just assume the spec variables of the remote node. This is very useful
|
// Note: here we just assume the spec variables of the remote node. This is very useful
|
||||||
// for testnets, but perhaps a security issue when it comes to mainnet.
|
// for testnets, but perhaps a security issue when it comes to mainnet.
|
||||||
//
|
//
|
||||||
// A damaging attack would be for a beacon node to convince the validator client of a
|
// A damaging attack would be for a beacon node to convince the validator client of a
|
||||||
// different `SLOTS_PER_EPOCH` variable. This could result in slashable messages being
|
// different `SLOTS_PER_EPOCH` variable. This could result in slashable messages being
|
||||||
// produced. We are safe from this because `SLOTS_PER_EPOCH` is a type-level constant
|
// produced. We are safe from this because `SLOTS_PER_EPOCH` is a type-level constant
|
||||||
// for Lighthouse.
|
// for Lighthouse.
|
||||||
context.eth2_config = remote_eth2_config;
|
context.eth2_config = eth2_config;
|
||||||
|
|
||||||
let slot_clock = SystemTimeSlotClock::new(
|
let slot_clock = SystemTimeSlotClock::new(
|
||||||
context.eth2_config.spec.genesis_slot,
|
context.eth2_config.spec.genesis_slot,
|
||||||
Duration::from_secs(genesis_time),
|
Duration::from_secs(genesis_time),
|
||||||
Duration::from_millis(context.eth2_config.spec.milliseconds_per_slot),
|
Duration::from_millis(context.eth2_config.spec.milliseconds_per_slot),
|
||||||
);
|
);
|
||||||
|
|
||||||
let fork_service = ForkServiceBuilder::new()
|
let fork_service = ForkServiceBuilder::new()
|
||||||
.slot_clock(slot_clock.clone())
|
.slot_clock(slot_clock.clone())
|
||||||
.beacon_node(beacon_node.clone())
|
.beacon_node(beacon_node.clone())
|
||||||
.runtime_context(context.service_context("fork".into()))
|
.runtime_context(context.service_context("fork".into()))
|
||||||
.build()?;
|
.build()?;
|
||||||
|
|
||||||
let validator_store: ValidatorStore<SystemTimeSlotClock, T> =
|
let validator_store: ValidatorStore<SystemTimeSlotClock, T> = match &config.key_source {
|
||||||
match &config.key_source {
|
// Load pre-existing validators from the data dir.
|
||||||
// Load pre-existing validators from the data dir.
|
//
|
||||||
//
|
// Use the `account_manager` to generate these files.
|
||||||
// Use the `account_manager` to generate these files.
|
KeySource::Disk => ValidatorStore::load_from_disk(
|
||||||
KeySource::Disk => ValidatorStore::load_from_disk(
|
config.data_dir.clone(),
|
||||||
config.data_dir.clone(),
|
genesis_validators_root,
|
||||||
genesis_validators_root,
|
context.eth2_config.spec.clone(),
|
||||||
context.eth2_config.spec.clone(),
|
fork_service.clone(),
|
||||||
fork_service.clone(),
|
log.clone(),
|
||||||
log.clone(),
|
)?,
|
||||||
)?,
|
// Generate ephemeral insecure keypairs for testing purposes.
|
||||||
// Generate ephemeral insecure keypairs for testing purposes.
|
//
|
||||||
//
|
// Do not use in production.
|
||||||
// Do not use in production.
|
KeySource::InsecureKeypairs(indices) => ValidatorStore::insecure_ephemeral_validators(
|
||||||
KeySource::InsecureKeypairs(indices) => {
|
&indices,
|
||||||
ValidatorStore::insecure_ephemeral_validators(
|
genesis_validators_root,
|
||||||
&indices,
|
context.eth2_config.spec.clone(),
|
||||||
genesis_validators_root,
|
fork_service.clone(),
|
||||||
context.eth2_config.spec.clone(),
|
log.clone(),
|
||||||
fork_service.clone(),
|
)?,
|
||||||
log.clone(),
|
};
|
||||||
)?
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
info!(
|
info!(
|
||||||
log,
|
log,
|
||||||
"Loaded validator keypair store";
|
"Loaded validator keypair store";
|
||||||
"voting_validators" => validator_store.num_voting_validators()
|
"voting_validators" => validator_store.num_voting_validators()
|
||||||
);
|
);
|
||||||
|
|
||||||
let duties_service = DutiesServiceBuilder::new()
|
let duties_service = DutiesServiceBuilder::new()
|
||||||
.slot_clock(slot_clock.clone())
|
.slot_clock(slot_clock.clone())
|
||||||
.validator_store(validator_store.clone())
|
.validator_store(validator_store.clone())
|
||||||
.beacon_node(beacon_node.clone())
|
.beacon_node(beacon_node.clone())
|
||||||
.runtime_context(context.service_context("duties".into()))
|
.runtime_context(context.service_context("duties".into()))
|
||||||
.allow_unsynced_beacon_node(config.allow_unsynced_beacon_node)
|
.allow_unsynced_beacon_node(config.allow_unsynced_beacon_node)
|
||||||
.build()?;
|
.build()?;
|
||||||
|
|
||||||
let block_service = BlockServiceBuilder::new()
|
let block_service = BlockServiceBuilder::new()
|
||||||
.duties_service(duties_service.clone())
|
.duties_service(duties_service.clone())
|
||||||
.slot_clock(slot_clock.clone())
|
.slot_clock(slot_clock.clone())
|
||||||
.validator_store(validator_store.clone())
|
.validator_store(validator_store.clone())
|
||||||
.beacon_node(beacon_node.clone())
|
.beacon_node(beacon_node.clone())
|
||||||
.runtime_context(context.service_context("block".into()))
|
.runtime_context(context.service_context("block".into()))
|
||||||
.build()?;
|
.build()?;
|
||||||
|
|
||||||
let attestation_service = AttestationServiceBuilder::new()
|
let attestation_service = AttestationServiceBuilder::new()
|
||||||
.duties_service(duties_service.clone())
|
.duties_service(duties_service.clone())
|
||||||
.slot_clock(slot_clock)
|
.slot_clock(slot_clock)
|
||||||
.validator_store(validator_store)
|
.validator_store(validator_store)
|
||||||
.beacon_node(beacon_node)
|
.beacon_node(beacon_node)
|
||||||
.runtime_context(context.service_context("attestation".into()))
|
.runtime_context(context.service_context("attestation".into()))
|
||||||
.build()?;
|
.build()?;
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
context,
|
context,
|
||||||
duties_service,
|
duties_service,
|
||||||
fork_service,
|
fork_service,
|
||||||
block_service,
|
block_service,
|
||||||
attestation_service,
|
attestation_service,
|
||||||
exit_signals: vec![],
|
exit_signals: vec![],
|
||||||
})
|
})
|
||||||
},
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn start_service(&mut self) -> Result<(), String> {
|
pub fn start_service(&mut self) -> Result<(), String> {
|
||||||
@@ -298,48 +258,39 @@ impl<T: EthSpec> ProductionValidatorClient<T> {
|
|||||||
|
|
||||||
/// Request the version from the node, looping back and trying again on failure. Exit once the node
|
/// Request the version from the node, looping back and trying again on failure. Exit once the node
|
||||||
/// has been contacted.
|
/// has been contacted.
|
||||||
fn wait_for_node<E: EthSpec>(
|
async fn wait_for_node<E: EthSpec>(
|
||||||
beacon_node: RemoteBeaconNode<E>,
|
beacon_node: RemoteBeaconNode<E>,
|
||||||
log: Logger,
|
log: Logger,
|
||||||
) -> impl Future<Item = RemoteBeaconNode<E>, Error = String> {
|
) -> Result<RemoteBeaconNode<E>, String> {
|
||||||
// Try to get the version string from the node, looping until success is returned.
|
// Try to get the version string from the node, looping until success is returned.
|
||||||
loop_fn(beacon_node.clone(), move |beacon_node| {
|
loop {
|
||||||
let log = log.clone();
|
let log = log.clone();
|
||||||
beacon_node
|
let result = beacon_node
|
||||||
.clone()
|
.clone()
|
||||||
.http
|
.http
|
||||||
.node()
|
.node()
|
||||||
.get_version()
|
.get_version()
|
||||||
.map_err(|e| format!("{:?}", e))
|
.await
|
||||||
.then(move |result| {
|
.map_err(|e| format!("{:?}", e));
|
||||||
let future: Box<dyn Future<Item = Loop<_, _>, Error = String> + Send> = match result
|
|
||||||
{
|
|
||||||
Ok(version) => {
|
|
||||||
info!(
|
|
||||||
log,
|
|
||||||
"Connected to beacon node";
|
|
||||||
"version" => version,
|
|
||||||
);
|
|
||||||
|
|
||||||
Box::new(future::ok(Loop::Break(beacon_node)))
|
match result {
|
||||||
}
|
Ok(version) => {
|
||||||
Err(e) => {
|
info!(
|
||||||
error!(
|
log,
|
||||||
log,
|
"Connected to beacon node";
|
||||||
"Unable to connect to beacon node";
|
"version" => version,
|
||||||
"error" => format!("{:?}", e),
|
);
|
||||||
);
|
|
||||||
|
|
||||||
Box::new(
|
return Ok(beacon_node);
|
||||||
Delay::new(Instant::now() + RETRY_DELAY)
|
}
|
||||||
.map_err(|e| format!("Failed to trigger delay: {:?}", e))
|
Err(e) => {
|
||||||
.and_then(|_| future::ok(Loop::Continue(beacon_node))),
|
error!(
|
||||||
)
|
log,
|
||||||
}
|
"Unable to connect to beacon node";
|
||||||
};
|
"error" => format!("{:?}", e),
|
||||||
|
);
|
||||||
future
|
delay_for(RETRY_DELAY).await;
|
||||||
})
|
}
|
||||||
})
|
}
|
||||||
.map(|_| beacon_node)
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
use crate::ProductionValidatorClient;
|
use crate::ProductionValidatorClient;
|
||||||
use exit_future::Signal;
|
use exit_future::Signal;
|
||||||
use futures::{Future, Stream};
|
use futures::{FutureExt, StreamExt};
|
||||||
use slog::{error, info};
|
use slog::{error, info};
|
||||||
use slot_clock::SlotClock;
|
use slot_clock::SlotClock;
|
||||||
use std::time::{Duration, Instant};
|
use tokio::time::{interval_at, Duration, Instant};
|
||||||
use tokio::timer::Interval;
|
|
||||||
use types::EthSpec;
|
use types::EthSpec;
|
||||||
|
|
||||||
/// Spawns a notifier service which periodically logs information about the node.
|
/// Spawns a notifier service which periodically logs information about the node.
|
||||||
@@ -26,66 +25,63 @@ pub fn spawn_notifier<T: EthSpec>(client: &ProductionValidatorClient<T>) -> Resu
|
|||||||
|
|
||||||
let duties_service = client.duties_service.clone();
|
let duties_service = client.duties_service.clone();
|
||||||
let log_1 = context.log.clone();
|
let log_1 = context.log.clone();
|
||||||
let log_2 = context.log.clone();
|
|
||||||
|
|
||||||
let interval_future = Interval::new(start_instant, interval_duration)
|
// Note: interval_at panics if `interval_duration` is 0
|
||||||
.map_err(
|
let interval_fut = interval_at(start_instant, interval_duration).for_each(move |_| {
|
||||||
move |e| error!(log_1, "Slot notifier timer failed"; "error" => format!("{:?}", e)),
|
let log = log_1.clone();
|
||||||
)
|
|
||||||
.for_each(move |_| {
|
|
||||||
let log = log_2.clone();
|
|
||||||
|
|
||||||
if let Some(slot) = duties_service.slot_clock.now() {
|
if let Some(slot) = duties_service.slot_clock.now() {
|
||||||
let epoch = slot.epoch(T::slots_per_epoch());
|
let epoch = slot.epoch(T::slots_per_epoch());
|
||||||
|
|
||||||
let total_validators = duties_service.total_validator_count();
|
let total_validators = duties_service.total_validator_count();
|
||||||
let proposing_validators = duties_service.proposer_count(epoch);
|
let proposing_validators = duties_service.proposer_count(epoch);
|
||||||
let attesting_validators = duties_service.attester_count(epoch);
|
let attesting_validators = duties_service.attester_count(epoch);
|
||||||
|
|
||||||
if total_validators == 0 {
|
if total_validators == 0 {
|
||||||
error!(log, "No validators present")
|
error!(log, "No validators present")
|
||||||
} else if total_validators == attesting_validators {
|
} else if total_validators == attesting_validators {
|
||||||
info!(
|
info!(
|
||||||
log_2,
|
log_1,
|
||||||
"All validators active";
|
"All validators active";
|
||||||
"proposers" => proposing_validators,
|
"proposers" => proposing_validators,
|
||||||
"active_validators" => attesting_validators,
|
"active_validators" => attesting_validators,
|
||||||
"total_validators" => total_validators,
|
"total_validators" => total_validators,
|
||||||
"epoch" => format!("{}", epoch),
|
"epoch" => format!("{}", epoch),
|
||||||
"slot" => format!("{}", slot),
|
"slot" => format!("{}", slot),
|
||||||
);
|
);
|
||||||
} else if attesting_validators > 0 {
|
} else if attesting_validators > 0 {
|
||||||
info!(
|
info!(
|
||||||
log_2,
|
log_1,
|
||||||
"Some validators active";
|
"Some validators active";
|
||||||
"proposers" => proposing_validators,
|
"proposers" => proposing_validators,
|
||||||
"active_validators" => attesting_validators,
|
"active_validators" => attesting_validators,
|
||||||
"total_validators" => total_validators,
|
"total_validators" => total_validators,
|
||||||
"epoch" => format!("{}", epoch),
|
"epoch" => format!("{}", epoch),
|
||||||
"slot" => format!("{}", slot),
|
"slot" => format!("{}", slot),
|
||||||
);
|
);
|
||||||
} else {
|
|
||||||
info!(
|
|
||||||
log_2,
|
|
||||||
"Awaiting activation";
|
|
||||||
"validators" => total_validators,
|
|
||||||
"epoch" => format!("{}", epoch),
|
|
||||||
"slot" => format!("{}", slot),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
error!(log, "Unable to read slot clock");
|
info!(
|
||||||
|
log_1,
|
||||||
|
"Awaiting activation";
|
||||||
|
"validators" => total_validators,
|
||||||
|
"epoch" => format!("{}", epoch),
|
||||||
|
"slot" => format!("{}", slot),
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
error!(log, "Unable to read slot clock");
|
||||||
|
}
|
||||||
|
|
||||||
Ok(())
|
futures::future::ready(())
|
||||||
});
|
});
|
||||||
|
|
||||||
let (exit_signal, exit) = exit_future::signal();
|
let (exit_signal, exit) = exit_future::signal();
|
||||||
let log = context.log.clone();
|
let log = context.log.clone();
|
||||||
client.context.executor.spawn(
|
let future = futures::future::select(
|
||||||
exit.until(interval_future)
|
interval_fut,
|
||||||
.map(move |_| info!(log, "Shutdown complete")),
|
exit.map(move |_| info!(log, "Shutdown complete")),
|
||||||
);
|
);
|
||||||
|
tokio::task::spawn(future);
|
||||||
|
|
||||||
Ok(exit_signal)
|
Ok(exit_signal)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
use bls::get_withdrawal_credentials;
|
use bls::get_withdrawal_credentials;
|
||||||
use deposit_contract::{encode_eth1_tx_data, DEPOSIT_GAS};
|
use deposit_contract::{encode_eth1_tx_data, DEPOSIT_GAS};
|
||||||
use futures::{Future, IntoFuture};
|
use futures::compat::Future01CompatExt;
|
||||||
use hex;
|
use hex;
|
||||||
use ssz::{Decode, Encode};
|
use ssz::{Decode, Encode};
|
||||||
use ssz_derive::{Decode, Encode};
|
use ssz_derive::{Decode, Encode};
|
||||||
@@ -303,29 +303,27 @@ impl ValidatorDirectoryBuilder {
|
|||||||
Ok(self)
|
Ok(self)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub 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,
|
||||||
) -> impl Future<Item = (Self, Hash256), Error = String> {
|
) -> Result<Hash256, String> {
|
||||||
self.get_deposit_data()
|
let (deposit_data, deposit_amount) = self.get_deposit_data()?;
|
||||||
.into_future()
|
web3.eth()
|
||||||
.and_then(move |(deposit_data, deposit_amount)| {
|
.send_transaction(TransactionRequest {
|
||||||
web3.eth()
|
from,
|
||||||
.send_transaction(TransactionRequest {
|
to: Some(deposit_contract),
|
||||||
from,
|
gas: Some(DEPOSIT_GAS.into()),
|
||||||
to: Some(deposit_contract),
|
gas_price: None,
|
||||||
gas: Some(DEPOSIT_GAS.into()),
|
value: Some(from_gwei(deposit_amount)),
|
||||||
gas_price: None,
|
data: Some(deposit_data.into()),
|
||||||
value: Some(from_gwei(deposit_amount)),
|
nonce: None,
|
||||||
data: Some(deposit_data.into()),
|
condition: None,
|
||||||
nonce: None,
|
|
||||||
condition: None,
|
|
||||||
})
|
|
||||||
.map_err(|e| format!("Failed to send transaction: {:?}", e))
|
|
||||||
})
|
})
|
||||||
.map(|tx| (self, tx))
|
.compat()
|
||||||
|
.await
|
||||||
|
.map_err(|e| format!("Failed to send transaction: {:?}", e))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn build(self) -> Result<ValidatorDirectory, String> {
|
pub fn build(self) -> Result<ValidatorDirectory, String> {
|
||||||
|
|||||||
Reference in New Issue
Block a user