2018-08-21 18:21:45 +02:00
# beacon_chain
2019-02-25 06:48:36 -08:00
# Copyright (c) 2018-2019 Status Research & Development GmbH
2018-08-21 18:21:45 +02:00
# Licensed and distributed under either of
# * MIT license (license terms in the root directory or at http://opensource.org/licenses/MIT).
# * Apache v2 license (license terms in the root directory or at http://www.apache.org/licenses/LICENSE-2.0).
# at your option. This file may not be copied, modified, or distributed except according to those terms.
2018-12-04 12:45:30 -06:00
# State transition, as described in
# https://github.com/ethereum/eth2.0-specs/blob/master/specs/core/0_beacon-chain.md#beacon-chain-state-transition-function
2018-08-21 18:21:45 +02:00
#
2018-12-04 12:45:30 -06:00
# The purpose of this code right is primarily educational, to help piece
# together the mechanics of the beacon state and to discover potential problem
2018-12-13 10:00:55 -06:00
# areas. The entry point is `updateState` which is at the bottom of the file!
2018-12-04 12:45:30 -06:00
#
# General notes about the code (TODO):
# * It's inefficient - we quadratically copy, allocate and iterate when there
# are faster options
# * Weird styling - the sections taken from the spec use python styling while
# the others use NEP-1 - helps grepping identifiers in spec
# * We mix procedural and functional styles for no good reason, except that the
# spec does so also.
2019-04-05 14:18:13 +00:00
# * There are likely lots of bugs.
2019-01-28 20:15:00 -08:00
# * For indices, we get a mix of uint64, ValidatorIndex and int - this is currently
2018-12-04 12:45:30 -06:00
# swept under the rug with casts
# * The spec uses uint64 for data types, but functions in the spec often assume
# signed bigint semantics - under- and overflows ensue
# * Sane error handling is missing in most cases (yay, we'll get the chance to
# debate exceptions again!)
# When updating the code, add TODO sections to mark where there are clear
# improvements to be made - other than that, keep things similar to spec for
# now.
2018-08-21 18:21:45 +02:00
2018-09-26 18:26:39 +02:00
import
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
algorithm , collections / sets , chronicles , math , options , sequtils , sets , tables ,
2019-04-05 14:18:13 +00:00
. / extras , . / ssz , . / beacon_node_types ,
2019-03-20 14:01:48 -06:00
. / spec / [ beaconstate , bitfield , crypto , datatypes , digest , helpers , validator ]
2018-11-28 23:23:40 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#block-header
2019-03-16 19:52:37 +00:00
proc processBlockHeader (
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
# Verify that the slots match
if not ( blck . slot = = state . slot ) :
notice " Block header: slot mismatch " ,
block_slot = humaneSlotNum ( blck . slot ) ,
state_slot = humaneSlotNum ( state . slot )
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify that the parent matches
2019-03-26 19:32:35 -06:00
if skipValidation notin flags and not ( blck . previous_block_root = =
2019-05-09 12:27:37 +00:00
signing_root ( state . latest_block_header ) ) :
2019-03-26 19:32:35 -06:00
notice " Block header: previous block root mismatch " ,
latest_block_header = state . latest_block_header ,
blck = shortLog ( blck ) ,
2019-05-09 12:27:37 +00:00
latest_block_header_root = shortLog ( signing_root ( state . latest_block_header ) )
2019-03-26 19:32:35 -06:00
return false
2019-03-16 19:52:37 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Save current block as the new latest block
state . latest_block_header = BeaconBlockHeader (
slot : blck . slot ,
previous_block_root : blck . previous_block_root ,
block_body_root : hash_tree_root ( blck . body ) ,
)
# Verify proposer is not slashed
let proposer = state . validator_registry [ get_beacon_proposer_index ( state ) ]
if proposer . slashed :
notice " Block header: proposer slashed "
return false
2019-03-16 19:52:37 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify proposer signature
2019-03-16 19:52:37 +00:00
if skipValidation notin flags and not bls_verify (
proposer . pubkey ,
2019-05-09 12:27:37 +00:00
signing_root ( blck ) . data ,
2019-03-16 19:52:37 +00:00
blck . signature ,
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_domain ( state , DOMAIN_BEACON_PROPOSER ) ) :
2019-03-16 19:52:37 +00:00
notice " Block header: invalid block header " ,
proposer_pubkey = proposer . pubkey ,
2019-05-09 12:27:37 +00:00
block_root = shortLog ( signing_root ( blck ) ) ,
2019-03-16 19:52:37 +00:00
block_signature = blck . signature
return false
true
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#randao
2019-01-25 11:35:22 -06:00
proc processRandao (
2018-12-21 16:37:46 -06:00
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
2018-12-04 12:45:30 -06:00
let
2019-05-23 11:13:02 +00:00
proposer_index = get_beacon_proposer_index ( state )
2018-12-11 11:55:45 -06:00
proposer = addr state . validator_registry [ proposer_index ]
2018-11-29 16:11:05 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify that the provided randao value is valid
2018-12-21 16:37:46 -06:00
if skipValidation notin flags :
2019-02-14 13:41:04 -08:00
if not bls_verify (
proposer . pubkey ,
2019-03-25 10:46:31 -06:00
hash_tree_root ( get_current_epoch ( state ) . uint64 ) . data ,
2019-03-16 19:52:37 +00:00
blck . body . randao_reveal ,
2019-04-29 16:48:30 +00:00
get_domain ( state , DOMAIN_RANDAO , get_current_epoch ( state ) ) ) :
2019-02-14 13:41:04 -08:00
notice " Randao mismatch " , proposer_pubkey = proposer . pubkey ,
message = get_current_epoch ( state ) ,
2019-03-16 19:52:37 +00:00
signature = blck . body . randao_reveal ,
2019-02-14 13:41:04 -08:00
slot = state . slot ,
blck_slot = blck . slot
return false
2018-11-29 16:11:05 -06:00
2019-03-16 19:52:37 +00:00
# Mix it in
2019-02-07 10:43:21 +00:00
let
mix = get_current_epoch ( state ) mod LATEST_RANDAO_MIXES_LENGTH
2019-03-16 19:52:37 +00:00
rr = eth2hash ( blck . body . randao_reveal . getBytes ( ) ) . data
2018-11-29 16:11:05 -06:00
2019-02-07 10:43:21 +00:00
for i , b in state . latest_randao_mixes [ mix ] . data :
state . latest_randao_mixes [ mix ] . data [ i ] = b xor rr [ i ]
2018-11-29 16:11:05 -06:00
2019-02-14 13:41:04 -08:00
true
2018-12-13 10:00:55 -06:00
2019-04-29 16:48:30 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.0/specs/core/0_beacon-chain.md#eth1-data
2019-03-18 15:42:42 +00:00
func processEth1Data ( state : var BeaconState , blck : BeaconBlock ) =
2019-04-29 16:48:30 +00:00
state . eth1_data_votes . add blck . body . eth1_data
if state . eth1_data_votes . count ( blck . body . eth1_data ) * 2 >
SLOTS_PER_ETH1_VOTING_PERIOD :
state . latest_eth1_data = blck . body . eth1_data
2018-12-13 10:00:55 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#is_slashable_validator
func is_slashable_validator ( validator : Validator , epoch : Epoch ) : bool =
# Check if ``validator`` is slashable.
( not validator . slashed ) and
( validator . activation_epoch < = epoch ) and
( epoch < validator . withdrawable_epoch )
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#proposer-slashings
2018-12-27 17:40:22 -06:00
proc processProposerSlashings (
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
2018-12-13 10:00:55 -06:00
if len ( blck . body . proposer_slashings ) > MAX_PROPOSER_SLASHINGS :
2019-01-25 11:35:22 -06:00
notice " PropSlash: too many! " ,
proposer_slashings = len ( blck . body . proposer_slashings )
2018-12-13 10:00:55 -06:00
return false
for proposer_slashing in blck . body . proposer_slashings :
2019-02-11 15:32:27 +01:00
let proposer = state . validator_registry [ proposer_slashing . proposer_index . int ]
2019-03-08 17:44:31 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify that the epoch is the same
2019-03-16 19:52:37 +00:00
if not ( slot_to_epoch ( proposer_slashing . header_1 . slot ) = =
slot_to_epoch ( proposer_slashing . header_2 . slot ) ) :
notice " PropSlash: epoch mismatch "
2019-03-08 17:44:31 +00:00
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# But the headers are different
2019-03-16 19:52:37 +00:00
if not ( proposer_slashing . header_1 ! = proposer_slashing . header_2 ) :
notice " PropSlash: headers not different "
2019-03-08 17:44:31 +00:00
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Check proposer is slashable
if not is_slashable_validator ( proposer , get_current_epoch ( state ) ) :
2019-03-08 17:44:31 +00:00
notice " PropSlash: slashed proposer "
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Signatures are valid
2018-12-27 17:40:22 -06:00
if skipValidation notin flags :
2019-03-16 19:52:37 +00:00
for i , header in @ [ proposer_slashing . header_1 , proposer_slashing . header_2 ] :
if not bls_verify (
proposer . pubkey ,
2019-05-09 12:27:37 +00:00
signing_root ( header ) . data ,
2019-03-16 19:52:37 +00:00
header . signature ,
get_domain (
2019-04-29 16:48:30 +00:00
state , DOMAIN_BEACON_PROPOSER , slot_to_epoch ( header . slot ) ) ) :
2019-03-16 19:52:37 +00:00
notice " PropSlash: invalid signature " ,
signature_index = i
return false
2018-12-13 10:00:55 -06:00
2019-02-22 09:56:45 +00:00
slashValidator ( state , proposer_slashing . proposer_index . ValidatorIndex )
2018-12-13 10:00:55 -06:00
2019-03-08 17:44:31 +00:00
true
2018-12-03 15:41:24 -06:00
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#verify_slashable_attestation
2019-04-29 16:48:30 +00:00
func verify_slashable_attestation ( state : BeaconState , slashable_attestation : IndexedAttestation ) : bool =
2019-02-05 18:40:27 +00:00
# Verify validity of ``slashable_attestation`` fields.
2019-03-20 14:01:48 -06:00
if anyIt ( slashable_attestation . custody_bitfield . bits , it ! = 0 ) : # [TO BE REMOVED IN PHASE 1]
2018-12-13 10:00:55 -06:00
return false
2019-02-05 18:40:27 +00:00
if len ( slashable_attestation . validator_indices ) = = 0 :
return false
2018-12-13 10:00:55 -06:00
2019-02-05 18:40:27 +00:00
for i in 0 .. < ( len ( slashable_attestation . validator_indices ) - 1 ) :
if slashable_attestation . validator_indices [ i ] > = slashable_attestation . validator_indices [ i + 1 ] :
return false
2018-12-13 10:00:55 -06:00
2019-03-20 14:01:48 -06:00
if not verify_bitfield ( slashable_attestation . custody_bitfield ,
len ( slashable_attestation . validator_indices ) ) :
2019-02-05 18:40:27 +00:00
return false
2019-04-29 16:48:30 +00:00
if len ( slashable_attestation . validator_indices ) > MAX_INDICES_PER_ATTESTATION :
2019-02-05 18:40:27 +00:00
return false
var
custody_bit_0_indices : seq [ uint64 ] = @ [ ]
custody_bit_1_indices : seq [ uint64 ] = @ [ ]
2018-12-13 10:00:55 -06:00
2019-02-07 23:07:15 +00:00
for i , validator_index in slashable_attestation . validator_indices :
2019-03-20 14:01:48 -06:00
if not get_bitfield_bit ( slashable_attestation . custody_bitfield , i ) :
2019-02-07 23:07:15 +00:00
custody_bit_0_indices . add ( validator_index )
else :
custody_bit_1_indices . add ( validator_index )
bls_verify_multiple (
@ [
bls_aggregate_pubkeys ( mapIt ( custody_bit_0_indices , state . validator_registry [ it . int ] . pubkey ) ) ,
bls_aggregate_pubkeys ( mapIt ( custody_bit_1_indices , state . validator_registry [ it . int ] . pubkey ) ) ,
] ,
@ [
hash_tree_root ( AttestationDataAndCustodyBit (
data : slashable_attestation . data , custody_bit : false ) ) ,
hash_tree_root ( AttestationDataAndCustodyBit (
data : slashable_attestation . data , custody_bit : true ) ) ,
] ,
slashable_attestation . aggregate_signature ,
get_domain (
2019-04-29 16:48:30 +00:00
state ,
2019-02-07 23:07:15 +00:00
DOMAIN_ATTESTATION ,
2019-04-29 16:48:30 +00:00
slot_to_epoch ( slashable_attestation . data . slot ) ,
2019-02-07 23:07:15 +00:00
) ,
)
2018-12-13 10:00:55 -06:00
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#attester-slashings
2019-01-30 07:35:42 -08:00
proc processAttesterSlashings ( state : var BeaconState , blck : BeaconBlock ) : bool =
2019-03-20 02:19:46 +00:00
## Process ``AttesterSlashing`` transaction.
## Note that this function mutates ``state``.
2019-02-05 18:40:27 +00:00
if len ( blck . body . attester_slashings ) > MAX_ATTESTER_SLASHINGS :
2019-01-25 11:35:22 -06:00
notice " CaspSlash: too many! "
2018-12-13 10:00:55 -06:00
return false
2019-02-05 18:40:27 +00:00
for attester_slashing in blck . body . attester_slashings :
2018-12-18 22:36:10 -06:00
let
2019-04-29 16:48:30 +00:00
slashable_attestation_1 = attester_slashing . attestation_1
slashable_attestation_2 = attester_slashing . attestation_2
2018-12-18 22:36:10 -06:00
2019-03-20 02:19:46 +00:00
# Check that the attestations are conflicting
2019-02-05 18:40:27 +00:00
if not ( slashable_attestation_1 . data ! = slashable_attestation_2 . data ) :
2019-01-25 11:35:22 -06:00
notice " CaspSlash: invalid data "
2018-12-13 10:00:55 -06:00
return false
if not (
2019-02-05 18:40:27 +00:00
is_double_vote ( slashable_attestation_1 . data , slashable_attestation_2 . data ) or
is_surround_vote ( slashable_attestation_1 . data , slashable_attestation_2 . data ) ) :
2019-01-25 11:35:22 -06:00
notice " CaspSlash: surround or double vote check failed "
2018-12-18 22:36:10 -06:00
return false
2019-02-05 18:40:27 +00:00
if not verify_slashable_attestation ( state , slashable_attestation_1 ) :
2019-01-25 11:35:22 -06:00
notice " CaspSlash: invalid votes 1 "
2018-12-18 22:36:10 -06:00
return false
2019-02-05 18:40:27 +00:00
if not verify_slashable_attestation ( state , slashable_attestation_2 ) :
2019-01-25 11:35:22 -06:00
notice " CaspSlash: invalid votes 2 "
2018-12-13 10:00:55 -06:00
return false
2019-02-05 18:40:27 +00:00
let
2019-03-08 17:44:31 +00:00
indices2 = toSet ( slashable_attestation_2 . validator_indices )
2019-02-05 18:40:27 +00:00
slashable_indices =
2019-03-20 02:19:46 +00:00
slashable_attestation_1 . validator_indices . filterIt (
it in indices2 and not state . validator_registry [ it . int ] . slashed )
2019-02-05 18:40:27 +00:00
if not ( len ( slashable_indices ) > = 1 ) :
notice " CaspSlash: no intersection "
return false
for index in slashable_indices :
2019-03-20 02:19:46 +00:00
slash_validator ( state , index . ValidatorIndex )
2018-12-13 10:00:55 -06:00
2019-02-22 09:56:45 +00:00
true
2018-12-13 10:00:55 -06:00
2019-04-29 16:48:30 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.0/specs/core/0_beacon-chain.md#attestations
2018-12-27 17:40:22 -06:00
proc processAttestations (
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
2018-12-13 10:00:55 -06:00
## Each block includes a number of attestations that the proposer chose. Each
## attestation represents an update to a specific shard and is signed by a
## committee of validators.
## Here we make sanity checks for each attestation and it to the state - most
## updates will happen at the epoch boundary where state updates happen in
## bulk.
if blck . body . attestations . len > MAX_ATTESTATIONS :
2019-01-25 11:35:22 -06:00
notice " Attestation: too many! " , attestations = blck . body . attestations . len
2018-12-13 10:00:55 -06:00
return false
2018-12-27 17:40:22 -06:00
if not blck . body . attestations . allIt ( checkAttestation ( state , it , flags ) ) :
2018-12-13 10:00:55 -06:00
return false
# All checks passed - update state
2019-03-22 18:33:12 +00:00
# Apply the attestations
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
var committee_count_cache = initTable [ Epoch , uint64 ] ( )
2019-03-22 18:33:12 +00:00
for attestation in blck . body . attestations :
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
let
epoch = attestation . data . target_epoch
committee_count = if epoch in committee_count_cache :
committee_count_cache [ epoch ]
else :
get_epoch_committee_count ( state , epoch )
committee_count_cache [ epoch ] = committee_count
let attestation_slot =
get_attestation_slot ( state , attestation , committee_count )
2019-03-22 18:33:12 +00:00
let pending_attestation = PendingAttestation (
data : attestation . data ,
aggregation_bitfield : attestation . aggregation_bitfield ,
2019-04-29 16:48:30 +00:00
inclusion_slot : state . slot ,
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
inclusion_delay : state . slot - attestation_slot ,
2019-04-29 16:48:30 +00:00
proposer_index : get_beacon_proposer_index ( state ) ,
2018-12-13 10:00:55 -06:00
)
2019-03-22 18:33:12 +00:00
if slot_to_epoch ( attestation . data . slot ) = = get_current_epoch ( state ) :
state . current_epoch_attestations . add ( pending_attestation )
else :
state . previous_epoch_attestations . add ( pending_attestation )
2018-12-13 10:00:55 -06:00
true
2019-03-24 18:21:23 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.1/specs/core/0_beacon-chain.md#deposits
proc processDeposits ( state : var BeaconState , blck : BeaconBlock ) : bool =
if not ( len ( blck . body . deposits ) < = MAX_DEPOSITS ) :
notice " processDeposits: too many deposits "
return false
for deposit in blck . body . deposits :
if not process_deposit ( state , deposit ) :
notice " processDeposits: deposit invalid "
return false
2019-02-22 09:53:37 -08:00
true
2019-01-16 03:39:16 -08:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#voluntary-exits
proc processVoluntaryExits (
2018-12-27 17:40:22 -06:00
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Process ``VoluntaryExit`` transaction.
2019-02-19 23:07:56 +00:00
if len ( blck . body . voluntary_exits ) > MAX_VOLUNTARY_EXITS :
2019-01-25 11:35:22 -06:00
notice " Exit: too many! "
2018-12-13 10:00:55 -06:00
return false
2019-02-19 23:07:56 +00:00
for exit in blck . body . voluntary_exits :
2019-02-11 15:32:27 +01:00
let validator = state . validator_registry [ exit . validator_index . int ]
2018-12-13 10:00:55 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify the validator is active
if not is_active_validator ( validator , get_current_epoch ( state ) ) :
notice " Exit: validator not active "
return false
2019-03-24 18:21:23 +00:00
# Verify the validator has not yet exited
if not ( validator . exit_epoch = = FAR_FUTURE_EPOCH ) :
notice " Exit: validator has exited "
2019-02-22 09:53:37 -08:00
return false
2019-03-24 18:21:23 +00:00
## Exits must specify an epoch when they become valid; they are not valid
## before then
2019-02-22 09:53:37 -08:00
if not ( get_current_epoch ( state ) > = exit . epoch ) :
2019-03-24 18:21:23 +00:00
notice " Exit: exit epoch not passed "
2019-02-22 09:53:37 -08:00
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify the validator has been active long enough
2019-03-24 18:21:23 +00:00
if not ( get_current_epoch ( state ) - validator . activation_epoch > =
PERSISTENT_COMMITTEE_PERIOD ) :
notice " Exit: not in validator set long enough "
return false
# Verify signature
2018-12-27 17:40:22 -06:00
if skipValidation notin flags :
if not bls_verify (
2019-05-09 12:27:37 +00:00
validator . pubkey , signing_root ( exit ) . data , exit . signature ,
2019-04-29 16:48:30 +00:00
get_domain ( state , DOMAIN_VOLUNTARY_EXIT , exit . epoch ) ) :
2019-01-25 11:35:22 -06:00
notice " Exit: invalid signature "
2018-12-27 17:40:22 -06:00
return false
2018-12-13 10:00:55 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Initiate exit
2019-02-22 09:53:37 -08:00
initiate_validator_exit ( state , exit . validator_index . ValidatorIndex )
true
2019-03-24 18:21:23 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.1/specs/core/0_beacon-chain.md#validator-registry-and-shuffling-seed-data
2019-03-20 02:19:46 +00:00
func update_registry_and_shuffling_data ( state : var BeaconState ) =
# First set previous shuffling data to current shuffling data
state . previous_shuffling_epoch = state . current_shuffling_epoch
state . previous_shuffling_start_shard = state . current_shuffling_start_shard
state . previous_shuffling_seed = state . current_shuffling_seed
let
current_epoch = get_current_epoch ( state )
next_epoch = current_epoch + 1
# Check if we should update, and if so, update
if should_update_validator_registry ( state ) :
update_validator_registry ( state )
# If we update the registry, update the shuffling data and shards as well
state . current_shuffling_epoch = next_epoch
state . current_shuffling_start_shard = (
state . current_shuffling_start_shard +
2019-05-09 12:27:37 +00:00
get_epoch_committee_count ( state , current_epoch ) mod SHARD_COUNT
2019-03-24 18:21:23 +00:00
) mod SHARD_COUNT
2019-03-20 02:19:46 +00:00
state . current_shuffling_seed =
generate_seed ( state , state . current_shuffling_epoch )
else :
## If processing at least one crosslink keeps failing, then reshuffle every
## power of two, but don't update the current_shuffling_start_shard
let epochs_since_last_registry_update =
current_epoch - state . validator_registry_update_epoch
if epochs_since_last_registry_update > 1 'u64 and
is_power_of_2 ( epochs_since_last_registry_update ) :
state . current_shuffling_epoch = next_epoch
state . current_shuffling_seed =
generate_seed ( state , state . current_shuffling_epoch )
2019-04-29 16:48:30 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.0/specs/core/0_beacon-chain.md#transfers
2019-02-22 09:53:37 -08:00
proc processTransfers ( state : var BeaconState , blck : BeaconBlock ,
flags : UpdateFlags ) : bool =
if not ( len ( blck . body . transfers ) < = MAX_TRANSFERS ) :
notice " Transfer: too many transfers "
return false
for transfer in blck . body . transfers :
2019-04-29 16:48:30 +00:00
let sender_balance = state . balances [ transfer . sender . int ]
2018-12-13 10:00:55 -06:00
2019-04-29 16:48:30 +00:00
## Verify the amount and fee are not individually too big (for anti-overflow
2019-03-20 02:19:46 +00:00
## purposes)
if not ( sender_balance > = max ( transfer . amount , transfer . fee ) ) :
notice " Transfer: sender balance too low for transfer amount or fee "
2018-12-13 10:00:55 -06:00
return false
2019-03-20 02:19:46 +00:00
# A transfer is valid in only one slot
2019-02-22 09:53:37 -08:00
if not ( state . slot = = transfer . slot ) :
notice " Transfer: slot mismatch "
return false
2019-04-29 16:48:30 +00:00
## Sender must be not yet eligible for activation, withdrawn, or transfer
## balance over MAX_EFFECTIVE_BALANCE
if not (
state . validator_registry [ transfer . sender . int ] . activation_epoch = =
FAR_FUTURE_EPOCH or
get_current_epoch ( state ) > =
2019-03-11 15:33:24 +00:00
state . validator_registry [
2019-03-18 15:42:42 +00:00
transfer . sender . int ] . withdrawable_epoch or
2019-04-29 16:48:30 +00:00
transfer . amount + transfer . fee + MAX_EFFECTIVE_BALANCE < =
state . balances [ transfer . sender . int ] ) :
notice " Transfer: only withdrawn or not-activated accounts with sufficient balance can transfer "
2019-02-22 09:53:37 -08:00
return false
2019-03-20 02:19:46 +00:00
# Verify that the pubkey is valid
2019-03-18 15:42:42 +00:00
let wc = state . validator_registry [ transfer . sender . int ] .
2019-02-22 09:53:37 -08:00
withdrawal_credentials
if not ( wc . data [ 0 ] = = BLS_WITHDRAWAL_PREFIX_BYTE and
wc . data [ 1 .. ^ 1 ] = = eth2hash ( transfer . pubkey . getBytes ) . data [ 1 .. ^ 1 ] ) :
notice " Transfer: incorrect withdrawal credentials "
return false
2019-03-20 02:19:46 +00:00
# Verify that the signature is valid
2019-02-22 09:53:37 -08:00
if skipValidation notin flags :
2019-02-22 11:17:53 -08:00
if not bls_verify (
2019-05-09 12:27:37 +00:00
transfer . pubkey , signing_root ( transfer ) . data , transfer . signature ,
2019-04-29 16:48:30 +00:00
get_domain ( state , DOMAIN_TRANSFER ) ) :
2019-02-22 09:53:37 -08:00
notice " Transfer: incorrect signature "
return false
2019-04-29 16:48:30 +00:00
# Process the transfer
decrease_balance (
state , transfer . sender . ValidatorIndex , transfer . amount + transfer . fee )
increase_balance (
state , transfer . recipient . ValidatorIndex , transfer . amount )
increase_balance ( state , get_beacon_proposer_index ( state ) , transfer . fee )
# Verify balances are not dust
if not (
0 'u64 < state . balances [ transfer . sender . int ] and
state . balances [ transfer . sender . int ] < MIN_DEPOSIT_AMOUNT ) :
notice " Transfer: sender balance too low for transfer amount or fee "
return false
if not (
0 'u64 < state . balances [ transfer . recipient . int ] and
state . balances [ transfer . recipient . int ] < MIN_DEPOSIT_AMOUNT ) :
notice " Transfer: sender balance too low for transfer amount or fee "
return false
2019-02-22 09:53:37 -08:00
true
2018-12-13 10:00:55 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#per-slot-processing
2019-03-26 19:32:35 -06:00
func advance_slot ( state : var BeaconState ) =
2018-12-18 22:36:10 -06:00
## Time on the beacon chain moves in slots. Every time we make it to a new
2018-12-27 14:14:37 -06:00
## slot, a proposer creates a block to represent the state of the beacon
2018-12-18 22:36:10 -06:00
## chain at that time. In case the proposer is missing, it may happen that
## the no block is produced during the slot.
2019-03-21 15:31:36 +00:00
2018-12-18 22:36:10 -06:00
state . slot + = 1
2019-02-22 09:56:45 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#state-caching
2019-03-16 19:52:37 +00:00
func cacheState ( state : var BeaconState ) =
2019-03-25 10:46:31 -06:00
let previous_slot_state_root = hash_tree_root ( state )
2019-03-16 19:52:37 +00:00
# store the previous slot's post state transition root
state . latest_state_roots [ state . slot mod SLOTS_PER_HISTORICAL_ROOT ] =
previous_slot_state_root
# cache state root in stored latest_block_header if empty
if state . latest_block_header . state_root = = ZERO_HASH :
state . latest_block_header . state_root = previous_slot_state_root
# store latest known block for previous slot
state . latest_block_roots [ state . slot mod SLOTS_PER_HISTORICAL_ROOT ] =
2019-05-09 12:27:37 +00:00
signing_root ( state . latest_block_header )
2018-12-18 22:36:10 -06:00
proc processBlock (
2018-12-21 16:37:46 -06:00
state : var BeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
2018-12-13 10:00:55 -06:00
## When there's a new block, we need to verify that the block is sane and
## update the state accordingly
# TODO when there's a failure, we should reset the state!
# TODO probably better to do all verification first, then apply state changes
2019-03-22 18:33:12 +00:00
if not processBlockHeader ( state , blck , flags ) :
notice " Block header not valid " , slot = humaneSlotNum ( state . slot )
return false
2018-12-18 22:36:10 -06:00
2018-12-21 16:37:46 -06:00
if not processRandao ( state , blck , flags ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Randao failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
2019-03-18 15:42:42 +00:00
processEth1Data ( state , blck )
2018-12-13 10:00:55 -06:00
2018-12-27 17:40:22 -06:00
if not processProposerSlashings ( state , blck , flags ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Proposer slashing failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
2019-01-30 07:35:42 -08:00
if not processAttesterSlashings ( state , blck ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Attester slashing failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
2018-12-03 15:41:24 -06:00
2018-12-27 17:40:22 -06:00
if not processAttestations ( state , blck , flags ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Attestation processing failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
if not processDeposits ( state , blck ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Deposit processing failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
if not processVoluntaryExits ( state , blck , flags ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Exit processing failure " , slot = humaneSlotNum ( state . slot )
2018-12-13 10:00:55 -06:00
return false
2019-02-22 09:53:37 -08:00
if not processTransfers ( state , blck , flags ) :
2019-04-29 18:10:01 +02:00
debug " [Block processing] Transfer processing failure " , slot = humaneSlotNum ( state . slot )
2019-02-22 09:53:37 -08:00
return false
2018-12-13 10:00:55 -06:00
2019-02-22 09:53:37 -08:00
true
2018-12-13 10:00:55 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#helper-functions-1
func get_total_active_balance ( state : BeaconState ) : Gwei =
2019-03-21 15:31:36 +00:00
return get_total_balance (
state ,
2019-05-09 12:27:37 +00:00
get_active_validator_indices ( state , get_current_epoch ( state ) ) )
2019-03-21 15:31:36 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
func get_matching_source_attestations ( state : BeaconState , epoch : Epoch ) :
seq [ PendingAttestation ] =
doAssert epoch in @ [ get_current_epoch ( state ) , get_previous_epoch ( state ) ]
if epoch = = get_current_epoch ( state ) :
state . current_epoch_attestations
else :
state . previous_epoch_attestations
func get_matching_target_attestations ( state : BeaconState , epoch : Epoch ) :
seq [ PendingAttestation ] =
filterIt (
get_matching_source_attestations ( state , epoch ) ,
it . data . target_root = = get_block_root ( state , epoch )
)
func get_matching_head_attestations ( state : BeaconState , epoch : Epoch ) :
seq [ PendingAttestation ] =
filterIt (
get_matching_source_attestations ( state , epoch ) ,
it . data . beacon_block_root = =
get_block_root_at_slot ( state , get_attestation_slot ( state , it ) )
)
2019-03-20 02:19:46 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#get_attesting_indices
func get_attesting_indices ( state : BeaconState ,
attestation_data : AttestationData ,
bitfield : BitField ) : HashSet [ ValidatorIndex ] =
## Return the sorted attesting indices corresponding to ``attestation_data``
## and ``bitfield``.
## The spec goes through a lot of hoops to sort things, and sometimes
## constructs sets from the results here. The basic idea is to always
## just do the right thing and keep it in a HashSet.
result = initSet [ ValidatorIndex ] ( )
let committee =
get_crosslink_committee ( state , attestation_data . target_epoch ,
attestation_data . shard )
doAssert verify_bitfield ( bitfield , len ( committee ) )
for i , index in committee :
if get_bitfield_bit ( bitfield , i ) :
result . incl index
# TODO remove this 0.5ish one when callers disappear
2019-03-20 02:19:46 +00:00
func get_attesting_indices (
state : BeaconState ,
2019-03-28 13:50:14 -07:00
attestations : openArray [ PendingAttestation ] ) : HashSet [ ValidatorIndex ] =
2019-03-20 02:19:46 +00:00
# Union of attesters that participated in some attestations
2019-03-28 14:27:50 -07:00
result = initSet [ ValidatorIndex ] ( )
for attestation in attestations :
for validator_index in get_attestation_participants (
state , attestation . data , attestation . aggregation_bitfield ) :
result . incl validator_index
2019-03-20 02:19:46 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# TODO this cached version corresponds to the 0.5ish get_attesting_indices
2019-03-27 14:10:15 -07:00
func get_attesting_indices_cached (
state : BeaconState ,
2019-04-05 14:18:13 +00:00
attestations : openArray [ PendingAttestation ] , cache : var StateCache ) :
HashSet [ ValidatorIndex ] =
2019-03-27 14:10:15 -07:00
# Union of attesters that participated in some attestations
2019-03-28 14:27:50 -07:00
result = initSet [ ValidatorIndex ] ( )
for attestation in attestations :
for validator_index in get_attestation_participants_cached (
state , attestation . data , attestation . aggregation_bitfield ,
2019-04-05 14:18:13 +00:00
cache ) :
2019-03-28 14:27:50 -07:00
result . incl validator_index
2019-03-27 14:10:15 -07:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
func get_unslashed_attesting_indices (
state : BeaconState , attestations : seq [ PendingAttestation ] ) :
HashSet [ ValidatorIndex ] =
result = initSet [ ValidatorIndex ] ( )
for a in attestations :
result = result . union ( get_attesting_indices (
state , a . data , a . aggregation_bitfield ) )
for index in result :
if state . validator_registry [ index ] . slashed :
result . excl index
# TODO check for blob/v0.5.0 removal
func get_previous_total_balance ( state : BeaconState ) : Gwei =
get_total_balance (
state ,
get_active_validator_indices ( state , get_previous_epoch ( state ) ) )
2019-03-20 02:19:46 +00:00
func get_attesting_balance ( state : BeaconState ,
attestations : seq [ PendingAttestation ] ) : Gwei =
get_total_balance ( state , get_attesting_indices ( state , attestations ) )
2019-03-27 14:10:15 -07:00
func get_attesting_balance_cached (
2019-04-05 14:18:13 +00:00
state : BeaconState , attestations : seq [ PendingAttestation ] ,
cache : var StateCache ) : Gwei =
2019-03-27 16:38:34 -07:00
get_total_balance ( state , get_attesting_indices_cached (
2019-04-05 14:18:13 +00:00
state , attestations , cache ) )
2019-03-27 14:10:15 -07:00
2019-03-21 15:31:36 +00:00
func get_current_epoch_boundary_attestations ( state : BeaconState ) :
seq [ PendingAttestation ] =
filterIt (
state . current_epoch_attestations ,
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
it . data . target_root = = get_block_root_at_slot (
2019-03-21 15:31:36 +00:00
state , get_epoch_start_slot ( get_current_epoch ( state ) ) ) )
2019-03-20 02:19:46 +00:00
func get_previous_epoch_boundary_attestations ( state : BeaconState ) :
seq [ PendingAttestation ] =
filterIt (
state . previous_epoch_attestations ,
it . data . target_root = =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_block_root_at_slot ( state , get_epoch_start_slot ( get_previous_epoch ( state ) ) ) )
2019-03-20 02:19:46 +00:00
func get_previous_epoch_matching_head_attestations ( state : BeaconState ) :
seq [ PendingAttestation ] =
filterIt (
state . previous_epoch_attestations ,
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
it . data . beacon_block_root = = get_block_root_at_slot ( state , it . data . slot ) )
2019-03-20 02:19:46 +00:00
2019-03-21 15:31:36 +00:00
# Not exactly in spec, but for get_winning_root_and_participants
func lowerThan ( candidate , current : Eth2Digest ) : bool =
# return true iff candidate is "lower" than current, per spec rule:
# "ties broken in favor of lexicographically higher hash
for i , v in current . data :
if v > candidate . data [ i ] : return true
false
2019-03-27 16:38:34 -07:00
func get_winning_root_and_participants (
2019-04-05 14:18:13 +00:00
state : BeaconState , shard : Shard , cache : var StateCache ) :
2019-03-28 14:27:50 -07:00
tuple [ a : Eth2Digest , b : HashSet [ ValidatorIndex ] ] =
2019-03-21 15:31:36 +00:00
let
all_attestations =
concat ( state . current_epoch_attestations ,
state . previous_epoch_attestations )
valid_attestations =
filterIt (
all_attestations ,
it . data . previous_crosslink = = state . latest_crosslinks [ shard ] )
all_roots = mapIt ( valid_attestations , it . data . crosslink_data_root )
# handle when no attestations for shard available
if len ( all_roots ) = = 0 :
2019-03-28 14:27:50 -07:00
return ( ZERO_HASH , initSet [ ValidatorIndex ] ( ) )
2019-03-21 15:31:36 +00:00
2019-03-27 14:10:15 -07:00
# 0.5.1 spec has less-than-ideal get_attestations_for nested function.
var attestations_for = initTable [ Eth2Digest , seq [ PendingAttestation ] ] ( )
for valid_attestation in valid_attestations :
if valid_attestation . data . crosslink_data_root in attestations_for :
attestations_for [ valid_attestation . data . crosslink_data_root ] . add (
valid_attestation )
else :
attestations_for [ valid_attestation . data . crosslink_data_root ] =
@ [ valid_attestation ]
2019-03-21 15:31:36 +00:00
## Winning crosslink root is the root with the most votes for it, ties broken
## in favor of lexicographically higher hash
var
winning_root : Eth2Digest
winning_root_balance = 0 'u64
for r in all_roots :
2019-03-27 16:38:34 -07:00
let root_balance = get_attesting_balance_cached (
2019-04-05 14:18:13 +00:00
state , attestations_for . getOrDefault ( r ) , cache )
2019-03-21 15:31:36 +00:00
if ( root_balance > winning_root_balance or
( root_balance = = winning_root_balance and
lowerThan ( winning_root , r ) ) ) :
winning_root = r
winning_root_balance = root_balance
( winning_root ,
2019-03-27 18:20:11 -07:00
get_attesting_indices_cached (
state ,
2019-04-05 14:18:13 +00:00
attestations_for . getOrDefault ( winning_root ) , cache ) )
2019-03-21 15:31:36 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#justification
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
func process_justification_and_finalization ( state : var BeaconState ) =
2019-03-21 15:31:36 +00:00
var
new_justified_epoch = state . current_justified_epoch
new_finalized_epoch = state . finalized_epoch
## Rotate the justification bitfield up one epoch to make room for the
## current epoch
state . justification_bitfield = state . justification_bitfield shl 1
# If the previous epoch gets justified, fill the second last bit
let previous_boundary_attesting_balance =
get_attesting_balance (
state , get_previous_epoch_boundary_attestations ( state ) )
if previous_boundary_attesting_balance * 3 'u64 > =
get_previous_total_balance ( state ) * 2 'u64 :
new_justified_epoch = get_current_epoch ( state ) - 1
state . justification_bitfield = state . justification_bitfield or 2
# If the current epoch gets justified, fill the last bit
let current_boundary_attesting_balance =
get_attesting_balance (
state , get_current_epoch_boundary_attestations ( state ) )
if current_boundary_attesting_balance * 3 'u64 > =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_total_active_balance ( state ) * 2 'u64 :
2019-03-21 15:31:36 +00:00
new_justified_epoch = get_current_epoch ( state )
state . justification_bitfield = state . justification_bitfield or 1
# Process finalizations
let
bitfield = state . justification_bitfield
current_epoch = get_current_epoch ( state )
## The 2nd/3rd/4th most recent epochs are all justified, the 2nd using the
## 4th as source
if ( bitfield shr 1 ) mod 8 = = 0b111 and
state . previous_justified_epoch = = current_epoch - 3 :
new_finalized_epoch = state . previous_justified_epoch
## The 2nd/3rd most recent epochs are both justified, the 2nd using the 3rd
## as source
if ( bitfield shr 1 ) mod 4 = = 0b11 and
state . previous_justified_epoch = = current_epoch - 2 :
new_finalized_epoch = state . previous_justified_epoch
## The 1st/2nd/3rd most recent epochs are all justified, the 1st using the
## 3rd as source
if ( bitfield shr 0 ) mod 8 = = 0b111 and
state . current_justified_epoch = = current_epoch - 2 :
new_finalized_epoch = state . current_justified_epoch
## The 1st/2nd most recent epochs are both justified, the 1st using the 2nd
## as source
if ( bitfield shr 0 ) mod 4 = = 0b11 and
state . current_justified_epoch = = current_epoch - 1 :
new_finalized_epoch = state . current_justified_epoch
# Update state jusification/finality fields
state . previous_justified_epoch = state . current_justified_epoch
state . previous_justified_root = state . current_justified_root
if new_justified_epoch ! = state . current_justified_epoch :
state . current_justified_epoch = new_justified_epoch
state . current_justified_root =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_block_root_at_slot ( state , get_epoch_start_slot ( new_justified_epoch ) )
2019-03-21 15:31:36 +00:00
if new_finalized_epoch ! = state . finalized_epoch :
state . finalized_epoch = new_finalized_epoch
state . finalized_root =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_block_root_at_slot ( state , get_epoch_start_slot ( new_finalized_epoch ) )
2019-03-21 15:31:36 +00:00
2019-03-22 18:33:12 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.1/specs/core/0_beacon-chain.md#crosslinks
2019-03-27 14:32:35 -07:00
func process_crosslinks (
2019-04-05 14:18:13 +00:00
state : var BeaconState , per_epoch_cache : var StateCache ) =
2019-03-21 15:31:36 +00:00
let
current_epoch = get_current_epoch ( state )
previous_epoch = current_epoch - 1
next_epoch = current_epoch + 1
2019-03-27 14:10:15 -07:00
2019-03-22 18:33:12 +00:00
## TODO is it actually correct to be setting state.latest_crosslinks[shard]
## to something pre-GENESIS_EPOCH, ever? I guess the intent is if there are
## a quorum of participants for get_epoch_start_slot(previous_epoch), when
## state.slot == GENESIS_SLOT, then there will be participants for a quorum
## in the current-epoch (i.e. genesis epoch) version of that shard?
#for slot in get_epoch_start_slot(previous_epoch).uint64 ..<
for slot in max (
GENESIS_SLOT . uint64 , get_epoch_start_slot ( previous_epoch ) . uint64 ) .. <
2019-03-21 15:31:36 +00:00
get_epoch_start_slot ( next_epoch ) . uint64 :
2019-03-27 14:32:35 -07:00
for cas in get_crosslink_committees_at_slot_cached (
2019-04-05 14:18:13 +00:00
state , slot , false , per_epoch_cache ) :
2019-03-21 15:31:36 +00:00
let
( crosslink_committee , shard ) = cas
2019-03-27 18:20:11 -07:00
# In general, it'll loop over the same shards twice, and
# get_winning_root_and_participants is defined to return
# the same results from the previous epoch as current.
( winning_root , participants ) =
2019-04-05 14:18:13 +00:00
if shard notin per_epoch_cache . winning_root_participants_cache :
get_winning_root_and_participants ( state , shard , per_epoch_cache )
2019-03-27 18:20:11 -07:00
else :
2019-04-05 14:18:13 +00:00
( ZERO_HASH , per_epoch_cache . winning_root_participants_cache [ shard ] )
2019-03-21 15:31:36 +00:00
participating_balance = get_total_balance ( state , participants )
total_balance = get_total_balance ( state , crosslink_committee )
2019-03-22 18:33:12 +00:00
2019-04-05 14:18:13 +00:00
per_epoch_cache . winning_root_participants_cache [ shard ] = participants
2019-03-27 18:20:11 -07:00
2019-03-21 15:31:36 +00:00
if 3 'u64 * participating_balance > = 2 'u64 * total_balance :
2019-03-22 18:33:12 +00:00
# Check not from spec; seems kludgy
doAssert slot > = GENESIS_SLOT
2019-03-21 15:31:36 +00:00
state . latest_crosslinks [ shard ] = Crosslink (
epoch : slot_to_epoch ( slot ) ,
crosslink_data_root : winning_root
)
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#rewards-and-penalties
func get_base_reward ( state : BeaconState , index : ValidatorIndex ) : Gwei =
let adjusted_quotient =
integer_squareroot ( get_total_active_balance ( state ) ) div BASE_REWARD_QUOTIENT
if adjusted_quotient = = 0 :
2019-03-20 02:19:46 +00:00
return 0
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
state . validator_registry [ index ] . effective_balance div adjusted_quotient div
BASE_REWARDS_PER_EPOCH
2019-03-20 02:19:46 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#rewards-and-penalties
func get_attestation_deltas ( state : BeaconState ) :
2019-03-20 02:19:46 +00:00
tuple [ a : seq [ Gwei ] , b : seq [ Gwei ] ] =
let
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
previous_epoch = get_previous_epoch ( state )
total_balance = get_total_active_balance ( state )
var
rewards = repeat ( 0 'u64 , len ( state . validator_registry ) )
penalties = repeat ( 0 'u64 , len ( state . validator_registry ) )
eligible_validator_indices : seq [ ValidatorIndex ] = @ [ ]
2019-03-20 02:19:46 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
for index , v in state . validator_registry :
if is_active_validator ( v , previous_epoch ) or
( v . slashed and previous_epoch + 1 < v . withdrawable_epoch ) :
eligible_validator_indices . add index . ValidatorIndex
2019-03-27 14:32:35 -07:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Micro-incentives for matching FFG source, FFG target, and head
2019-03-20 02:19:46 +00:00
let
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
matching_source_attestations =
get_matching_source_attestations ( state , previous_epoch )
matching_target_attestations =
get_matching_target_attestations ( state , previous_epoch )
2019-03-20 02:19:46 +00:00
matching_head_attestations =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
get_matching_head_attestations ( state , previous_epoch )
for attestations in
@ [ matching_source_attestations , matching_target_attestations ,
matching_head_attestations ] :
let
unslashed_attesting_indices =
get_unslashed_attesting_indices ( state , attestations )
attesting_balance = get_attesting_balance ( state , attestations )
for index in eligible_validator_indices :
if index in unslashed_attesting_indices :
rewards [ index ] + =
get_base_reward ( state , index ) * attesting_balance div total_balance
else :
penalties [ index ] + = get_base_reward ( state , index )
if matching_source_attestations . len = = 0 :
return ( rewards , penalties )
# Proposer and inclusion delay micro-rewards
for index in get_unslashed_attesting_indices ( state , matching_source_attestations ) :
doAssert matching_source_attestations . len > 0
var attestation = matching_source_attestations [ 0 ]
for a in matching_source_attestations :
if index notin get_attesting_indices ( state , a . data , a . aggregation_bitfield ) :
continue
if a . inclusion_delay < attestation . inclusion_delay :
attestation = a
rewards [ attestation . proposer_index ] + = get_base_reward ( state , index ) div
PROPOSER_REWARD_QUOTIENT
rewards [ index ] + =
get_base_reward ( state , index ) * MIN_ATTESTATION_INCLUSION_DELAY div
attestation . inclusion_delay
# Inactivity penalty
let finality_delay = previous_epoch - state . finalized_epoch
if finality_delay > MIN_EPOCHS_TO_INACTIVITY_PENALTY :
let matching_target_attesting_indices =
get_unslashed_attesting_indices ( state , matching_target_attestations )
for index in eligible_validator_indices :
penalties [ index ] + =
BASE_REWARDS_PER_EPOCH . uint64 * get_base_reward ( state , index )
if index notin matching_target_attesting_indices :
penalties [ index ] + =
state . validator_registry [ index ] . effective_balance *
finality_delay div INACTIVITY_PENALTY_QUOTIENT
( rewards , penalties )
# blob/0.5.1
2019-04-05 14:18:13 +00:00
func get_crosslink_deltas ( state : BeaconState , cache : var StateCache ) :
2019-03-21 15:31:36 +00:00
tuple [ a : seq [ Gwei ] , b : seq [ Gwei ] ] =
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
var
rewards = repeat ( 0 'u64 , len ( state . validator_registry ) )
penalties = repeat ( 0 'u64 , len ( state . validator_registry ) )
2019-03-21 15:31:36 +00:00
let
previous_epoch_start_slot =
get_epoch_start_slot ( get_previous_epoch ( state ) )
current_epoch_start_slot =
get_epoch_start_slot ( get_current_epoch ( state ) )
for slot in previous_epoch_start_slot . uint64 .. <
current_epoch_start_slot . uint64 :
2019-04-05 14:18:13 +00:00
for cas in get_crosslink_committees_at_slot_cached ( state , slot , false , cache ) :
2019-03-21 15:31:36 +00:00
let
( crosslink_committee , shard ) = cas
2019-03-27 18:39:00 -07:00
( winning_root , participants ) =
2019-04-05 14:18:13 +00:00
if shard notin cache . winning_root_participants_cache :
get_winning_root_and_participants ( state , shard , cache )
2019-03-27 18:39:00 -07:00
else :
2019-04-05 14:18:13 +00:00
( ZERO_HASH , cache . winning_root_participants_cache [ shard ] )
2019-03-21 15:31:36 +00:00
participating_balance = get_total_balance ( state , participants )
total_balance = get_total_balance ( state , crosslink_committee )
for index in crosslink_committee :
2019-03-28 14:27:50 -07:00
if index in participants :
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
rewards [ index ] + =
2019-03-21 15:31:36 +00:00
get_base_reward ( state , index ) * participating_balance div
total_balance
else :
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
penalties [ index ] + = get_base_reward ( state , index )
2019-03-21 15:31:36 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
( rewards , penalties )
2019-03-21 15:31:36 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#rewards-and-penalties
func process_rewards_and_penalties (
state : var BeaconState , cache : var StateCache ) =
2019-03-21 15:31:36 +00:00
let
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
( rewards1 , penalties1 ) = get_attestation_deltas ( state )
( rewards2 , penalties2 ) = get_crosslink_deltas ( state , cache )
2019-03-21 15:31:36 +00:00
for i in 0 .. < len ( state . validator_registry ) :
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
increase_balance ( state , i . ValidatorIndex , rewards1 [ i ] + rewards2 [ i ] )
decrease_balance ( state , i . ValidatorIndex , penalties1 [ i ] + penalties2 [ i ] )
2019-03-21 15:31:36 +00:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#slashings
2019-02-26 09:54:08 +00:00
func process_slashings ( state : var BeaconState ) =
let
current_epoch = get_current_epoch ( state )
active_validator_indices = get_active_validator_indices (
2019-05-09 12:27:37 +00:00
state , current_epoch )
2019-02-26 09:54:08 +00:00
total_balance = get_total_balance ( state , active_validator_indices )
2019-03-20 02:19:46 +00:00
# Compute `total_penalties`
total_at_start = state . latest_slashed_balances [
( current_epoch + 1 ) mod LATEST_SLASHED_EXIT_LENGTH ]
total_at_end =
state . latest_slashed_balances [ current_epoch mod
LATEST_SLASHED_EXIT_LENGTH ]
total_penalties = total_at_end - total_at_start
2019-02-26 09:54:08 +00:00
for index , validator in state . validator_registry :
2019-03-08 17:44:31 +00:00
if validator . slashed and current_epoch = = validator . withdrawable_epoch -
LATEST_SLASHED_EXIT_LENGTH div 2 :
2019-02-26 09:54:08 +00:00
let
penalty = max (
get_effective_balance ( state , index . ValidatorIndex ) *
min ( total_penalties * 3 , total_balance ) div total_balance ,
get_effective_balance ( state , index . ValidatorIndex ) div
MIN_PENALTY_QUOTIENT )
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
decrease_balance ( state , index . ValidatorIndex , penalty )
2019-02-26 09:54:08 +00:00
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#final-updates
func finish_epoch_update ( state : var BeaconState ) =
let
current_epoch = get_current_epoch ( state )
next_epoch = current_epoch + 1
# Set active index root
let index_root_position =
( next_epoch + ACTIVATION_EXIT_DELAY ) mod LATEST_ACTIVE_INDEX_ROOTS_LENGTH
state . latest_active_index_roots [ index_root_position ] =
2019-03-25 10:46:31 -06:00
hash_tree_root ( get_active_validator_indices (
2019-05-09 12:27:37 +00:00
state , next_epoch + ACTIVATION_EXIT_DELAY )
2019-03-20 02:19:46 +00:00
)
# Set total slashed balances
state . latest_slashed_balances [ next_epoch mod LATEST_SLASHED_EXIT_LENGTH ] = (
state . latest_slashed_balances [ current_epoch mod LATEST_SLASHED_EXIT_LENGTH ]
)
# Set randao mix
state . latest_randao_mixes [ next_epoch mod LATEST_RANDAO_MIXES_LENGTH ] =
get_randao_mix ( state , current_epoch )
# Set historical root accumulator
if next_epoch mod ( SLOTS_PER_HISTORICAL_ROOT div SLOTS_PER_EPOCH ) . uint64 = = 0 :
let historical_batch = HistoricalBatch (
block_roots : state . latest_block_roots ,
state_roots : state . latest_state_roots ,
)
2019-03-25 10:46:31 -06:00
state . historical_roots . add ( hash_tree_root ( historical_batch ) )
2019-03-20 02:19:46 +00:00
# Rotate current/previous epoch attestations
state . previous_epoch_attestations = state . current_epoch_attestations
state . current_epoch_attestations = @ [ ]
2019-03-21 15:31:36 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#per-epoch-processing
2019-04-05 14:18:13 +00:00
func get_empty_per_epoch_cache ( ) : StateCache =
result . crosslink_committee_cache =
initTable [ tuple [ a : uint64 , b : bool ] , seq [ CrosslinkCommittee ] ] ( )
result . winning_root_participants_cache =
initTable [ Shard , HashSet [ ValidatorIndex ] ] ( )
2018-12-14 10:12:39 -06:00
func processEpoch ( state : var BeaconState ) =
2019-03-21 15:31:36 +00:00
if not ( state . slot > GENESIS_SLOT and
( state . slot + 1 ) mod SLOTS_PER_EPOCH = = 0 ) :
2018-12-14 10:12:39 -06:00
return
2018-12-03 15:41:24 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#justification-and-finalization
process_justification_and_finalization ( state )
2019-02-26 09:54:08 +00:00
2019-04-05 14:18:13 +00:00
var per_epoch_cache = get_empty_per_epoch_cache ( )
2019-05-09 12:27:37 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.1/specs/core/0_beacon-chain.md#crosslinks
2019-04-05 14:18:13 +00:00
process_crosslinks ( state , per_epoch_cache )
2018-12-14 10:12:39 -06:00
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#rewards-and-penalties
process_rewards_and_penalties ( state , per_epoch_cache )
2018-12-14 10:12:39 -06:00
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#ejections
2019-03-08 17:44:31 +00:00
process_ejections ( state )
2019-01-26 11:32:10 -08:00
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#validator-registry-and-shuffling-seed-data
update_registry_and_shuffling_data ( state )
2018-12-27 14:14:37 -06:00
2019-02-26 09:54:08 +00:00
## Regardless of whether or not a validator set change happens run
## process_slashings(state) and process_exit_queue(state)
process_slashings ( state )
2019-03-20 02:19:46 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#final-updates
finish_epoch_update ( state )
2018-12-03 15:41:24 -06:00
2019-04-29 16:48:30 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.0/specs/core/0_beacon-chain.md#state-root-verification
2018-12-21 16:37:46 -06:00
proc verifyStateRoot ( state : BeaconState , blck : BeaconBlock ) : bool =
2019-03-25 10:46:31 -06:00
let state_root = hash_tree_root ( state )
2018-12-21 16:37:46 -06:00
if state_root ! = blck . state_root :
2019-01-25 11:35:22 -06:00
notice " Block: root verification failed " ,
block_state_root = blck . state_root , state_root
2018-12-21 16:37:46 -06:00
false
else :
true
2019-03-26 19:32:35 -06:00
proc advanceState * ( state : var BeaconState ) =
2019-03-21 15:31:36 +00:00
## Sometimes we need to update the state even though we don't have a block at
## hand - this happens for example when a block proposer fails to produce a
## a block.
## https://github.com/ethereum/eth2.0-specs/blob/v0.5.0/specs/core/0_beacon-chain.md#beacon-chain-state-transition-function
## We now define the state transition function. At a high level the state
## transition is made up of four parts:
2019-03-22 18:33:12 +00:00
## 1. State caching, which happens at the start of every slot.
## The state caching, caches the state root of the previous slot
2019-03-21 15:31:36 +00:00
cacheState ( state )
2019-03-22 18:33:12 +00:00
## 2. The per-epoch transitions, which happens at the start of the first
2019-03-21 15:31:36 +00:00
## slot of every epoch.
2019-03-22 18:33:12 +00:00
## The per-epoch transitions focus on the validator registry, including
## adjusting balances and activating and exiting validators, as well as
## processing crosslinks and managing block justification/finalization.
2019-03-21 15:31:36 +00:00
processEpoch ( state )
2019-03-22 18:33:12 +00:00
## 3. The per-slot transitions, which happens at every slot.
## The per-slot transitions focus on the slot counter and block roots
## records updates.
2019-03-26 19:32:35 -06:00
advance_slot ( state )
2019-03-21 15:31:36 +00:00
2019-03-08 10:40:17 -06:00
proc updateState * (
2019-03-26 19:32:35 -06:00
state : var BeaconState , new_block : BeaconBlock , flags : UpdateFlags ) : bool =
2018-12-13 10:00:55 -06:00
## Time in the beacon chain moves by slots. Every time (haha.) that happens,
## we will update the beacon state. Normally, the state updates will be driven
## by the contents of a new block, but it may happen that the block goes
## missing - the state updates happen regardless.
2019-03-08 10:40:17 -06:00
##
2018-12-13 10:00:55 -06:00
## Each call to this function will advance the state by one slot - new_block,
2019-03-08 10:40:17 -06:00
## must match that slot. If the update fails, the state will remain unchanged.
2018-12-18 22:36:10 -06:00
##
2018-12-21 16:37:46 -06:00
## The flags are used to specify that certain validations should be skipped
## for the new block. This is done during block proposal, to create a state
## whose hash can be included in the new block.
2018-12-13 10:00:55 -06:00
#
# TODO this function can be written with a loop inside to handle all empty
# slots up to the slot of the new_block - but then again, why not eagerly
# update the state as time passes? Something to ponder...
2018-12-21 16:37:46 -06:00
# One reason to keep it this way is that you need to look ahead if you're
# the block proposer, though in reality we only need a partial update for
# that
2019-03-08 10:40:17 -06:00
# TODO There's a discussion about what this function should do, and when:
# https://github.com/ethereum/eth2.0-specs/issues/284
2018-12-04 12:45:30 -06:00
# TODO check to which extent this copy can be avoided (considering forks etc),
# for now, it serves as a reminder that we need to handle invalid blocks
# somewhere..
2019-03-08 10:40:17 -06:00
# many functions will mutate `state` partially without rolling back
2018-12-04 12:45:30 -06:00
# the changes in case of failure (look out for `var BeaconState` and
# bool return values...)
2019-03-21 15:31:36 +00:00
## TODO, of cacheState/processEpoch/processSlot/processBloc, only the last
## might fail, so should this bother capturing here, or?
2018-12-27 14:14:37 -06:00
var old_state = state
2018-12-13 10:00:55 -06:00
2019-03-21 15:31:36 +00:00
# These should never fail.
2019-03-26 19:32:35 -06:00
advanceState ( state )
2018-12-21 16:37:46 -06:00
2019-03-08 10:40:17 -06:00
# Block updates - these happen when there's a new block being suggested
# by the block proposer. Every actor in the network will update its state
# according to the contents of this block - but first they will validate
# that the block is sane.
# TODO what should happen if block processing fails?
# https://github.com/ethereum/eth2.0-specs/issues/293
if processBlock ( state , new_block , flags ) :
# This is a bit awkward - at the end of processing we verify that the
# state we arrive at is what the block producer thought it would be -
# meaning that potentially, it could fail verification
if skipValidation in flags or verifyStateRoot ( state , new_block ) :
# State root is what it should be - we're done!
return true
# Block processing failed, roll back changes
state = old_state
false
2018-12-27 14:14:37 -06:00
2019-03-26 19:32:35 -06:00
proc skipSlots * ( state : var BeaconState , slot : Slot ,
2019-03-08 10:40:17 -06:00
afterSlot : proc ( state : BeaconState ) = nil ) =
2019-02-28 15:21:29 -06:00
if state . slot < slot :
2019-03-14 07:33:56 -06:00
debug " Advancing state with empty slots " ,
2019-02-28 15:21:29 -06:00
targetSlot = humaneSlotNum ( slot ) ,
stateSlot = humaneSlotNum ( state . slot )
while state . slot < slot :
2019-03-26 19:32:35 -06:00
advanceState ( state )
2019-03-08 10:40:17 -06:00
if not afterSlot . isNil :
afterSlot ( state )
2019-02-28 15:21:29 -06:00
2019-05-04 08:10:45 -06:00
# TODO hashed versions of above - not in spec
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# https://github.com/ethereum/eth2.0-specs/blob/v0.6.2/specs/core/0_beacon-chain.md#state-caching
2019-05-04 08:10:45 -06:00
func cacheState ( state : var HashedBeaconState ) =
let previous_slot_state_root = state . root
# store the previous slot's post state transition root
state . data . latest_state_roots [ state . data . slot mod SLOTS_PER_HISTORICAL_ROOT ] =
previous_slot_state_root
# cache state root in stored latest_block_header if empty
if state . data . latest_block_header . state_root = = ZERO_HASH :
state . data . latest_block_header . state_root = previous_slot_state_root
# store latest known block for previous slot
state . data . latest_block_roots [ state . data . slot mod SLOTS_PER_HISTORICAL_ROOT ] =
2019-05-09 12:27:37 +00:00
signing_root ( state . data . latest_block_header )
2019-05-04 08:10:45 -06:00
proc advanceState * ( state : var HashedBeaconState ) =
cacheState ( state )
processEpoch ( state . data )
advance_slot ( state . data )
proc updateState * (
state : var HashedBeaconState , blck : BeaconBlock , flags : UpdateFlags ) : bool =
var old_state = state
advanceState ( state )
if processBlock ( state . data , blck , flags ) :
if skipValidation in flags or verifyStateRoot ( state . data , blck ) :
# State root is what it should be - we're done!
# TODO when creating a new block, state_root is not yet set.. comparing
# with zero hash here is a bit fragile however, but this whole thing
# should go away with proper hash caching
state . root =
if blck . state_root = = Eth2Digest ( ) : hash_tree_root ( state . data )
else : blck . state_root
return true
# Block processing failed, roll back changes
state = old_state
false
proc skipSlots * ( state : var HashedBeaconState , slot : Slot ,
afterSlot : proc ( state : HashedBeaconState ) = nil ) =
if state . data . slot < slot :
debug " Advancing state with empty slots " ,
targetSlot = humaneSlotNum ( slot ) ,
stateSlot = humaneSlotNum ( state . data . slot )
while state . data . slot < slot :
advanceState ( state )
if not afterSlot . isNil :
afterSlot ( state )
2018-12-27 14:14:37 -06:00
# TODO document this:
# Jacek Sieka
# @arnetheduck
# Dec 21 11:32
# question about making attestations: in the attestation we carry slot and a justified_slot - just to make sure, this justified_slot is the slot that was justified when the state was at slot, not whatever the client may be seeing now? effectively, because we're attesting to MIN_ATTESTATION_INCLUSION_DELAYold states, it might be that we know about a newer justified slot, but don't include it - correct?
# Danny Ryan
# @djrtwo
# Dec 21 11:34
# You are attesting to what you see as the head of the chain at that slot
# The MIN_ATTESTATION_INCLUSION_DELAY is just how many slots must past before this message can be included on chain
# so whatever the justified_slot was inside the state that was associate with the head you are attesting to
# Jacek Sieka
# @arnetheduck
# Dec 21 11:37
# can I revise an attestation, once I get new information (about the shard or state)?
# Danny Ryan
# @djrtwo
# Dec 21 11:37
# We are attesting to the exact current state at that slot. The MIN_ATTESTATION_INCLUSION_DELAY is to attempt to reduce centralization risk in light of fast block times (ensure attestations have time to fully propagate so fair playing field on including them on chain)
# No, if you create two attestations for the same slot, you can be slashed
# https://github.com/ethereum/eth2.0-specs/blob/master/specs/core/0_beacon-chain.md#is_double_vote
# Jacek Sieka
# @arnetheduck
# Dec 21 11:39
# is there any interest for me to publish my attestation before MIN_ATTESTATION_INCLUSION_DELAY-1 time has passed?
# (apart from the risk of it not being picked up on time)
# Danny Ryan
# @djrtwo
# Dec 21 11:40
# that’ s the main risk.
# Note, we’ re a bit unsure about MIN_ATTESTATION_INCLUSION_DELAY because it might open up an attestors timing strategy too much. In the case where MIN_ATTESTATION_INCLUSION_DELAY is removed, we just set it to 1
# part of validator honesty assumption is to attest during your slot. That said, a rational actor might act in any number of interesting ways..
# Jacek Sieka
# @arnetheduck
# Dec 21 11:59
# I can probably google this somehow, but bls signatures, anyone knows off the top of their head if they have to be combined one by one, or can two group signatures be combined? what happens to overlap then?
# Danny Ryan
# @djrtwo
# Dec 21 12:00
# Yeah, you can do any linear combination of signatures. but you have to remember the linear combination of pubkeys that constructed
# if you have two instances of a signature from pubkey p, then you need 2*p in the group pubkey
# because the attestation bitfield is only 1 bit per pubkey right now, attestations do not support this
# it could be extended to support N overlaps up to N times per pubkey if we had N bits per validator instead of 1
# We are shying away from this for the time being. If there end up being substantial difficulties in network layer aggregation, then adding bits to aid in supporting overlaps is one potential solution
# Jacek Sieka
# @arnetheduck
# Dec 21 12:02
# ah nice, you anticipated my followup question there :) so it's not a straight-off set union operation
# Danny Ryan
# @djrtwo
# Dec 21 12:02
# depending on the particular network level troubles we run into
# right
# aggregatng sigs and pubkeys are both just ec adds https://github.com/ethereum/py-evm/blob/d82b10ae361cde6abbac62f171fcea7809c4e3cf/eth/_utils/bls.py#L191-L202
# subtractions work too (i suppose this is obvious). You can linearly combine sigs or pubs in any way
# Jacek Sieka
# @arnetheduck
# Dec 21 12:05
# hm.. well, because one thing I'm thinking of is the scenario where someone attests to some shard head and I receive that attestation.. now, let's say that's an honest attestation, but within that same slot, I have more fresh information about a shard for example.. now, I can either sign the data in the old attestation or churn out a new one, risking that neither of these get enough votes to be useful - does that sound.. accurate?
# Danny Ryan
# @djrtwo
# Dec 21 12:08
# So you won’ t just be signing the head of the shard. This isn’ t specified yet, but it would be targeting some recent epoch boundary to ensure higher chance of consensus.
# If your recent info is about a better fork in the shard than the one you see the other attester signed, then you are better off signing that fork because if it is winning in your few of the shard chain fork choice, then you would assume it is winning in the view of most attesters shard fork choice
# If some strange circumstance arose in which you saw a majority of attestations that signed something you think is unexpected before you signed, a rational actor might defect to this majority. An honest actor would sign what they believe to be true
# in practice, the actor would have to wait some amount of time past when they should have attested to gather this info.
# also, at the end of the day the validator has to compute the non-outsourcable proof of custody bit, so if the other validators are signing off on some shard chain fork they don’ t know about, then they can’ t attest to that data anyway
# (for fear of signing a bad custody bit)
# so their rational move is to just attest to the data they acutally know about and can accurately compute their proof of custody bit on
# Jacek Sieka
# @arnetheduck
# Dec 21 12:58
# what's justified_block_root doing in attestation_data? isn't that available already as get_block_root(state, attestation.data.justified_slot)?
# also, when we sign hash_tree_root(attestation.data) + bytes1(0) - what's the purpose of the 0 byte, given we have domain already?
# Danny Ryan
# @djrtwo
# Dec 21 13:03
# 0 byte is a stub for the proof of custody bit in phase 0
# If the attestation is included in a short range fork but still votes for the chain it is added to’ s justified_block_root/slot, then we want to count the casper vote
# likely if I see the head of the chain as different from what ends up being the canonical chain, my view of the latest justified block might still be in accordance with the canonical chain
# if my attesation is included in a fork, the head i voted on doesn’ t necessarily lead back to the justified block in the fork. Without including justified_block_root, my vote could be used in any fork for the same epoch even if the block at that justified_slot height was different
# Danny Ryan
# @djrtwo
# Dec 21 13:14
# Long story short, because attestations can be included in forks of the head they are actually attesting to, we can’ t be sure of the justified_block that was being voted on by just usng the justified_slot. The security of properties of Casper FFG require that the voter makes a firm commitment to the actual source block, not just the height of the source block
# Jacek Sieka
# @arnetheduck
# Dec 21 13:23
# ok. that's quite a piece. I'm gonna have to draw some diagrams I think :)
# ah. ok. actually makes sense.. I think :)
# Jacek Sieka
# @arnetheduck
# Dec 21 13:31
# how does that interact then with the following check:
More 0.6.1 updates (#274)
* implement get_churn_limit from 0.6.1; update initiate_validator_exit and get_previous_epoch to 0.6.1; remove obsoleted/non-spec reduce_balance, replaced by decrease_balance; mark BeaconBlock as 0.6.1
* rename get_block_root to get_block_root_at_slot and introduce epoch-oriented get_block_root; implement get_attestation_slot, get_matching_source_attestations, get_matching_target_attestations, get_matching_head_attestations, 0.6.1 get_attesting_indices, get_unslashed_attesting_indices, get_attestation_deltas, process_rewards_and_penalties; rm get_inactivity_penalty
* update Validator and processVoluntaryExits to 0.6.1; rm obsolete inclusion_slots/inclusion_distances
* rm removed activate_validator; mark DepositData, misc values, Gwei values, Randao processing, state caching as 0.6.1; update GENESIS_SLOT to 64, since it doesn't quite yet work at 0
* mark BeaconBlockHeader as 0.6.1; update BeaconBlockBody to 0.6.1; rename WHISTLEBLOWER_REWARD_QUOTIENT to WHISTLEBLOWING_REWARD_QUOTIENT; update slash_validator to 0.6.1
* implement 0.6.2 is_slashable_validator; update processProposerSlashings to 0.6.2; mark state caching as 0.6.2
* mark get_total_active_balance and process_slashings as 0.6.2; update get_base_reward to 0.6.2
* rm prepare_validator_for_withdrawal, process_exit_queue; mark mainnet misc constants as 0.6.2; update process block header to 0.6.2
* address mratsim's code review comment
2019-05-29 10:08:56 +00:00
# Verify that attestation.data.justified_block_root is equal to get_block_root_at_slot(state, attestation.data.justified_slot).
2018-12-27 14:14:37 -06:00
# Danny Ryan
# @djrtwo
# Dec 21 13:32
# ah, my bad above. We only include an attestation on chain if it is for the correct source
# That’ s one of the bare minimum requirements to get it included on chain. Without the justified_block_root, we can’ t do that check
# essentially that checks if this attestation is relevant at all to the current fork’ s consensus.
# if the justified_block is wrong, then we know the target of the vote and the head of the attestation are wrong too
# sorry for the slight mix up there
# logic still holds — the justified_slot alone is not actually a firm commitment to a particular chain history. You need the associated hash
# Jacek Sieka
# @arnetheduck
# Dec 21 13:35
# can't you just follow Block.parent_root?
# well, that, and ultimately.. Block.state_root
# Danny Ryan
# @djrtwo
# Dec 21 13:37
# The block the attestation is included in might not be for the same fork the attestation was made
# we first make sure that the attestation and the block that it’ s included in match at the justified_slot. if not, throw it out
# then in the incentives, we give some extra reward if the epoch_boundary_root and the chain match
# and some extra reward if the beacon_block_root match
# if all three match, then the attestation is fully agreeing with the canonical chain. +1 casper vote and strengthening the head of the fork choice
# if just justified_block_root and epoch_boundary_root match then the attestation agrees enough to successfully cast an ffg vote
# if just justified_block_root match, then at least the attestation agrees on the base of the fork choice, but this isn’ t enough to cast an FFG vote
# Jacek Sieka
# @arnetheduck
# Dec 21 13:41
# if not, throw it out
# it = block or attestation?
# Danny Ryan
# @djrtwo
# Dec 21 13:42
# well, if you are buildling the block ,you shouldn’ t include it (thus throw it out of current consideration). If you are validating a block you just received and that conditon fails for an attestation, throw the block out because it included a bad attestation and is thus invalid
# The block producer knows when producing the block if they are including bad attestations or other data that will fail state transition
# and should not do that
# Jacek Sieka
# @arnetheduck
# Dec 21 13:43
# yeah, that makes sense, just checking
# ok, I think I'm gonna let that sink in a bit before asking more questions.. thanks :)