2020-09-29 03:46:54 +00:00
|
|
|
use crate::state_id::StateId;
|
|
|
|
use beacon_chain::{BeaconChain, BeaconChainTypes};
|
|
|
|
use eth2::{
|
|
|
|
lighthouse::{GlobalValidatorInclusionData, ValidatorInclusionData},
|
|
|
|
types::ValidatorId,
|
|
|
|
};
|
2021-07-27 07:01:01 +00:00
|
|
|
use state_processing::per_epoch_processing::{
|
|
|
|
altair::participation_cache::Error as ParticipationCacheError, process_epoch,
|
|
|
|
EpochProcessingSummary,
|
|
|
|
};
|
|
|
|
use types::{BeaconState, ChainSpec, Epoch, EthSpec};
|
2020-09-29 03:46:54 +00:00
|
|
|
|
2021-07-27 07:01:01 +00:00
|
|
|
/// Returns the state in the last slot of `epoch`.
|
|
|
|
fn end_of_epoch_state<T: BeaconChainTypes>(
|
2020-09-29 03:46:54 +00:00
|
|
|
epoch: Epoch,
|
|
|
|
chain: &BeaconChain<T>,
|
2021-07-27 07:01:01 +00:00
|
|
|
) -> Result<BeaconState<T::EthSpec>, warp::reject::Rejection> {
|
2020-09-29 03:46:54 +00:00
|
|
|
let target_slot = epoch.end_slot(T::EthSpec::slots_per_epoch());
|
2022-07-25 08:23:00 +00:00
|
|
|
// The execution status is not returned, any functions which rely upon this method might return
|
|
|
|
// optimistic information without explicitly declaring so.
|
|
|
|
let (state, _execution_status) = StateId::from_slot(target_slot).state(chain)?;
|
|
|
|
Ok(state)
|
2021-07-27 07:01:01 +00:00
|
|
|
}
|
2020-09-29 03:46:54 +00:00
|
|
|
|
2021-07-27 07:01:01 +00:00
|
|
|
/// Generate an `EpochProcessingSummary` for `state`.
|
|
|
|
///
|
|
|
|
/// ## Notes
|
|
|
|
///
|
|
|
|
/// Will mutate `state`, transitioning it to the next epoch.
|
|
|
|
fn get_epoch_processing_summary<T: EthSpec>(
|
|
|
|
state: &mut BeaconState<T>,
|
|
|
|
spec: &ChainSpec,
|
2021-08-31 23:31:36 +00:00
|
|
|
) -> Result<EpochProcessingSummary<T>, warp::reject::Rejection> {
|
2021-07-27 07:01:01 +00:00
|
|
|
process_epoch(state, spec)
|
|
|
|
.map_err(|e| warp_utils::reject::custom_server_error(format!("{:?}", e)))
|
|
|
|
}
|
2020-09-29 03:46:54 +00:00
|
|
|
|
2021-07-27 07:01:01 +00:00
|
|
|
fn convert_cache_error(error: ParticipationCacheError) -> warp::reject::Rejection {
|
|
|
|
warp_utils::reject::custom_server_error(format!("{:?}", error))
|
|
|
|
}
|
2020-09-29 03:46:54 +00:00
|
|
|
|
2021-07-27 07:01:01 +00:00
|
|
|
/// Returns information about *all validators* (i.e., global) and how they performed during a given
|
|
|
|
/// epoch.
|
|
|
|
pub fn global_validator_inclusion_data<T: BeaconChainTypes>(
|
|
|
|
epoch: Epoch,
|
|
|
|
chain: &BeaconChain<T>,
|
|
|
|
) -> Result<GlobalValidatorInclusionData, warp::Rejection> {
|
|
|
|
let mut state = end_of_epoch_state(epoch, chain)?;
|
|
|
|
let summary = get_epoch_processing_summary(&mut state, &chain.spec)?;
|
2020-09-29 03:46:54 +00:00
|
|
|
|
|
|
|
Ok(GlobalValidatorInclusionData {
|
2021-07-27 07:01:01 +00:00
|
|
|
current_epoch_active_gwei: summary.current_epoch_total_active_balance(),
|
|
|
|
previous_epoch_active_gwei: summary.previous_epoch_total_active_balance(),
|
|
|
|
current_epoch_target_attesting_gwei: summary
|
|
|
|
.current_epoch_target_attesting_balance()
|
|
|
|
.map_err(convert_cache_error)?,
|
|
|
|
previous_epoch_target_attesting_gwei: summary
|
|
|
|
.previous_epoch_target_attesting_balance()
|
|
|
|
.map_err(convert_cache_error)?,
|
|
|
|
previous_epoch_head_attesting_gwei: summary
|
|
|
|
.previous_epoch_head_attesting_balance()
|
|
|
|
.map_err(convert_cache_error)?,
|
2020-09-29 03:46:54 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns information about a single validator and how it performed during a given epoch.
|
|
|
|
pub fn validator_inclusion_data<T: BeaconChainTypes>(
|
|
|
|
epoch: Epoch,
|
|
|
|
validator_id: &ValidatorId,
|
|
|
|
chain: &BeaconChain<T>,
|
|
|
|
) -> Result<Option<ValidatorInclusionData>, warp::Rejection> {
|
2021-07-27 07:01:01 +00:00
|
|
|
let mut state = end_of_epoch_state(epoch, chain)?;
|
2020-09-29 03:46:54 +00:00
|
|
|
|
|
|
|
state
|
|
|
|
.update_pubkey_cache()
|
|
|
|
.map_err(warp_utils::reject::beacon_state_error)?;
|
|
|
|
|
|
|
|
let validator_index = match validator_id {
|
|
|
|
ValidatorId::Index(index) => *index as usize,
|
|
|
|
ValidatorId::PublicKey(pubkey) => {
|
|
|
|
if let Some(index) = state
|
|
|
|
.get_validator_index(pubkey)
|
|
|
|
.map_err(warp_utils::reject::beacon_state_error)?
|
|
|
|
{
|
|
|
|
index
|
|
|
|
} else {
|
|
|
|
return Ok(None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-27 07:01:01 +00:00
|
|
|
// Obtain the validator *before* transitioning the state into the next epoch.
|
|
|
|
let validator = if let Ok(validator) = state.get_validator(validator_index) {
|
|
|
|
validator.clone()
|
|
|
|
} else {
|
|
|
|
return Ok(None);
|
|
|
|
};
|
|
|
|
|
|
|
|
let summary = get_epoch_processing_summary(&mut state, &chain.spec)?;
|
|
|
|
|
|
|
|
Ok(Some(ValidatorInclusionData {
|
|
|
|
is_slashed: validator.slashed,
|
|
|
|
is_withdrawable_in_current_epoch: validator.is_withdrawable_at(epoch),
|
|
|
|
is_active_unslashed_in_current_epoch: summary
|
|
|
|
.is_active_unslashed_in_current_epoch(validator_index),
|
|
|
|
is_active_unslashed_in_previous_epoch: summary
|
|
|
|
.is_active_unslashed_in_previous_epoch(validator_index),
|
|
|
|
current_epoch_effective_balance_gwei: validator.effective_balance,
|
|
|
|
is_current_epoch_target_attester: summary
|
|
|
|
.is_current_epoch_target_attester(validator_index)
|
|
|
|
.map_err(convert_cache_error)?,
|
|
|
|
is_previous_epoch_target_attester: summary
|
|
|
|
.is_previous_epoch_target_attester(validator_index)
|
|
|
|
.map_err(convert_cache_error)?,
|
|
|
|
is_previous_epoch_head_attester: summary
|
|
|
|
.is_previous_epoch_head_attester(validator_index)
|
|
|
|
.map_err(convert_cache_error)?,
|
|
|
|
}))
|
2020-09-29 03:46:54 +00:00
|
|
|
}
|