2018-11-14 16:01:57 -05:00
# Ethereum 2.0 Phase 0 -- The Beacon Chain
2018-09-20 13:20:49 +08:00
2019-04-24 13:37:50 -06:00
**NOTICE**: This document is a work in progress for researchers and implementers.
2018-09-20 13:20:49 +08:00
2018-11-28 15:23:37 +08:00
## Table of contents
2018-12-01 13:29:19 +08:00
<!-- TOC -->
- [Ethereum 2.0 Phase 0 -- The Beacon Chain ](#ethereum-20-phase-0----the-beacon-chain )
- [Table of contents ](#table-of-contents )
- [Introduction ](#introduction )
- [Notation ](#notation )
- [Terminology ](#terminology )
- [Constants ](#constants )
2018-12-04 10:57:09 +00:00
- [Misc ](#misc )
- [Deposit contract ](#deposit-contract )
2019-02-06 10:34:19 -06:00
- [Gwei values ](#gwei-values )
2018-12-04 10:57:09 +00:00
- [Initial values ](#initial-values )
- [Time parameters ](#time-parameters )
2019-01-27 17:25:29 -07:00
- [State list lengths ](#state-list-lengths )
2018-12-04 10:57:09 +00:00
- [Reward and penalty quotients ](#reward-and-penalty-quotients )
2019-03-31 09:02:10 +04:00
- [Max operations per block ](#max-operations-per-block )
2018-12-04 10:57:09 +00:00
- [Signature domains ](#signature-domains )
2018-12-01 13:29:19 +08:00
- [Data structures ](#data-structures )
2019-03-07 11:02:13 -07:00
- [Misc dependencies ](#misc-dependencies )
2019-01-17 03:18:30 -08:00
- [`Fork` ](#fork )
2019-03-07 11:02:13 -07:00
- [`Crosslink` ](#crosslink )
2019-01-17 05:43:47 -06:00
- [`Eth1Data` ](#eth1data )
2019-03-07 11:02:13 -07:00
- [`AttestationData` ](#attestationdata )
- [`AttestationDataAndCustodyBit` ](#attestationdataandcustodybit )
2019-03-26 13:40:19 -06:00
- [`IndexedAttestation` ](#indexedattestation )
2019-03-07 11:02:13 -07:00
- [`DepositData` ](#depositdata )
- [`BeaconBlockHeader` ](#beaconblockheader )
- [`Validator` ](#validator )
- [`PendingAttestation` ](#pendingattestation )
2019-03-08 18:13:05 +01:00
- [`HistoricalBatch` ](#historicalbatch )
2019-03-31 09:02:10 +04:00
- [Beacon operations ](#beacon-operations )
2019-03-07 11:02:13 -07:00
- [`ProposerSlashing` ](#proposerslashing )
- [`AttesterSlashing` ](#attesterslashing )
- [`Attestation` ](#attestation )
- [`Deposit` ](#deposit )
- [`VoluntaryExit` ](#voluntaryexit )
- [`Transfer` ](#transfer )
- [Beacon blocks ](#beacon-blocks )
- [`BeaconBlockBody` ](#beaconblockbody )
- [`BeaconBlock` ](#beaconblock )
- [Beacon state ](#beacon-state )
- [`BeaconState` ](#beaconstate )
2019-01-19 15:58:24 +08:00
- [Custom Types ](#custom-types )
2019-01-28 12:15:43 -07:00
- [Helper functions ](#helper-functions )
2019-03-04 15:49:21 -07:00
- [`xor` ](#xor )
2019-01-28 12:15:43 -07:00
- [`hash` ](#hash )
- [`hash_tree_root` ](#hash_tree_root )
2019-04-08 09:51:13 +08:00
- [`signing_root` ](#signing_root )
2019-01-28 12:15:43 -07:00
- [`slot_to_epoch` ](#slot_to_epoch )
2019-02-03 10:36:21 +01:00
- [`get_previous_epoch` ](#get_previous_epoch )
2019-01-28 12:15:43 -07:00
- [`get_current_epoch` ](#get_current_epoch )
- [`get_epoch_start_slot` ](#get_epoch_start_slot )
- [`is_active_validator` ](#is_active_validator )
2019-03-19 11:08:17 +00:00
- [`is_slashable_validator` ](#is_slashable_validator )
2019-01-28 12:15:43 -07:00
- [`get_active_validator_indices` ](#get_active_validator_indices )
2019-03-21 10:04:20 -06:00
- [`increase_balance` ](#increase_balance )
- [`decrease_balance` ](#decrease_balance )
2019-02-08 05:08:25 +08:00
- [`get_permuted_index` ](#get_permuted_index )
2019-03-19 11:24:36 -06:00
- [`get_split_offset` ](#get_split_offset )
2019-01-28 12:15:43 -07:00
- [`get_epoch_committee_count` ](#get_epoch_committee_count )
2019-04-09 05:52:32 -05:00
- [`get_shard_delta` ](#get_shard_delta )
2019-03-14 20:28:44 -05:00
- [`compute_committee` ](#compute_committee )
2019-01-28 12:15:43 -07:00
- [`get_crosslink_committees_at_slot` ](#get_crosslink_committees_at_slot )
2019-04-24 14:23:51 +10:00
- [`get_block_root_at_slot` ](#get_block_root_at_slot )
2019-01-28 12:15:43 -07:00
- [`get_block_root` ](#get_block_root )
2019-03-04 11:45:41 -07:00
- [`get_state_root` ](#get_state_root )
2019-01-28 12:15:43 -07:00
- [`get_randao_mix` ](#get_randao_mix )
- [`get_active_index_root` ](#get_active_index_root )
- [`generate_seed` ](#generate_seed )
- [`get_beacon_proposer_index` ](#get_beacon_proposer_index )
2019-02-28 21:07:10 -06:00
- [`verify_merkle_branch` ](#verify_merkle_branch )
2019-04-18 14:20:34 +10:00
- [`get_attesting_indices` ](#get_attesting_indices )
2019-01-28 12:15:43 -07:00
- [`int_to_bytes1`, `int_to_bytes2`, ... ](#int_to_bytes1-int_to_bytes2- )
2019-02-08 05:12:58 +08:00
- [`bytes_to_int` ](#bytes_to_int )
2019-02-03 11:43:33 +01:00
- [`get_total_balance` ](#get_total_balance )
2019-01-28 12:15:43 -07:00
- [`get_domain` ](#get_domain )
2019-01-28 13:33:50 -07:00
- [`get_bitfield_bit` ](#get_bitfield_bit )
- [`verify_bitfield` ](#verify_bitfield )
2019-03-26 13:40:19 -06:00
- [`convert_to_indexed` ](#convert_to_indexed )
- [`verify_indexed_attestation` ](#verify_indexed_attestation )
2019-01-28 12:15:43 -07:00
- [`is_double_vote` ](#is_double_vote )
- [`is_surround_vote` ](#is_surround_vote )
- [`integer_squareroot` ](#integer_squareroot )
2019-02-20 07:45:19 +00:00
- [`get_delayed_activation_exit_epoch` ](#get_delayed_activation_exit_epoch )
2019-04-13 22:14:05 -05:00
- [`get_churn_limit` ](#get_churn_limit )
2019-01-28 12:15:43 -07:00
- [`bls_verify` ](#bls_verify )
- [`bls_verify_multiple` ](#bls_verify_multiple )
- [`bls_aggregate_pubkeys` ](#bls_aggregate_pubkeys )
2019-01-28 21:32:36 -07:00
- [Routines for updating validator status ](#routines-for-updating-validator-status )
2019-01-28 12:15:43 -07:00
- [`initiate_validator_exit` ](#initiate_validator_exit )
2019-02-12 12:24:19 +00:00
- [`slash_validator` ](#slash_validator )
- [On genesis ](#on-genesis )
2018-12-01 13:29:19 +08:00
- [Beacon chain state transition function ](#beacon-chain-state-transition-function )
2019-03-08 09:48:46 -07:00
- [State caching ](#state-caching )
2019-01-29 21:17:05 +08:00
- [Per-epoch processing ](#per-epoch-processing )
2019-03-02 00:40:43 -06:00
- [Helper functions ](#helper-functions-1 )
2019-04-18 19:08:34 +10:00
- [Justification and finalization ](#justification-and-finalization )
2019-01-29 21:17:05 +08:00
- [Crosslinks ](#crosslinks )
- [Rewards and penalties ](#rewards-and-penalties )
2019-04-18 19:08:34 +10:00
- [Registry updates ](#registry-updates )
2019-04-06 21:07:03 +11:00
- [Slashings ](#slashings )
2019-01-29 21:17:05 +08:00
- [Final updates ](#final-updates )
2019-03-07 12:05:34 -07:00
- [Per-slot processing ](#per-slot-processing )
- [Per-block processing ](#per-block-processing )
- [Block header ](#block-header )
- [RANDAO ](#randao )
2019-04-18 16:53:02 +08:00
- [Eth1 data ](#eth1-data )
2019-03-31 09:02:10 +04:00
- [Operations ](#operations )
2019-03-08 18:34:51 +01:00
- [Proposer slashings ](#proposer-slashings )
- [Attester slashings ](#attester-slashings )
- [Attestations ](#attestations )
- [Deposits ](#deposits )
- [Voluntary exits ](#voluntary-exits )
- [Transfers ](#transfers )
2019-03-07 12:05:34 -07:00
- [State root verification ](#state-root-verification )
2018-12-01 13:29:19 +08:00
<!-- /TOC -->
2018-11-28 15:23:37 +08:00
## Introduction
2018-09-20 13:20:49 +08:00
2018-11-14 16:01:57 -05:00
This document represents the specification for Phase 0 of Ethereum 2.0 -- The Beacon Chain.
2019-03-27 14:00:28 -05:00
At the core of Ethereum 2.0 is a system chain called the "beacon chain". The beacon chain stores and manages the registry of [validators ](#dfn-validator ). In the initial deployment phases of Ethereum 2.0, the only mechanism to become a [validator ](#dfn-validator ) is to make a one-way ETH transaction to a deposit contract on Ethereum 1.0. Activation as a [validator ](#dfn-validator ) happens when Ethereum 1.0 deposit receipts are processed by the beacon chain, the activation balance is reached, and a queuing process is completed. Exit is either voluntary or done forcibly as a penalty for misbehavior.
2018-09-20 13:20:49 +08:00
2019-03-27 14:00:28 -05:00
The primary source of load on the beacon chain is "attestations". Attestations are simultaneously availability votes for a shard block and proof-of-stake votes for a beacon block. A sufficient number of attestations for the same shard block create a "crosslink", confirming the shard segment up to that shard block into the beacon chain. Crosslinks also serve as infrastructure for asynchronous cross-shard communication.
2018-09-20 13:20:49 +08:00
2018-11-28 20:56:25 -05:00
## Notation
2019-02-11 14:30:36 -07:00
Code snippets appearing in `this style` are to be interpreted as Python code.
2018-11-28 15:23:37 +08:00
## Terminology
2018-09-20 13:20:49 +08:00
2019-03-27 14:00:28 -05:00
* **Validator** < a id = "dfn-validator" ></ a > - a registered participant in the beacon chain. You can become one by sending ether into the Ethereum 1.0 deposit contract.
2019-01-28 19:16:11 +00:00
* **Active validator** < a id = "dfn-active-validator" ></ a > - an active participant in the Ethereum 2.0 consensus invited to, among other things, propose and attest to blocks and vote for crosslinks.
2018-11-30 19:14:17 +01:00
* **Committee** - a (pseudo-) randomly sampled subset of [active validators ](#dfn-active-validator ). When a committee is referred to collectively, as in "this committee attests to X", this is assumed to mean "some subset of that committee that contains enough [validators ](#dfn-validator ) that the protocol recognizes it as representing the committee".
2019-03-27 14:00:28 -05:00
* **Proposer** - the [validator ](#dfn-validator ) that creates a beacon chain block.
2018-11-30 12:31:09 -05:00
* **Attester** - a [validator ](#dfn-validator ) that is part of a committee that needs to sign off on a beacon chain block while simultaneously creating a link (crosslink) to a recent shard block on a particular shard chain.
2018-09-20 13:20:49 +08:00
* **Beacon chain** - the central PoS chain that is the base of the sharding system.
2018-10-01 21:21:15 -05:00
* **Shard chain** - one of the chains on which user transactions take place and account data is stored.
2018-12-13 23:01:32 -05:00
* **Block root** - a 32-byte Merkle root of a beacon chain block or shard chain block. Previously called "block hash".
2019-03-27 14:00:28 -05:00
* **Crosslink** - a set of signatures from a committee attesting to a block in a shard chain that can be included into the beacon chain. Crosslinks are the main means by which the beacon chain "learns about" the updated state of shard chains.
* **Slot** - a period during which one proposer has the ability to create a beacon chain block and some attesters have the ability to make attestations.
* **Epoch** - an aligned span of slots during which all [validators ](#dfn-validator ) get exactly one chance to make an attestation.
2019-04-12 19:54:33 +10:00
* **Finalized**, **justified** - see the [Casper FFG paper ](https://arxiv.org/abs/1710.09437 ).
2019-03-27 14:00:28 -05:00
* **Withdrawal period** - the number of slots between a [validator ](#dfn-validator ) exit and the [validator ](#dfn-validator ) balance being withdrawable.
* **Genesis time** - the Unix time of the genesis beacon chain block at slot 0.
2018-09-20 13:20:49 +08:00
2018-11-28 15:23:37 +08:00
## Constants
2018-09-20 13:20:49 +08:00
2019-04-07 16:21:50 +10:00
Note: the default mainnet values for the constants are included here for spec-design purposes.
The different configurations for mainnet, testnets, and yaml-based testing can be found in the `configs/constant_presets/` directory.
These configurations are updated for releases, but may be out of sync during `dev` changes.
2019-04-03 13:35:40 +11:00
2018-12-04 10:57:09 +00:00
### Misc
2019-02-12 13:37:30 +00:00
| Name | Value |
2019-02-16 18:11:29 +09:00
| - | - |
2019-02-12 13:37:30 +00:00
| `SHARD_COUNT` | `2**10` (= 1,024) |
| `TARGET_COMMITTEE_SIZE` | `2**7` (= 128) |
2019-04-18 14:20:34 +10:00
| `MAX_INDICES_PER_ATTESTATION` | `2**12` (= 4,096) |
2019-04-13 22:14:05 -05:00
| `MIN_PER_EPOCH_CHURN_LIMIT` | `2**2` (= 4) |
2019-04-14 17:28:45 +10:00
| `CHURN_LIMIT_QUOTIENT` | `2**16` (= 65,536) |
2019-04-20 15:17:33 +10:00
| `BASE_REWARDS_PER_EPOCH` | `5` |
2019-02-12 13:37:30 +00:00
| `SHUFFLE_ROUND_COUNT` | 90 |
2018-11-30 14:28:33 +00:00
2019-03-16 13:46:45 +01:00
* For the safety of crosslinks `TARGET_COMMITTEE_SIZE` exceeds [the recommended minimum committee size of 111 ](https://vitalik.ca/files/Ithaca201807_Sharding.pdf ); with sufficient active validators (at least `SLOTS_PER_EPOCH * TARGET_COMMITTEE_SIZE` ), the shuffling algorithm ensures committee sizes of at least `TARGET_COMMITTEE_SIZE` . (Unbiasable randomness with a Verifiable Delay Function (VDF) will improve committee robustness and lower the safe minimum committee size.)
2018-12-04 10:57:09 +00:00
### Deposit contract
2019-02-06 10:34:19 -06:00
| Name | Value |
| - | - |
| `DEPOSIT_CONTRACT_TREE_DEPTH` | `2**5` (= 32) |
### Gwei values
2018-12-04 10:57:09 +00:00
| Name | Value | Unit |
| - | - | :-: |
2019-03-05 17:30:42 +08:00
| `MIN_DEPOSIT_AMOUNT` | `2**0 * 10**9` (= 1,000,000,000) | Gwei |
2019-04-16 16:16:13 +10:00
| `MAX_EFFECTIVE_BALANCE` | `2**5 * 10**9` (= 32,000,000,000) | Gwei |
2019-03-05 17:30:42 +08:00
| `EJECTION_BALANCE` | `2**4 * 10**9` (= 16,000,000,000) | Gwei |
2019-04-22 16:13:46 +10:00
| `EFFECTIVE_BALANCE_INCREMENT` | `2**0 * 10**9` (= 1,000,000,000) | Gwei |
2018-12-04 10:57:09 +00:00
### Initial values
| Name | Value |
| - | - |
2019-04-11 22:28:42 +10:00
| `GENESIS_SLOT` | `0` |
| `GENESIS_EPOCH` | `0` |
2019-01-25 17:33:15 -07:00
| `FAR_FUTURE_EPOCH` | `2**64 - 1` |
2019-01-17 16:30:04 +08:00
| `ZERO_HASH` | `int_to_bytes32(0)` |
| `BLS_WITHDRAWAL_PREFIX_BYTE` | `int_to_bytes1(0)` |
2018-12-04 10:57:09 +00:00
### Time parameters
2018-11-30 14:28:33 +00:00
| Name | Value | Unit | Duration |
| - | - | :-: | :-: |
| `MIN_ATTESTATION_INCLUSION_DELAY` | `2**2` (= 4) | slots | 24 seconds |
2019-02-12 22:38:29 +00:00
| `SLOTS_PER_EPOCH` | `2**6` (= 64) | slots | 6.4 minutes |
2019-01-30 18:22:25 -08:00
| `MIN_SEED_LOOKAHEAD` | `2**0` (= 1) | epochs | 6.4 minutes |
2019-01-31 07:56:48 -08:00
| `ACTIVATION_EXIT_DELAY` | `2**2` (= 4) | epochs | 25.6 minutes |
2019-04-17 14:30:03 +10:00
| `SLOTS_PER_ETH1_VOTING_PERIOD` | `2**10` (= 1,024) | slots | ~1.7 hours |
2019-03-08 15:16:06 -07:00
| `SLOTS_PER_HISTORICAL_ROOT` | `2**13` (= 8,192) | slots | ~13 hours |
2019-02-20 23:58:25 +08:00
| `MIN_VALIDATOR_WITHDRAWABILITY_DELAY` | `2**8` (= 256) | epochs | ~27 hours |
2019-03-08 15:16:06 -07:00
| `PERSISTENT_COMMITTEE_PERIOD` | `2**11` (= 2,048) | epochs | 9 days |
2019-04-13 11:56:55 +10:00
| `MAX_CROSSLINK_EPOCHS` | `2**6` (= 64) | epochs | ~7 hours |
2019-04-18 11:52:14 +10:00
| `MIN_EPOCHS_TO_INACTIVITY_PENALTY` | `2**2` (= 4) | epochs | 25.6 minutes |
2019-03-19 11:21:17 -05:00
* `MAX_CROSSLINK_EPOCHS` should be a small constant times `SHARD_COUNT // SLOTS_PER_EPOCH`
2019-01-27 17:25:29 -07:00
### State list lengths
2019-01-27 23:12:37 -06:00
| Name | Value | Unit | Duration |
| - | - | :-: | :-: |
| `LATEST_RANDAO_MIXES_LENGTH` | `2**13` (= 8,192) | epochs | ~36 days |
2019-01-31 07:56:48 -08:00
| `LATEST_ACTIVE_INDEX_ROOTS_LENGTH` | `2**13` (= 8,192) | epochs | ~36 days |
2019-01-30 18:22:25 -08:00
| `LATEST_SLASHED_EXIT_LENGTH` | `2**13` (= 8,192) | epochs | ~36 days |
2018-11-30 14:28:33 +00:00
2018-12-04 10:57:09 +00:00
### Reward and penalty quotients
2018-10-03 08:27:39 +01:00
2018-11-30 14:28:33 +00:00
| Name | Value |
| - | - |
2019-01-17 02:55:05 -08:00
| `BASE_REWARD_QUOTIENT` | `2**5` (= 32) |
2019-03-25 14:30:59 +00:00
| `WHISTLEBLOWING_REWARD_QUOTIENT` | `2**9` (= 512) |
2019-03-26 13:21:49 +00:00
| `PROPOSER_REWARD_QUOTIENT` | `2**3` (= 8) |
2019-04-17 14:06:28 +10:00
| `INACTIVITY_PENALTY_QUOTIENT` | `2**25` (= 33,554,432) |
2019-04-22 16:34:50 +10:00
| `MIN_SLASHING_PENALTY_QUOTIENT` | `2**5` (= 32) |
2018-10-03 08:27:39 +01:00
2019-04-20 01:45:18 -05:00
* **The `BASE_REWARD_QUOTIENT` is NOT final. Once all other protocol details are finalized it will be adjusted, to target a theoretical maximum total issuance of `2**21` ETH per year if `2**27` ETH is validating (and therefore `2**20` per year if `2**25` ETH is validating, etc etc)**
2019-03-20 08:16:39 +00:00
* The `INACTIVITY_PENALTY_QUOTIENT` equals `INVERSE_SQRT_E_DROP_TIME**2` where `INVERSE_SQRT_E_DROP_TIME := 2**12 epochs` (~18 days) is the time it takes the inactivity penalty to reduce the balance of non-participating [validators ](#dfn-validator ) to about `1/sqrt(e) ~= 60.6%` . Indeed, the balance retained by offline [validators ](#dfn-validator ) after `n` epochs is about `(1 - 1/INACTIVITY_PENALTY_QUOTIENT)**(n**2/2)` so after `INVERSE_SQRT_E_DROP_TIME` epochs it is roughly `(1 - 1/INACTIVITY_PENALTY_QUOTIENT)**(INACTIVITY_PENALTY_QUOTIENT/2) ~= 1/sqrt(e)` .
2018-12-04 10:57:09 +00:00
2019-03-31 09:02:10 +04:00
### Max operations per block
2018-10-04 14:16:31 +01:00
2018-12-06 19:02:23 -06:00
| Name | Value |
2018-12-06 19:07:26 -06:00
| - | - |
2018-12-06 19:02:23 -06:00
| `MAX_PROPOSER_SLASHINGS` | `2**4` (= 16) |
2019-01-28 13:33:50 -07:00
| `MAX_ATTESTER_SLASHINGS` | `2**0` (= 1) |
2018-12-06 19:02:23 -06:00
| `MAX_ATTESTATIONS` | `2**7` (= 128) |
| `MAX_DEPOSITS` | `2**4` (= 16) |
2019-02-12 12:24:19 +00:00
| `MAX_VOLUNTARY_EXITS` | `2**4` (= 16) |
2019-04-19 18:09:29 +10:00
| `MAX_TRANSFERS` | `0` |
2018-10-04 14:16:31 +01:00
2018-12-04 10:57:09 +00:00
### Signature domains
2018-11-19 20:13:58 -05:00
2018-12-10 13:55:11 -06:00
| Name | Value |
2018-11-30 14:28:33 +00:00
| - | - |
2019-03-31 20:48:44 +04:00
| `DOMAIN_BEACON_PROPOSER` | `0` |
2019-03-03 05:04:28 -06:00
| `DOMAIN_RANDAO` | `1` |
| `DOMAIN_ATTESTATION` | `2` |
| `DOMAIN_DEPOSIT` | `3` |
| `DOMAIN_VOLUNTARY_EXIT` | `4` |
2019-03-03 20:14:03 -06:00
| `DOMAIN_TRANSFER` | `5` |
2018-11-30 14:28:33 +00:00
2018-10-04 11:09:39 +01:00
## Data structures
2018-11-30 14:28:33 +00:00
2019-04-12 18:33:53 -05:00
The following data structures are defined as [SimpleSerialize (SSZ) ](../simple-serialize.md ) objects.
2019-01-19 15:58:24 +08:00
2019-03-07 11:02:13 -07:00
The types are defined topologically to aid in facilitating an executable version of the spec.
2018-11-30 14:28:33 +00:00
2019-03-07 11:02:13 -07:00
### Misc dependencies
2018-11-30 14:28:33 +00:00
2019-03-07 11:02:13 -07:00
#### `Fork`
2018-11-30 14:28:33 +00:00
```python
{
2019-03-07 11:02:13 -07:00
# Previous fork version
2019-03-08 09:57:09 +01:00
'previous_version': 'bytes4',
2019-03-07 11:02:13 -07:00
# Current fork version
2019-03-08 09:57:09 +01:00
'current_version': 'bytes4',
2019-03-07 11:02:13 -07:00
# Fork epoch number
'epoch': 'uint64',
2018-11-30 14:28:33 +00:00
}
```
2019-03-07 11:02:13 -07:00
#### `Crosslink`
2018-09-20 13:20:49 +08:00
```python
2018-10-04 11:09:39 +01:00
{
2019-03-07 11:02:13 -07:00
# Epoch number
'epoch': 'uint64',
2019-04-07 17:55:38 +10:00
# Root of the previous crosslink
'previous_crosslink_root': 'bytes32',
2019-04-20 22:48:02 -07:00
# Root of the crosslinked shard data since the previous crosslink
2019-03-07 11:02:13 -07:00
'crosslink_data_root': 'bytes32',
2018-12-06 17:51:01 -06:00
}
```
2019-03-07 11:02:13 -07:00
#### `Eth1Data`
2018-12-06 17:51:01 -06:00
```python
{
2019-03-07 11:02:13 -07:00
# Root of the deposit tree
'deposit_root': 'bytes32',
2019-03-11 23:30:08 -05:00
# Total number of deposits
'deposit_count': 'uint64',
2019-03-07 11:02:13 -07:00
# Block hash
'block_hash': 'bytes32',
2018-12-06 17:51:01 -06:00
}
```
2019-03-07 11:02:13 -07:00
#### `AttestationData`
2018-11-20 02:40:04 -05:00
2018-10-01 21:21:15 -05:00
```python
2018-10-04 11:09:39 +01:00
{
2019-03-12 10:17:34 +00:00
# LMD GHOST vote
2018-10-16 12:55:03 +01:00
'slot': 'uint64',
2019-01-17 16:33:59 +08:00
'beacon_block_root': 'bytes32',
2019-03-12 10:17:34 +00:00
# FFG vote
'source_epoch': 'uint64',
'source_root': 'bytes32',
'target_root': 'bytes32',
# Crosslink vote
'shard': 'uint64',
2019-04-08 12:59:42 +10:00
'previous_crosslink_root': 'bytes32',
2019-02-22 00:21:56 -06:00
'crosslink_data_root': 'bytes32',
2018-10-01 21:21:15 -05:00
}
```
2019-03-07 11:02:13 -07:00
#### `AttestationDataAndCustodyBit`
2018-12-23 05:48:11 -05:00
```python
{
# Attestation data
2019-01-23 09:57:48 +00:00
'data': AttestationData,
2019-01-02 13:35:57 +00:00
# Custody bit
2019-01-28 18:31:13 +00:00
'custody_bit': 'bool',
2018-12-23 05:48:11 -05:00
}
```
2019-03-26 13:40:19 -06:00
#### `IndexedAttestation`
2018-12-06 19:02:23 -06:00
2019-03-07 11:02:13 -07:00
```python
{
# Validator indices
2019-03-28 11:26:04 -06:00
'custody_bit_0_indices': ['uint64'],
'custody_bit_1_indices': ['uint64'],
2019-03-07 11:02:13 -07:00
# Attestation data
'data': AttestationData,
# Aggregate signature
2019-04-17 10:16:01 +10:00
'signature': 'bytes96',
2019-03-07 11:02:13 -07:00
}
```
2019-03-15 11:18:06 +00:00
#### `DepositData`
2018-11-20 02:40:04 -05:00
```python
{
2019-03-07 11:02:13 -07:00
# BLS pubkey
'pubkey': 'bytes48',
# Withdrawal credentials
'withdrawal_credentials': 'bytes32',
2018-12-31 15:14:14 +00:00
# Amount in Gwei
'amount': 'uint64',
2019-03-15 11:18:06 +00:00
# Container self-signature
2019-03-31 20:45:57 +04:00
'signature': 'bytes96',
2018-12-13 23:36:57 +08:00
}
```
2019-03-07 11:02:13 -07:00
#### `BeaconBlockHeader`
2018-12-06 19:13:19 -06:00
```python
{
2019-03-07 11:02:13 -07:00
'slot': 'uint64',
'previous_block_root': 'bytes32',
'state_root': 'bytes32',
'block_body_root': 'bytes32',
'signature': 'bytes96',
}
```
#### `Validator`
```python
{
# BLS public key
2019-01-17 16:51:50 +08:00
'pubkey': 'bytes48',
2018-12-06 19:13:19 -06:00
# Withdrawal credentials
2019-01-17 16:33:59 +08:00
'withdrawal_credentials': 'bytes32',
2019-03-28 08:58:18 -05:00
# Epoch when became eligible for activation
'activation_eligibility_epoch': 'uint64',
2019-03-07 11:02:13 -07:00
# Epoch when validator activated
'activation_epoch': 'uint64',
# Epoch when validator exited
'exit_epoch': 'uint64',
# Epoch when validator is eligible to withdraw
'withdrawable_epoch': 'uint64',
# Was the validator slashed
'slashed': 'bool',
2019-04-22 16:34:50 +10:00
# Effective balance
2019-04-20 15:17:33 +10:00
'effective_balance': 'uint64',
2019-03-07 11:02:13 -07:00
}
```
#### `PendingAttestation`
```python
{
# Attester aggregation bitfield
'aggregation_bitfield': 'bytes',
# Attestation data
'data': AttestationData,
# Inclusion slot
'inclusion_slot': 'uint64',
2019-04-24 14:23:51 +10:00
# Proposer index
'proposer_index': 'uint64',
2019-03-07 11:02:13 -07:00
}
```
2019-03-08 18:13:05 +01:00
#### `HistoricalBatch`
```python
{
2019-03-08 18:14:00 +01:00
# Block roots
2019-03-08 18:13:05 +01:00
'block_roots': ['bytes32', SLOTS_PER_HISTORICAL_ROOT],
2019-03-08 18:14:00 +01:00
# State roots
2019-03-08 18:13:05 +01:00
'state_roots': ['bytes32', SLOTS_PER_HISTORICAL_ROOT],
}
```
2019-03-31 09:02:10 +04:00
### Beacon operations
2019-03-07 11:02:13 -07:00
#### `ProposerSlashing`
```python
{
# Proposer index
'proposer_index': 'uint64',
# First block header
'header_1': BeaconBlockHeader,
# Second block header
'header_2': BeaconBlockHeader,
}
```
#### `AttesterSlashing`
```python
{
2019-03-22 06:10:44 -05:00
# First attestation
2019-03-26 13:40:19 -06:00
'attestation_1': IndexedAttestation,
2019-03-22 06:10:44 -05:00
# Second attestation
2019-03-26 13:40:19 -06:00
'attestation_2': IndexedAttestation,
2019-03-07 11:02:13 -07:00
}
```
#### `Attestation`
```python
{
# Attester aggregation bitfield
'aggregation_bitfield': 'bytes',
# Attestation data
'data': AttestationData,
# Custody bitfield
'custody_bitfield': 'bytes',
# BLS aggregate signature
2019-04-17 10:16:01 +10:00
'signature': 'bytes96',
2018-12-06 19:13:19 -06:00
}
```
2019-03-07 11:02:13 -07:00
#### `Deposit`
2018-12-06 19:13:19 -06:00
2019-03-07 11:02:13 -07:00
```python
{
# Branch in the deposit tree
'proof': ['bytes32', DEPOSIT_CONTRACT_TREE_DEPTH],
# Index in the deposit tree
'index': 'uint64',
# Data
2019-03-15 11:18:06 +00:00
'data': DepositData,
2019-03-07 11:02:13 -07:00
}
```
#### `VoluntaryExit`
2018-12-06 19:02:23 -06:00
```python
{
2019-01-26 15:27:50 -07:00
# Minimum epoch for processing exit
'epoch': 'uint64',
2018-12-06 19:02:23 -06:00
# Index of the exiting validator
2019-01-27 09:01:11 +00:00
'validator_index': 'uint64',
2018-12-06 19:02:23 -06:00
# Validator signature
2019-01-18 03:19:38 +08:00
'signature': 'bytes96',
2018-12-06 19:02:23 -06:00
}
```
2019-03-07 11:02:13 -07:00
#### `Transfer`
2019-02-12 16:41:38 -06:00
```python
{
# Sender index
2019-03-03 05:59:11 -06:00
'sender': 'uint64',
2019-02-12 16:41:38 -06:00
# Recipient index
2019-03-06 08:55:39 -07:00
'recipient': 'uint64',
2019-02-12 16:41:38 -06:00
# Amount in Gwei
'amount': 'uint64',
# Fee in Gwei for block proposer
'fee': 'uint64',
# Inclusion slot
'slot': 'uint64',
# Sender withdrawal pubkey
'pubkey': 'bytes48',
# Sender signature
'signature': 'bytes96',
}
```
2019-03-07 11:02:13 -07:00
### Beacon blocks
2018-12-06 19:02:23 -06:00
2019-03-07 11:02:13 -07:00
#### `BeaconBlockBody`
2018-11-20 02:40:04 -05:00
```python
{
2019-03-07 11:02:13 -07:00
'randao_reveal': 'bytes96',
'eth1_data': Eth1Data,
'proposer_slashings': [ProposerSlashing],
'attester_slashings': [AttesterSlashing],
'attestations': [Attestation],
'deposits': [Deposit],
'voluntary_exits': [VoluntaryExit],
'transfers': [Transfer],
2019-03-03 05:04:28 -06:00
}
```
2019-03-07 11:02:13 -07:00
#### `BeaconBlock`
2019-03-03 05:04:28 -06:00
```python
{
2019-03-07 11:02:13 -07:00
# Header
2019-03-03 05:04:28 -06:00
'slot': 'uint64',
'previous_block_root': 'bytes32',
'state_root': 'bytes32',
2019-03-07 11:02:13 -07:00
'body': BeaconBlockBody,
2019-02-15 00:23:03 +00:00
'signature': 'bytes96',
2018-11-20 02:40:04 -05:00
}
```
2019-03-07 11:02:13 -07:00
### Beacon state
2018-09-20 13:20:49 +08:00
2018-11-30 14:28:33 +00:00
#### `BeaconState`
2018-09-20 13:20:49 +08:00
```python
2018-10-04 11:09:39 +01:00
{
2018-12-04 21:28:31 +00:00
# Misc
'slot': 'uint64',
'genesis_time': 'uint64',
2019-01-17 03:18:30 -08:00
'fork': Fork, # For versioning hard forks
2018-12-04 21:28:31 +00:00
2018-11-30 14:28:33 +00:00
# Validator registry
2019-01-17 20:01:13 +09:00
'validator_registry': [Validator],
2019-03-12 15:58:31 +00:00
'balances': ['uint64'],
2018-11-30 14:28:33 +00:00
# Randomness and committees
2019-03-06 14:46:52 +01:00
'latest_randao_mixes': ['bytes32', LATEST_RANDAO_MIXES_LENGTH],
2019-03-19 11:21:17 -05:00
'latest_start_shard': 'uint64',
2018-12-20 16:01:00 -05:00
2018-11-30 14:28:33 +00:00
# Finality
2019-03-03 04:28:13 -06:00
'previous_epoch_attestations': [PendingAttestation],
'current_epoch_attestations': [PendingAttestation],
2019-01-26 08:16:32 -07:00
'previous_justified_epoch': 'uint64',
2019-03-12 10:17:34 +00:00
'current_justified_epoch': 'uint64',
2019-03-11 14:38:10 -06:00
'previous_justified_root': 'bytes32',
2019-03-12 10:17:34 +00:00
'current_justified_root': 'bytes32',
2018-12-04 10:57:09 +00:00
'justification_bitfield': 'uint64',
2019-01-26 08:16:32 -07:00
'finalized_epoch': 'uint64',
2019-03-12 10:26:34 +00:00
'finalized_root': 'bytes32',
2018-11-30 14:28:33 +00:00
# Recent state
2019-04-02 16:00:36 +11:00
'current_crosslinks': [Crosslink, SHARD_COUNT],
'previous_crosslinks': [Crosslink, SHARD_COUNT],
2019-03-06 14:46:52 +01:00
'latest_block_roots': ['bytes32', SLOTS_PER_HISTORICAL_ROOT],
2019-03-07 08:56:29 -07:00
'latest_state_roots': ['bytes32', SLOTS_PER_HISTORICAL_ROOT],
2019-03-06 14:46:52 +01:00
'latest_active_index_roots': ['bytes32', LATEST_ACTIVE_INDEX_ROOTS_LENGTH],
'latest_slashed_balances': ['uint64', LATEST_SLASHED_EXIT_LENGTH], # Balances slashed at every withdrawal period
2019-03-03 05:04:28 -06:00
'latest_block_header': BeaconBlockHeader, # `latest_block_header.state_root == ZERO_HASH` temporarily
2019-03-07 09:44:55 -07:00
'historical_roots': ['bytes32'],
2018-11-30 14:28:33 +00:00
2019-01-17 05:43:47 -06:00
# Ethereum 1.0 chain data
'latest_eth1_data': Eth1Data,
2019-04-17 14:30:03 +10:00
'eth1_data_votes': [Eth1Data],
2019-03-16 05:16:47 +01:00
'deposit_index': 'uint64',
2018-09-20 13:20:49 +08:00
}
```
2019-01-19 15:58:24 +08:00
## Custom Types
2019-01-02 13:18:35 +00:00
2019-01-19 15:58:24 +08:00
We define the following Python custom types for type hinting and readability:
2019-01-02 13:18:35 +00:00
2019-01-27 09:01:11 +00:00
| Name | SSZ equivalent | Description |
2019-01-19 15:58:24 +08:00
| - | - | - |
2019-01-31 07:58:31 -08:00
| `Slot` | `uint64` | a slot number |
| `Epoch` | `uint64` | an epoch number |
| `Shard` | `uint64` | a shard number |
2019-02-12 12:24:19 +00:00
| `ValidatorIndex` | `uint64` | a validator registry index |
2019-01-27 09:01:11 +00:00
| `Gwei` | `uint64` | an amount in Gwei |
| `Bytes32` | `bytes32` | 32 bytes of binary data |
2019-02-01 20:42:11 -08:00
| `BLSPubkey` | `bytes48` | a BLS12-381 public key |
| `BLSSignature` | `bytes96` | a BLS12-381 signature |
2018-12-04 10:57:09 +00:00
2019-01-28 12:15:43 -07:00
## Helper functions
2018-12-04 11:47:41 +00:00
2018-11-30 14:28:33 +00:00
Note: The definitions below are for specification purposes and are not necessarily optimal implementations.
2018-12-04 11:47:41 +00:00
2019-03-04 15:49:21 -07:00
### `xor`
```python
def xor(bytes1: Bytes32, bytes2: Bytes32) -> Bytes32:
return bytes(a ^ b for a, b in zip(bytes1, bytes2))
```
2019-01-28 12:15:43 -07:00
### `hash`
2018-12-04 11:47:41 +00:00
2019-03-15 10:51:08 +00:00
The `hash` function is SHA256.
2018-12-04 11:47:41 +00:00
2018-12-11 20:30:28 +00:00
Note: We aim to migrate to a S[T/N]ARK-friendly hash function in a future Ethereum 2.0 deployment phase.
2018-12-04 11:47:41 +00:00
2019-01-28 12:15:43 -07:00
### `hash_tree_root`
2018-12-04 11:47:41 +00:00
2019-04-12 18:33:53 -05:00
`def hash_tree_root(object: SSZSerializable) -> Bytes32` is a function for hashing objects into a single root utilizing a hash tree structure. `hash_tree_root` is defined in the [SimpleSerialize spec ](../simple-serialize.md#merkleization ).
2018-12-04 11:47:41 +00:00
2019-04-08 09:51:13 +08:00
### `signing_root`
2019-02-15 00:23:03 +00:00
2019-04-12 18:33:53 -05:00
`def signing_root(object: SSZContainer) -> Bytes32` is a function defined in the [SimpleSerialize spec ](../simple-serialize.md#self-signed-containers ) to compute signing messages.
2019-02-15 00:23:03 +00:00
2019-01-28 12:15:43 -07:00
### `slot_to_epoch`
2018-12-04 10:57:09 +00:00
```python
2019-01-31 07:58:31 -08:00
def slot_to_epoch(slot: Slot) -> Epoch:
2019-01-30 17:25:39 +08:00
"""
2019-01-30 23:01:38 +08:00
Return the epoch number of the given ``slot` `.
2019-01-30 17:25:39 +08:00
"""
2019-02-12 22:38:29 +00:00
return slot // SLOTS_PER_EPOCH
2018-12-04 10:57:09 +00:00
```
2019-02-03 10:36:21 +01:00
### `get_previous_epoch`
```python
2019-02-12 12:24:19 +00:00
def get_previous_epoch(state: BeaconState) -> Epoch:
2019-02-03 10:36:21 +01:00
"""`
Return the previous epoch of the given ``state` `.
2019-04-14 22:12:39 +10:00
Return the current epoch if it's genesis epoch.
2019-02-03 10:36:21 +01:00
"""
2019-04-11 22:48:27 +10:00
current_epoch = get_current_epoch(state)
return (current_epoch - 1) if current_epoch > GENESIS_EPOCH else current_epoch
2019-02-03 10:36:21 +01:00
```
2019-01-28 12:15:43 -07:00
### `get_current_epoch`
2018-09-20 13:20:49 +08:00
2019-01-26 07:31:09 -07:00
```python
2019-01-31 07:58:31 -08:00
def get_current_epoch(state: BeaconState) -> Epoch:
2019-01-30 17:25:39 +08:00
"""
2019-01-30 23:01:38 +08:00
Return the current epoch of the given ``state` `.
2019-01-30 17:25:39 +08:00
"""
2019-01-26 07:31:09 -07:00
return slot_to_epoch(state.slot)
2018-11-22 08:24:20 -05:00
```
2018-09-20 13:20:49 +08:00
2019-01-28 12:15:43 -07:00
### `get_epoch_start_slot`
2018-12-11 20:30:28 +00:00
2019-01-27 07:54:46 -07:00
```python
2019-01-31 07:58:31 -08:00
def get_epoch_start_slot(epoch: Epoch) -> Slot:
2019-01-30 17:25:39 +08:00
"""
2019-01-30 23:01:38 +08:00
Return the starting slot of the given ``epoch` `.
2019-01-30 17:25:39 +08:00
"""
2019-02-12 22:38:29 +00:00
return epoch * SLOTS_PER_EPOCH
2019-01-27 07:54:46 -07:00
```
2018-12-14 09:29:49 -06:00
2019-01-28 12:15:43 -07:00
### `is_active_validator`
2019-04-24 14:23:51 +10:00
2018-12-13 03:00:53 +08:00
```python
2019-01-31 07:58:31 -08:00
def is_active_validator(validator: Validator, epoch: Epoch) -> bool:
2018-12-07 12:18:55 +00:00
"""
2019-01-30 23:01:38 +08:00
Check if ``validator` ` is active.
2018-12-07 12:18:55 +00:00
"""
2019-01-26 08:16:32 -07:00
return validator.activation_epoch < = epoch < validator.exit_epoch
2018-12-07 12:18:55 +00:00
```
2019-03-19 11:08:17 +00:00
### `is_slashable_validator`
2019-04-24 14:23:51 +10:00
2019-03-19 11:08:17 +00:00
```python
def is_slashable_validator(validator: Validator, epoch: Epoch) -> bool:
"""
Check if ``validator` ` is slashable.
"""
2019-04-24 14:23:51 +10:00
return validator.slashed is False and (validator.activation_epoch < = epoch < validator.withdrawable_epoch )
2019-03-19 11:08:17 +00:00
```
2019-01-28 12:15:43 -07:00
### `get_active_validator_indices`
2018-09-20 13:20:49 +08:00
```python
2019-04-14 17:28:45 +10:00
def get_active_validator_indices(state: BeaconState, epoch: Epoch) -> List[ValidatorIndex]:
2018-11-30 14:28:33 +00:00
"""
2019-04-14 17:28:45 +10:00
Get active validator indices at ``epoch` `.
2018-11-30 14:28:33 +00:00
"""
2019-04-14 17:28:45 +10:00
return [i for i, v in enumerate(state.validator_registry) if is_active_validator(v, epoch)]
2018-09-20 13:20:49 +08:00
```
2018-11-04 23:26:24 +01:00
2019-03-21 10:04:20 -06:00
### `increase_balance`
2019-03-07 01:38:03 -06:00
2019-03-21 10:04:20 -06:00
```python
2019-03-22 12:56:54 +08:00
def increase_balance(state: BeaconState, index: ValidatorIndex, delta: Gwei) -> None:
"""
2019-04-20 15:17:33 +10:00
Increase validator balance by ``delta` `.
2019-03-22 12:56:54 +08:00
"""
2019-04-20 15:17:33 +10:00
state.balances[index] += delta
2019-03-21 10:04:20 -06:00
```
2019-03-07 01:38:03 -06:00
2019-03-21 10:04:20 -06:00
### `decrease_balance`
2019-03-07 01:38:03 -06:00
2019-03-21 10:04:20 -06:00
```python
2019-03-22 12:56:54 +08:00
def decrease_balance(state: BeaconState, index: ValidatorIndex, delta: Gwei) -> None:
"""
2019-04-20 15:17:33 +10:00
Decrease validator balance by ``delta` ` with underflow protection.
2019-03-22 12:56:54 +08:00
"""
2019-04-22 16:34:50 +10:00
state.balances[index] = 0 if delta > state.balances[index] else state.balances[index] - delta
2019-03-21 10:04:20 -06:00
```
2019-03-07 01:38:03 -06:00
2019-02-08 05:08:25 +08:00
### `get_permuted_index`
2018-09-20 13:20:49 +08:00
```python
2019-02-07 21:55:33 -06:00
def get_permuted_index(index: int, list_size: int, seed: Bytes32) -> int:
2018-10-07 14:21:37 +08:00
"""
2019-03-20 08:16:39 +00:00
Return `p(index)` in a pseudorandom permutation `p` of `0...list_size - 1` with ``seed` ` as entropy.
2018-10-07 14:21:37 +08:00
2019-02-06 18:33:22 -06:00
Utilizes 'swap or not' shuffling found in
https://link.springer.com/content/pdf/10.1007%2F978-3-642-32009-5_1.pdf
See the 'generalized domain' algorithm on page 3.
2018-10-07 14:21:37 +08:00
"""
2019-02-15 01:14:59 +11:00
assert index < list_size
2019-02-16 17:27:47 +11:00
assert list_size < = 2**40
2019-04-18 14:43:24 +10:00
2019-02-07 21:55:33 -06:00
for round in range(SHUFFLE_ROUND_COUNT):
2019-02-07 21:51:56 -06:00
pivot = bytes_to_int(hash(seed + int_to_bytes1(round))[0:8]) % list_size
2019-02-08 05:08:25 +08:00
flip = (pivot - index) % list_size
position = max(index, flip)
source = hash(seed + int_to_bytes1(round) + int_to_bytes4(position // 256))
byte = source[(position % 256) // 8]
bit = (byte >> (position % 8)) % 2
index = flip if bit else index
2018-10-07 14:21:37 +08:00
2019-02-08 05:08:25 +08:00
return index
2018-09-20 13:20:49 +08:00
```
2019-03-19 11:24:36 -06:00
### `get_split_offset`
2018-09-20 13:20:49 +08:00
```python
2019-03-21 08:24:26 -06:00
def get_split_offset(list_size: int, chunks: int, index: int) -> int:
2019-04-07 01:24:50 +11:00
"""
Returns a value such that for a list L, chunk count k and index i,
split(L, k)[i] == L[get_split_offset(len(L), k, i): get_split_offset(len(L), k, i+1)]
"""
return (list_size * index) // chunks
2018-09-20 13:20:49 +08:00
```
2019-01-28 12:15:43 -07:00
### `get_epoch_committee_count`
2019-01-07 18:53:33 -06:00
```python
2019-04-09 05:52:32 -05:00
def get_epoch_committee_count(state: BeaconState, epoch: Epoch) -> int:
2019-01-30 17:25:39 +08:00
"""
2019-04-23 07:16:52 -07:00
Return the number of committees at ``epoch` `.
2019-01-30 17:25:39 +08:00
"""
2019-04-24 14:23:51 +10:00
active_validator_indices = get_active_validator_indices(state, epoch)
2019-01-07 18:53:33 -06:00
return max(
1,
min(
2019-02-12 22:38:29 +00:00
SHARD_COUNT // SLOTS_PER_EPOCH,
2019-04-24 14:23:51 +10:00
len(active_validator_indices) // SLOTS_PER_EPOCH // TARGET_COMMITTEE_SIZE,
2019-01-07 18:53:33 -06:00
)
2019-02-12 22:38:29 +00:00
) * SLOTS_PER_EPOCH
2019-01-07 18:53:33 -06:00
```
2019-04-09 05:52:32 -05:00
### `get_shard_delta`
```python
def get_shard_delta(state: BeaconState, epoch: Epoch) -> int:
2019-04-23 07:16:52 -07:00
"""
2019-04-23 08:36:40 -07:00
Return the number of shards to increment ``state.latest_start_shard`` during ``epoch` `.
2019-04-23 07:16:52 -07:00
"""
2019-04-09 05:52:32 -05:00
return min(get_epoch_committee_count(state, epoch), SHARD_COUNT - SHARD_COUNT // SLOTS_PER_EPOCH)
```
2019-03-14 20:28:44 -05:00
### `compute_committee`
2018-09-20 13:20:49 +08:00
```python
2019-03-15 09:43:38 +00:00
def compute_committee(validator_indices: List[ValidatorIndex],
2019-03-14 20:28:44 -05:00
seed: Bytes32,
index: int,
total_committees: int) -> List[ValidatorIndex]:
2018-11-30 14:28:33 +00:00
"""
2019-03-15 09:45:20 +00:00
Return the ``index``'th shuffled committee out of a total ``total_committees` `
using ``validator_indices`` and ``seed` `.
2018-11-30 14:28:33 +00:00
"""
2019-03-14 20:28:44 -05:00
start_offset = get_split_offset(len(validator_indices), total_committees, index)
end_offset = get_split_offset(len(validator_indices), total_committees, index + 1)
return [
validator_indices[get_permuted_index(i, len(validator_indices), seed)]
for i in range(start_offset, end_offset)
]
2019-01-07 18:53:33 -06:00
```
2018-10-07 14:21:37 +08:00
2019-04-17 14:06:28 +10:00
Note: this definition and the next few definitions are highly inefficient as algorithms, as they re-calculate many sub-expressions. Production implementations are expected to appropriately use caching/memoization to avoid redoing work.
2018-10-14 08:30:52 -05:00
2019-01-28 12:15:43 -07:00
### `get_crosslink_committees_at_slot`
2018-09-20 13:20:49 +08:00
```python
2019-01-14 16:06:33 -08:00
def get_crosslink_committees_at_slot(state: BeaconState,
2019-03-26 12:32:24 -06:00
slot: Slot) -> List[Tuple[List[ValidatorIndex], Shard]]:
2018-11-30 14:28:33 +00:00
"""
2019-01-30 23:01:38 +08:00
Return the list of ``(committee, shard)`` tuples for the ``slot` `.
2018-11-30 14:28:33 +00:00
"""
2019-01-26 15:39:57 -07:00
epoch = slot_to_epoch(slot)
current_epoch = get_current_epoch(state)
2019-02-03 10:36:21 +01:00
previous_epoch = get_previous_epoch(state)
2019-01-26 15:39:57 -07:00
next_epoch = current_epoch + 1
2019-01-07 18:53:33 -06:00
2019-01-29 16:29:51 -08:00
assert previous_epoch < = epoch < = next_epoch
2019-04-14 17:28:45 +10:00
indices = get_active_validator_indices(state, epoch)
2019-01-26 15:39:57 -07:00
2019-02-22 12:51:09 -07:00
if epoch == current_epoch:
2019-03-19 17:13:25 +00:00
start_shard = state.latest_start_shard
2019-02-22 12:51:09 -07:00
elif epoch == previous_epoch:
2019-04-09 05:52:32 -05:00
previous_shard_delta = get_shard_delta(state, previous_epoch)
start_shard = (state.latest_start_shard - previous_shard_delta) % SHARD_COUNT
2019-01-29 16:29:51 -08:00
elif epoch == next_epoch:
2019-04-09 05:52:32 -05:00
current_shard_delta = get_shard_delta(state, current_epoch)
start_shard = (state.latest_start_shard + current_shard_delta) % SHARD_COUNT
2019-01-23 15:41:25 -06:00
2019-04-09 05:52:32 -05:00
committees_per_epoch = get_epoch_committee_count(state, epoch)
2019-02-12 22:38:29 +00:00
committees_per_slot = committees_per_epoch // SLOTS_PER_EPOCH
2019-03-19 11:15:51 -06:00
offset = slot % SLOTS_PER_EPOCH
2019-03-19 17:13:25 +00:00
slot_start_shard = (start_shard + committees_per_slot * offset) % SHARD_COUNT
2019-03-26 10:13:17 -06:00
seed = generate_seed(state, epoch)
2019-01-07 18:53:33 -06:00
2019-01-10 11:01:19 -06:00
return [
2019-01-14 21:54:29 +08:00
(
2019-03-19 11:40:24 -06:00
compute_committee(indices, seed, committees_per_slot * offset + i, committees_per_epoch),
2019-01-14 21:54:29 +08:00
(slot_start_shard + i) % SHARD_COUNT,
)
2019-01-10 11:01:19 -06:00
for i in range(committees_per_slot)
]
2018-11-30 14:28:33 +00:00
```
2018-09-20 13:20:49 +08:00
2019-04-24 14:23:51 +10:00
### `get_block_root_at_slot`
2018-11-30 14:28:33 +00:00
```python
2019-04-24 14:23:51 +10:00
def get_block_root_at_slot(state: BeaconState,
slot: Slot) -> Bytes32:
2018-11-30 14:28:33 +00:00
"""
2019-01-30 23:01:38 +08:00
Return the block root at a recent ``slot` `.
2018-11-30 14:28:33 +00:00
"""
2019-03-03 05:04:28 -06:00
assert slot < state.slot < = slot + SLOTS_PER_HISTORICAL_ROOT
return state.latest_block_roots[slot % SLOTS_PER_HISTORICAL_ROOT]
2018-09-20 13:20:49 +08:00
```
2019-04-24 14:23:51 +10:00
### `get_block_root`
2019-03-04 11:45:41 -07:00
```python
2019-04-24 14:23:51 +10:00
def get_block_root(state: BeaconState,
epoch: Epoch) -> Bytes32:
2019-03-04 11:45:41 -07:00
"""
2019-04-24 14:23:51 +10:00
Return the block root at a recent ``epoch` `.
2019-03-04 11:45:41 -07:00
"""
2019-04-24 14:23:51 +10:00
return get_block_root_at_slot(state, get_epoch_start_slot(epoch))
2019-03-04 11:45:41 -07:00
```
2018-09-20 13:20:49 +08:00
2019-03-04 11:45:41 -07:00
### `get_state_root`
```python
def get_state_root(state: BeaconState,
slot: Slot) -> Bytes32:
"""
Return the state root at a recent ``slot` `.
"""
assert slot < state.slot < = slot + SLOTS_PER_HISTORICAL_ROOT
return state.latest_state_roots[slot % SLOTS_PER_HISTORICAL_ROOT]
```
2019-04-24 11:31:24 -06:00
2019-01-28 12:15:43 -07:00
### `get_randao_mix`
2019-01-10 11:01:19 -06:00
```python
def get_randao_mix(state: BeaconState,
2019-01-31 07:58:31 -08:00
epoch: Epoch) -> Bytes32:
2019-01-10 11:01:19 -06:00
"""
2019-01-30 23:01:38 +08:00
Return the randao mix at a recent ``epoch` `.
2019-01-10 11:01:19 -06:00
"""
2019-01-27 23:12:37 -06:00
assert get_current_epoch(state) - LATEST_RANDAO_MIXES_LENGTH < epoch < = get_current_epoch ( state )
2019-01-27 10:22:27 -07:00
return state.latest_randao_mixes[epoch % LATEST_RANDAO_MIXES_LENGTH]
2019-01-10 11:01:19 -06:00
```
2019-01-28 12:15:43 -07:00
### `get_active_index_root`
2019-01-18 21:06:21 -06:00
```python
def get_active_index_root(state: BeaconState,
2019-01-31 07:58:31 -08:00
epoch: Epoch) -> Bytes32:
2019-01-18 21:06:21 -06:00
"""
2019-01-30 23:01:38 +08:00
Return the index root at a recent ``epoch` `.
2019-01-18 21:06:21 -06:00
"""
2019-01-31 07:56:48 -08:00
assert get_current_epoch(state) - LATEST_ACTIVE_INDEX_ROOTS_LENGTH + ACTIVATION_EXIT_DELAY < epoch < = get_current_epoch ( state ) + ACTIVATION_EXIT_DELAY
return state.latest_active_index_roots[epoch % LATEST_ACTIVE_INDEX_ROOTS_LENGTH]
2019-01-18 21:06:21 -06:00
```
2019-01-28 12:15:43 -07:00
### `generate_seed`
2019-01-25 16:07:54 -07:00
```python
def generate_seed(state: BeaconState,
2019-01-31 07:58:31 -08:00
epoch: Epoch) -> Bytes32:
2019-01-10 11:01:19 -06:00
"""
2019-01-25 17:33:15 -07:00
Generate a seed for the given ``epoch` `.
2019-01-10 11:01:19 -06:00
"""
2019-01-25 16:07:54 -07:00
return hash(
2019-01-30 18:22:25 -08:00
get_randao_mix(state, epoch - MIN_SEED_LOOKAHEAD) +
2019-02-01 21:02:09 -08:00
get_active_index_root(state, epoch) +
int_to_bytes32(epoch)
2019-01-25 16:07:54 -07:00
)
2019-01-10 11:01:19 -06:00
```
2019-01-28 12:15:43 -07:00
### `get_beacon_proposer_index`
2018-11-20 02:40:04 -05:00
```python
2019-04-17 12:32:50 +10:00
def get_beacon_proposer_index(state: BeaconState) -> ValidatorIndex:
2018-11-30 14:28:33 +00:00
"""
2019-04-17 12:32:50 +10:00
Return the beacon proposer index at ``state.slot` `.
2018-11-30 14:28:33 +00:00
"""
2019-02-26 15:55:27 -07:00
current_epoch = get_current_epoch(state)
2019-04-17 12:32:50 +10:00
first_committee, _ = get_crosslink_committees_at_slot(state, state.slot)[0]
2019-04-24 14:23:51 +10:00
MAX_RANDOM_BYTE = 2**8 - 1
2019-05-01 09:33:22 +01:00
seed = generate_seed(state, current_epoch)
2019-03-14 18:57:17 +00:00
i = 0
while True:
2019-04-24 14:35:02 +10:00
candidate_index = first_committee[(current_epoch + i) % len(first_committee)]
2019-05-01 09:34:09 +01:00
random_byte = hash(seed + int_to_bytes8(i // 32))[i % 32]
2019-04-22 16:34:50 +10:00
effective_balance = state.validator_registry[candidate_index].effective_balance
2019-04-24 14:46:28 +10:00
if effective_balance * MAX_RANDOM_BYTE >= MAX_EFFECTIVE_BALANCE * random_byte:
2019-04-22 16:34:50 +10:00
return candidate_index
2019-03-13 21:42:49 -05:00
i += 1
2018-11-20 02:40:04 -05:00
```
2019-02-28 21:07:10 -06:00
### `verify_merkle_branch`
```python
2019-03-01 19:01:40 -06:00
def verify_merkle_branch(leaf: Bytes32, proof: List[Bytes32], depth: int, index: int, root: Bytes32) -> bool:
2019-02-28 21:07:10 -06:00
"""
2019-03-01 19:01:40 -06:00
Verify that the given ``leaf`` is on the merkle branch ``proof` `
starting with the given ``root` `.
2019-02-28 21:07:10 -06:00
"""
value = leaf
for i in range(depth):
if index // (2**i) % 2:
2019-03-01 19:01:40 -06:00
value = hash(proof[i] + value)
2019-02-28 21:07:10 -06:00
else:
2019-03-01 19:01:40 -06:00
value = hash(value + proof[i])
2019-02-28 21:07:10 -06:00
return value == root
```
2019-04-18 14:20:34 +10:00
### `get_attesting_indices`
2019-03-27 08:31:56 -06:00
```python
2019-04-18 14:20:34 +10:00
def get_attesting_indices(state: BeaconState,
2019-04-18 18:11:18 +10:00
attestation_data: AttestationData,
bitfield: bytes) -> List[ValidatorIndex]:
2018-11-30 14:28:33 +00:00
"""
2019-04-18 14:20:34 +10:00
Return the sorted attesting indices corresponding to ``attestation_data`` and ``bitfield` `.
2018-11-30 14:28:33 +00:00
"""
2019-04-18 18:11:18 +10:00
crosslink_committees = get_crosslink_committees_at_slot(state, attestation_data.slot)
crosslink_committee = [committee for committee, shard in crosslink_committees if shard == attestation_data.shard][0]
2019-01-26 18:59:07 +00:00
assert verify_bitfield(bitfield, len(crosslink_committee))
2019-04-17 12:32:50 +10:00
return sorted([index for i, index in enumerate(crosslink_committee) if get_bitfield_bit(bitfield, i) == 0b1])
2018-11-20 02:40:04 -05:00
```
2019-01-28 12:15:43 -07:00
### `int_to_bytes1`, `int_to_bytes2`, ...
2018-11-30 14:28:33 +00:00
2019-02-02 16:41:59 -08:00
`int_to_bytes1(x): return x.to_bytes(1, 'little')` , `int_to_bytes2(x): return x.to_bytes(2, 'little')` , and so on for all integers, particularly 1, 2, 3, 4, 8, 32, 48, 96.
2018-09-20 13:20:49 +08:00
2019-02-08 05:12:58 +08:00
### `bytes_to_int`
```python
def bytes_to_int(data: bytes) -> int:
return int.from_bytes(data, 'little')
```
2018-09-20 13:20:49 +08:00
2019-02-03 11:43:33 +01:00
### `get_total_balance`
2019-02-03 11:14:02 +01:00
```python
2019-04-22 16:13:46 +10:00
def get_total_balance(state: BeaconState, indices: List[ValidatorIndex]) -> Gwei:
2019-02-03 11:14:02 +01:00
"""
2019-03-04 06:45:55 -08:00
Return the combined effective balance of an array of ``validators` `.
2019-02-03 11:14:02 +01:00
"""
2019-04-22 16:13:46 +10:00
return sum([state.validator_registry[index].effective_balance for index in indices])
2019-02-03 11:14:02 +01:00
```
2019-01-28 12:15:43 -07:00
### `get_domain`
2018-12-05 06:41:11 -08:00
```python
2019-03-31 04:55:24 -05:00
def get_domain(state: BeaconState,
domain_type: int,
2019-04-02 22:30:26 +04:00
message_epoch: int=None) -> int:
2019-01-30 17:25:39 +08:00
"""
2019-04-02 22:30:26 +04:00
Return the signature domain (fork version concatenated with domain type) of a message.
2019-01-30 17:25:39 +08:00
"""
2019-04-02 22:30:26 +04:00
epoch = get_current_epoch(state) if message_epoch is None else message_epoch
fork_version = state.fork.previous_version if epoch < state.fork.epoch else state . fork . current_version
return bytes_to_int(fork_version + int_to_bytes4(domain_type))
2018-12-05 06:41:11 -08:00
```
2019-01-28 13:33:50 -07:00
### `get_bitfield_bit`
2019-01-23 12:40:59 +00:00
```python
def get_bitfield_bit(bitfield: bytes, i: int) -> int:
2019-01-23 13:52:52 +00:00
"""
Extract the bit in ``bitfield`` at position ``i` `.
"""
2019-02-12 12:52:07 +01:00
return (bitfield[i // 8] >> (i % 8)) % 2
2019-01-23 12:40:59 +00:00
```
2019-01-28 13:33:50 -07:00
### `verify_bitfield`
2019-01-23 12:40:59 +00:00
```python
2019-01-26 18:59:07 +00:00
def verify_bitfield(bitfield: bytes, committee_size: int) -> bool:
2019-01-23 13:52:52 +00:00
"""
2019-01-26 18:59:07 +00:00
Verify ``bitfield`` against the ``committee_size` `.
2019-01-23 13:52:52 +00:00
"""
2019-01-26 18:59:07 +00:00
if len(bitfield) != (committee_size + 7) // 8:
2019-01-23 12:40:59 +00:00
return False
2019-02-05 11:51:06 -08:00
# Check `bitfield` is padded with zero bits only
2019-02-05 11:49:52 -08:00
for i in range(committee_size, len(bitfield) * 8):
2019-01-27 09:01:11 +00:00
if get_bitfield_bit(bitfield, i) == 0b1:
2019-01-23 12:40:59 +00:00
return False
return True
2018-12-05 06:41:11 -08:00
```
2019-03-26 13:40:19 -06:00
### `convert_to_indexed`
2018-12-06 19:02:23 -06:00
```python
2019-04-03 23:22:27 +04:00
def convert_to_indexed(state: BeaconState, attestation: Attestation) -> IndexedAttestation:
2019-01-23 13:52:52 +00:00
"""
2019-04-03 23:24:46 +04:00
Convert ``attestation` ` to (almost) indexed-verifiable form.
2019-01-23 13:52:52 +00:00
"""
2019-04-18 14:20:34 +10:00
attesting_indices = get_attesting_indices(state, attestation.data, attestation.aggregation_bitfield)
custody_bit_1_indices = get_attesting_indices(state, attestation.data, attestation.custody_bitfield)
2019-03-28 11:26:04 -06:00
custody_bit_0_indices = [index for index in attesting_indices if index not in custody_bit_1_indices]
2018-12-07 20:09:49 +00:00
2019-03-26 13:40:19 -06:00
return IndexedAttestation(
2019-03-28 11:26:04 -06:00
custody_bit_0_indices=custody_bit_0_indices,
custody_bit_1_indices=custody_bit_1_indices,
2019-03-22 06:10:44 -05:00
data=attestation.data,
2019-04-18 21:17:29 -06:00
signature=attestation.signature,
2019-03-22 06:10:44 -05:00
)
```
2019-03-26 13:40:19 -06:00
### `verify_indexed_attestation`
2019-03-22 06:10:44 -05:00
```python
2019-03-26 13:40:19 -06:00
def verify_indexed_attestation(state: BeaconState, indexed_attestation: IndexedAttestation) -> bool:
2019-03-22 06:10:44 -05:00
"""
2019-03-26 13:40:19 -06:00
Verify validity of ``indexed_attestation` ` fields.
2019-03-22 06:10:44 -05:00
"""
2019-03-28 11:26:04 -06:00
custody_bit_0_indices = indexed_attestation.custody_bit_0_indices
custody_bit_1_indices = indexed_attestation.custody_bit_1_indices
2019-04-18 14:20:34 +10:00
# Ensure no duplicate indices across custody bits
2019-04-03 11:04:12 +11:00
assert len(set(custody_bit_0_indices).intersection(set(custody_bit_1_indices))) == 0
2019-03-28 11:26:04 -06:00
if len(custody_bit_1_indices) > 0: # [TO BE REMOVED IN PHASE 1]
2019-01-23 13:24:35 +00:00
return False
2019-04-18 14:20:34 +10:00
if not (1 < = len(custody_bit_0_indices) + len(custody_bit_1_indices) < = MAX_INDICES_PER_ATTESTATION):
2019-01-23 13:24:35 +00:00
return False
2019-03-28 11:26:04 -06:00
if custody_bit_0_indices != sorted(custody_bit_0_indices):
2019-01-23 12:40:59 +00:00
return False
2019-03-28 11:26:04 -06:00
if custody_bit_1_indices != sorted(custody_bit_1_indices):
return False
2019-01-23 13:24:35 +00:00
2019-02-06 01:11:00 +08:00
return bls_verify_multiple(
2018-12-23 05:48:11 -05:00
pubkeys=[
2019-01-23 13:24:35 +00:00
bls_aggregate_pubkeys([state.validator_registry[i].pubkey for i in custody_bit_0_indices]),
bls_aggregate_pubkeys([state.validator_registry[i].pubkey for i in custody_bit_1_indices]),
2018-12-23 05:48:11 -05:00
],
2019-02-07 22:09:41 +01:00
message_hashes=[
2019-03-26 13:40:19 -06:00
hash_tree_root(AttestationDataAndCustodyBit(data=indexed_attestation.data, custody_bit=0b0)),
hash_tree_root(AttestationDataAndCustodyBit(data=indexed_attestation.data, custody_bit=0b1)),
2018-12-23 05:48:11 -05:00
],
2019-04-17 10:16:01 +10:00
signature=indexed_attestation.signature,
2019-03-31 04:55:24 -05:00
domain=get_domain(state, DOMAIN_ATTESTATION, slot_to_epoch(indexed_attestation.data.slot)),
2018-12-13 02:55:28 +08:00
)
2018-12-06 19:02:23 -06:00
```
2019-01-28 12:15:43 -07:00
### `is_double_vote`
2018-12-13 12:09:39 -06:00
```python
def is_double_vote(attestation_data_1: AttestationData,
2019-01-30 17:25:39 +08:00
attestation_data_2: AttestationData) -> bool:
2018-12-13 12:17:39 -06:00
"""
2019-01-30 11:04:35 -08:00
Check if ``attestation_data_1`` and ``attestation_data_2` ` have the same target.
2018-12-13 12:17:39 -06:00
"""
2019-01-25 17:33:15 -07:00
target_epoch_1 = slot_to_epoch(attestation_data_1.slot)
target_epoch_2 = slot_to_epoch(attestation_data_2.slot)
2018-12-28 11:10:12 -06:00
return target_epoch_1 == target_epoch_2
2018-12-13 12:09:39 -06:00
```
2019-01-28 12:15:43 -07:00
### `is_surround_vote`
2018-12-13 12:09:39 -06:00
```python
def is_surround_vote(attestation_data_1: AttestationData,
2018-12-19 12:47:37 +01:00
attestation_data_2: AttestationData) -> bool:
2018-12-13 12:17:39 -06:00
"""
2019-01-30 11:03:13 -08:00
Check if ``attestation_data_1`` surrounds ``attestation_data_2` `.
2018-12-13 12:17:39 -06:00
"""
2019-03-12 10:17:34 +00:00
source_epoch_1 = attestation_data_1.source_epoch
source_epoch_2 = attestation_data_2.source_epoch
2019-01-25 17:33:15 -07:00
target_epoch_1 = slot_to_epoch(attestation_data_1.slot)
target_epoch_2 = slot_to_epoch(attestation_data_2.slot)
2019-01-30 06:46:43 -08:00
return source_epoch_1 < source_epoch_2 and target_epoch_2 < target_epoch_1
2018-12-13 12:09:39 -06:00
```
2019-01-28 12:15:43 -07:00
### `integer_squareroot`
2018-09-30 18:02:09 -05:00
2018-09-30 09:56:43 -05:00
```python
2018-11-30 14:28:33 +00:00
def integer_squareroot(n: int) -> int:
"""
2019-01-22 22:47:07 -06:00
The largest integer ``x`` such that ``x**2`` is less than or equal to ``n` `.
2018-11-30 14:28:33 +00:00
"""
2019-01-10 18:09:58 -06:00
assert n >= 0
2018-09-30 18:38:47 -05:00
x = n
y = (x + 1) // 2
while y < x:
x = y
y = (x + n // x) // 2
return x
2018-09-30 09:56:43 -05:00
```
2019-02-20 07:45:19 +00:00
### `get_delayed_activation_exit_epoch`
2019-01-22 22:47:07 -06:00
```python
2019-02-20 07:45:19 +00:00
def get_delayed_activation_exit_epoch(epoch: Epoch) -> Epoch:
2019-01-22 22:47:07 -06:00
"""
2019-02-20 07:45:19 +00:00
Return the epoch at which an activation or exit triggered in ``epoch` ` takes effect.
2019-01-22 22:47:07 -06:00
"""
2019-01-31 07:56:48 -08:00
return epoch + 1 + ACTIVATION_EXIT_DELAY
2019-01-22 22:47:07 -06:00
```
2019-04-13 22:14:05 -05:00
### `get_churn_limit`
2018-11-16 11:41:59 -05:00
```python
2019-04-13 22:14:05 -05:00
def get_churn_limit(state: BeaconState) -> int:
return max(
MIN_PER_EPOCH_CHURN_LIMIT,
2019-04-14 17:28:45 +10:00
len(get_active_validator_indices(state, get_current_epoch(state))) // CHURN_LIMIT_QUOTIENT
2019-02-28 21:07:10 -06:00
)
2019-04-14 16:49:17 +10:00
```
2019-03-12 11:07:20 -06:00
2019-01-28 12:15:43 -07:00
### `bls_verify`
2019-02-09 07:21:38 -08:00
2019-04-12 18:33:53 -05:00
`bls_verify` is a function for verifying a BLS signature, defined in the [BLS Signature spec ](../bls_signature.md#bls_verify ).
2018-12-10 13:55:11 -06:00
2019-01-28 12:15:43 -07:00
### `bls_verify_multiple`
2019-03-11 17:28:39 +01:00
2019-04-12 18:33:53 -05:00
`bls_verify_multiple` is a function for verifying a BLS signature constructed from multiple messages, defined in the [BLS Signature spec ](../bls_signature.md#bls_verify_multiple ).
2018-11-27 13:51:12 -06:00
2019-01-28 12:15:43 -07:00
### `bls_aggregate_pubkeys`
2018-12-14 09:39:14 -06:00
2019-04-12 18:33:53 -05:00
`bls_aggregate_pubkeys` is a function for aggregating multiple BLS public keys into a single aggregate key, defined in the [BLS Signature spec ](../bls_signature.md#bls_aggregate_pubkeys ).
2018-12-05 12:40:08 -06:00
2018-12-28 12:04:03 -06:00
### Routines for updating validator status
2018-10-01 17:19:16 -05:00
2018-12-31 15:14:14 +00:00
Note: All functions in this section mutate `state` .
2018-12-10 16:42:28 -06:00
2019-01-28 12:15:43 -07:00
#### `initiate_validator_exit`
2018-12-10 15:16:06 -06:00
```python
2019-01-19 15:58:24 +08:00
def initiate_validator_exit(state: BeaconState, index: ValidatorIndex) -> None:
2019-01-30 17:25:39 +08:00
"""
Initiate the validator of the given ``index` `.
Note that this function mutates ``state` `.
"""
2019-04-14 19:01:53 +10:00
# Return if validator already initiated exit
2018-12-13 13:45:08 -06:00
validator = state.validator_registry[index]
2019-04-14 09:13:53 +10:00
if validator.exit_epoch != FAR_FUTURE_EPOCH:
return
2018-10-11 14:14:48 -04:00
2019-04-14 19:01:53 +10:00
# Compute exit queue epoch
exit_epochs = [v.exit_epoch for v in state.validator_registry if v.exit_epoch != FAR_FUTURE_EPOCH]
2019-04-20 18:13:45 +10:00
exit_queue_epoch = max(exit_epochs + [get_delayed_activation_exit_epoch(get_current_epoch(state))])
2019-04-14 18:10:44 +10:00
exit_queue_churn = len([v for v in state.validator_registry if v.exit_epoch == exit_queue_epoch])
2019-04-14 18:50:05 +10:00
if exit_queue_churn >= get_churn_limit(state):
2019-04-14 18:10:44 +10:00
exit_queue_epoch += 1
2018-12-11 09:49:50 -06:00
2019-04-14 09:13:53 +10:00
# Set validator exit epoch and withdrawable epoch
2019-04-14 18:10:44 +10:00
validator.exit_epoch = exit_queue_epoch
2019-04-14 09:13:53 +10:00
validator.withdrawable_epoch = validator.exit_epoch + MIN_VALIDATOR_WITHDRAWABILITY_DELAY
2018-10-11 14:14:48 -04:00
```
2019-02-12 12:24:19 +00:00
#### `slash_validator`
2019-01-28 12:15:43 -07:00
2018-12-28 12:04:03 -06:00
```python
2019-03-25 14:49:35 +00:00
def slash_validator(state: BeaconState, slashed_index: ValidatorIndex, whistleblower_index: ValidatorIndex=None) -> None:
2019-01-30 17:25:39 +08:00
"""
2019-03-25 14:30:59 +00:00
Slash the validator with index ``slashed_index` `.
2019-01-30 17:25:39 +08:00
Note that this function mutates ``state` `.
"""
2019-04-20 15:17:33 +10:00
current_epoch = get_current_epoch(state)
2019-03-29 15:26:26 +08:00
initiate_validator_exit(state, slashed_index)
2019-03-25 14:30:59 +00:00
state.validator_registry[slashed_index].slashed = True
2019-04-22 16:34:50 +10:00
state.validator_registry[slashed_index].withdrawable_epoch = current_epoch + LATEST_SLASHED_EXIT_LENGTH
slashed_balance = state.validator_registry[slashed_index].effective_balance
2019-04-20 15:17:33 +10:00
state.latest_slashed_balances[current_epoch % LATEST_SLASHED_EXIT_LENGTH] += slashed_balance
2019-03-25 14:30:59 +00:00
2019-04-17 12:32:50 +10:00
proposer_index = get_beacon_proposer_index(state)
2019-03-25 14:49:35 +00:00
if whistleblower_index is None:
whistleblower_index = proposer_index
2019-03-25 14:30:59 +00:00
whistleblowing_reward = slashed_balance // WHISTLEBLOWING_REWARD_QUOTIENT
proposer_reward = whistleblowing_reward // PROPOSER_REWARD_QUOTIENT
increase_balance(state, proposer_index, proposer_reward)
2019-03-25 14:49:35 +00:00
increase_balance(state, whistleblower_index, whistleblowing_reward - proposer_reward)
2019-03-25 14:54:43 +00:00
decrease_balance(state, slashed_index, whistleblowing_reward)
2018-12-28 12:04:03 -06:00
```
2019-02-12 12:24:19 +00:00
## On genesis
2018-09-20 13:20:49 +08:00
2019-03-03 05:04:28 -06:00
When enough full deposits have been made to the deposit contract, an `Eth2Genesis` log is emitted. Construct a corresponding `genesis_state` and `genesis_block` as follows:
* Let `genesis_validator_deposits` be the list of deposits, ordered chronologically, up to and including the deposit that triggered the `Eth2Genesis` log.
* Let `genesis_time` be the timestamp specified in the `Eth2Genesis` log.
* Let `genesis_eth1_data` be the `Eth1Data` object where:
* `genesis_eth1_data.deposit_root` is the `deposit_root` contained in the `Eth2Genesis` log.
2019-03-11 23:30:08 -05:00
* `genesis_eth1_data.deposit_count` is the `deposit_count` contained in the `Eth2Genesis` log.
2019-03-03 05:04:28 -06:00
* `genesis_eth1_data.block_hash` is the hash of the Ethereum 1.0 block that emitted the `Eth2Genesis` log.
* Let `genesis_state = get_genesis_beacon_state(genesis_validator_deposits, genesis_time, genesis_eth1_data)` .
2019-04-19 18:26:54 +10:00
* Let `genesis_block = BeaconBlock(state_root=hash_tree_root(genesis_state))` .
2018-11-16 11:41:59 -05:00
2018-10-02 11:20:07 -04:00
```python
2019-02-12 12:24:19 +00:00
def get_genesis_beacon_state(genesis_validator_deposits: List[Deposit],
2018-12-13 08:08:51 -05:00
genesis_time: int,
2019-03-03 05:04:28 -06:00
genesis_eth1_data: Eth1Data) -> BeaconState:
2019-01-30 17:25:39 +08:00
"""
2019-02-12 12:24:19 +00:00
Get the genesis ``BeaconState` `.
2019-01-30 17:25:39 +08:00
"""
2019-04-19 18:26:54 +10:00
state = BeaconState(genesis_time=genesis_time, latest_eth1_data=genesis_eth1_data)
2018-10-09 15:33:22 +08:00
2019-02-12 12:24:19 +00:00
# Process genesis deposits
for deposit in genesis_validator_deposits:
2019-02-15 00:23:03 +00:00
process_deposit(state, deposit)
2018-12-10 15:16:06 -06:00
2019-02-12 12:24:19 +00:00
# Process genesis activations
2019-04-25 16:03:02 +08:00
for validator in state.validator_registry:
2019-04-24 14:46:28 +10:00
if validator.effective_balance >= MAX_EFFECTIVE_BALANCE:
2019-04-22 15:12:30 +10:00
validator.activation_eligibility_epoch = GENESIS_EPOCH
validator.activation_epoch = GENESIS_EPOCH
2018-12-10 15:16:06 -06:00
2019-04-14 17:28:45 +10:00
genesis_active_index_root = hash_tree_root(get_active_validator_indices(state, GENESIS_EPOCH))
2019-01-31 07:56:48 -08:00
for index in range(LATEST_ACTIVE_INDEX_ROOTS_LENGTH):
state.latest_active_index_roots[index] = genesis_active_index_root
2018-12-10 15:16:06 -06:00
2018-11-15 08:20:25 -05:00
return state
2018-10-02 11:20:07 -04:00
```
2018-09-20 13:20:49 +08:00
2019-01-28 12:15:43 -07:00
## Beacon chain state transition function
2018-12-28 12:04:03 -06:00
2019-03-27 14:00:28 -05:00
We now define the state transition function. At a high level, the state transition is made up of four parts:
2018-12-28 12:04:03 -06:00
2019-03-08 09:48:46 -07:00
1. State caching, which happens at the start of every slot.
2019-03-07 12:05:34 -07:00
2. The per-epoch transitions, which happens at the start of the first slot of every epoch.
3. The per-slot transitions, which happens at every slot.
4. The per-block transitions, which happens at every block.
2018-12-28 12:04:03 -06:00
2019-03-07 13:36:22 -07:00
Transition section notes:
2019-04-13 09:56:19 +10:00
* The state caching caches the state root of the previous slot and updates block and state roots records.
2019-03-08 09:48:46 -07:00
* The per-epoch transitions focus on the [validator ](#dfn-validator ) registry, including adjusting balances and activating and exiting [validators ](#dfn-validator ), as well as processing crosslinks and managing block justification/finalization.
2019-04-13 09:56:19 +10:00
* The per-slot transitions focus on the slot counter.
2019-03-07 13:36:22 -07:00
* The per-block transitions generally focus on verifying aggregate signatures and saving temporary records relating to the per-block activity in the `BeaconState` .
2018-12-28 12:04:03 -06:00
2019-02-11 14:30:36 -07:00
Beacon blocks that trigger unhandled Python exceptions (e.g. out-of-range list accesses) and failed `assert` s during the state transition are considered invalid.
2019-04-17 14:06:28 +10:00
Note: If there are skipped slots between a block and its parent block, run the steps in the [state-root ](#state-caching ), [per-epoch ](#per-epoch-processing ), and [per-slot ](#per-slot-processing ) sections once for each skipped slot and then once for the slot containing the new block.
2018-12-28 12:04:03 -06:00
2019-03-08 09:48:46 -07:00
### State caching
2018-09-20 13:20:49 +08:00
2019-03-03 05:04:28 -06:00
At every `slot > GENESIS_SLOT` run the following function:
2018-12-12 08:02:50 -05:00
2019-03-03 05:04:28 -06:00
```python
2019-03-08 09:48:46 -07:00
def cache_state(state: BeaconState) -> None:
2019-04-18 11:09:30 +10:00
# Cache latest known state root (for previous slot)
latest_state_root = hash_tree_root(state)
state.latest_state_roots[state.slot % SLOTS_PER_HISTORICAL_ROOT] = latest_state_root
2019-03-08 09:48:46 -07:00
2019-04-18 11:09:30 +10:00
# Store latest known state root (for previous slot) in latest_block_header if it is empty
2019-03-08 09:48:46 -07:00
if state.latest_block_header.state_root == ZERO_HASH:
2019-04-18 11:09:30 +10:00
state.latest_block_header.state_root = latest_state_root
2019-03-08 09:48:46 -07:00
2019-04-18 11:09:30 +10:00
# Cache latest known block root (for previous slot)
2019-04-17 09:38:10 -06:00
latest_block_root = signing_root(state.latest_block_header)
state.latest_block_roots[state.slot % SLOTS_PER_HISTORICAL_ROOT] = latest_block_root
2019-03-03 05:04:28 -06:00
```
2018-09-20 13:20:49 +08:00
2019-03-07 12:05:34 -07:00
### Per-epoch processing
2018-12-10 10:11:41 +00:00
2019-03-07 12:05:34 -07:00
The steps below happen when `state.slot > GENESIS_SLOT and (state.slot + 1) % SLOTS_PER_EPOCH == 0` .
2018-12-10 10:11:41 +00:00
2019-03-07 12:05:34 -07:00
#### Helper functions
2019-04-17 12:32:50 +10:00
We define epoch transition helper functions:
2018-09-20 13:20:49 +08:00
2019-03-03 05:04:28 -06:00
```python
2019-04-22 16:13:46 +10:00
def get_total_active_balance(state: BeaconState) -> Gwei:
2019-04-14 17:28:45 +10:00
return get_total_balance(state, get_active_validator_indices(state, get_current_epoch(state)))
2019-03-03 05:04:28 -06:00
```
2018-09-20 13:20:49 +08:00
2019-03-07 12:05:34 -07:00
```python
2019-04-20 15:17:33 +10:00
def get_matching_source_attestations(state: BeaconState, epoch: Epoch) -> List[PendingAttestation]:
2019-04-24 20:57:31 +10:00
assert epoch in (get_current_epoch(state), get_previous_epoch(state))
2019-04-20 15:17:33 +10:00
return state.current_epoch_attestations if epoch == get_current_epoch(state) else state.previous_epoch_attestations
2019-03-07 12:05:34 -07:00
```
2018-10-11 13:57:30 -04:00
2019-03-03 05:04:28 -06:00
```python
2019-04-20 15:17:33 +10:00
def get_matching_target_attestations(state: BeaconState, epoch: Epoch) -> List[PendingAttestation]:
2019-03-07 12:05:34 -07:00
return [
2019-04-20 15:17:33 +10:00
a for a in get_matching_source_attestations(state, epoch)
2019-04-24 14:23:51 +10:00
if a.data.target_root == get_block_root(state, epoch)
2019-03-07 12:05:34 -07:00
]
2019-03-03 05:04:28 -06:00
```
2018-11-30 14:28:33 +00:00
2019-03-07 12:05:34 -07:00
```python
2019-04-20 15:17:33 +10:00
def get_matching_head_attestations(state: BeaconState, epoch: Epoch) -> List[PendingAttestation]:
2019-03-07 12:05:34 -07:00
return [
2019-04-20 15:17:33 +10:00
a for a in get_matching_source_attestations(state, epoch)
2019-04-24 14:23:51 +10:00
if a.data.beacon_block_root == get_block_root_at_slot(state, a.data.slot)
2019-03-07 12:05:34 -07:00
]
```
2018-11-16 07:54:03 -05:00
2019-03-07 12:05:34 -07:00
```python
2019-04-17 14:06:28 +10:00
def get_unslashed_attesting_indices(state: BeaconState, attestations: List[PendingAttestation]) -> List[ValidatorIndex]:
output = set()
for a in attestations:
2019-04-19 09:37:11 +08:00
output = output.union(get_attesting_indices(state, a.data, a.aggregation_bitfield))
2019-04-17 14:06:28 +10:00
return sorted(filter(lambda index: not state.validator_registry[index].slashed, list(output)))
2019-03-07 12:05:34 -07:00
```
2018-11-16 10:48:57 -05:00
2019-04-17 14:06:28 +10:00
```python
2019-04-22 16:13:46 +10:00
def get_attesting_balance(state: BeaconState, attestations: List[PendingAttestation]) -> Gwei:
return get_total_balance(state, get_unslashed_attesting_indices(state, attestations))
2019-04-17 14:06:28 +10:00
```
2018-11-16 07:54:03 -05:00
2019-03-07 12:05:34 -07:00
```python
2019-04-18 19:33:38 +10:00
def get_crosslink_from_attestation_data(state: BeaconState, data: AttestationData) -> Crosslink:
return Crosslink(
epoch=min(slot_to_epoch(data.slot), state.current_crosslinks[data.shard].epoch + MAX_CROSSLINK_EPOCHS),
previous_crosslink_root=data.previous_crosslink_root,
2019-04-18 11:16:50 -06:00
crosslink_data_root=data.crosslink_data_root,
2019-04-18 19:33:38 +10:00
)
```
2019-03-07 12:05:34 -07:00
```python
2019-04-24 14:23:51 +10:00
def get_winning_crosslink_and_attesting_indices(state: BeaconState, shard: Shard, epoch: Epoch) -> Tuple[Crosslink, List[ValidatorIndex]]:
2019-04-24 20:57:31 +10:00
shard_attestations = [a for a in get_matching_source_attestations(state, epoch) if a.data.shard == shard]
2019-04-18 19:33:38 +10:00
shard_crosslinks = [get_crosslink_from_attestation_data(state, a.data) for a in shard_attestations]
2019-04-18 11:16:50 -06:00
candidate_crosslinks = [
c for c in shard_crosslinks
if hash_tree_root(state.current_crosslinks[shard]) in (c.previous_crosslink_root, hash_tree_root(c))
2019-04-18 18:53:22 +10:00
]
2019-04-18 14:33:45 +10:00
if len(candidate_crosslinks) == 0:
2019-04-25 16:03:02 +08:00
return Crosslink(epoch=GENESIS_EPOCH), []
2019-03-07 12:05:34 -07:00
2019-04-18 19:33:38 +10:00
def get_attestations_for(crosslink: Crosslink) -> List[PendingAttestation]:
return [a for a in shard_attestations if get_crosslink_from_attestation_data(state, a.data) == crosslink]
2019-04-18 14:33:45 +10:00
# Winning crosslink has the crosslink data root with the most balance voting for it (ties broken lexicographically)
2019-04-18 19:33:38 +10:00
winning_crosslink = max(candidate_crosslinks, key=lambda crosslink: (
get_attesting_balance(state, get_attestations_for(crosslink)), crosslink.crosslink_data_root
))
2019-03-07 12:05:34 -07:00
2019-04-18 19:33:38 +10:00
return winning_crosslink, get_unslashed_attesting_indices(state, get_attestations_for(winning_crosslink))
2019-03-07 12:05:34 -07:00
```
2018-11-16 07:54:03 -05:00
2019-02-28 20:54:59 -06:00
```python
2019-04-17 11:35:37 +08:00
def get_earliest_attestation(state: BeaconState, attestations: List[PendingAttestation], index: ValidatorIndex) -> PendingAttestation:
2019-03-07 12:05:34 -07:00
return min([
2019-04-18 14:20:34 +10:00
a for a in attestations if index in get_attesting_indices(state, a.data, a.aggregation_bitfield)
2019-03-07 12:05:34 -07:00
], key=lambda a: a.inclusion_slot)
2019-02-28 20:54:59 -06:00
```
2018-11-16 07:54:03 -05:00
2019-04-18 19:08:34 +10:00
#### Justification and finalization
2019-03-07 12:05:34 -07:00
Run the following function:
2018-11-16 07:54:03 -05:00
2019-02-28 20:54:59 -06:00
```python
2019-04-18 19:08:34 +10:00
def process_justification_and_finalization(state: BeaconState) -> None:
2019-04-17 09:38:10 -06:00
if get_current_epoch(state) < = GENESIS_EPOCH + 1:
2019-04-11 22:28:42 +10:00
return
2019-04-20 16:10:25 +10:00
previous_epoch = get_previous_epoch(state)
current_epoch = get_current_epoch(state)
2019-04-18 13:31:19 +10:00
old_previous_justified_epoch = state.previous_justified_epoch
old_current_justified_epoch = state.current_justified_epoch
2019-03-12 12:32:11 -06:00
2019-04-16 14:59:35 +10:00
# Process justifications
state.previous_justified_epoch = state.current_justified_epoch
state.previous_justified_root = state.current_justified_root
2019-04-12 09:12:37 +10:00
state.justification_bitfield = (state.justification_bitfield < < 1 ) % 2 * * 64
2019-04-22 16:13:46 +10:00
previous_epoch_matching_target_balance = get_attesting_balance(state, get_matching_target_attestations(state, previous_epoch))
if previous_epoch_matching_target_balance * 3 >= get_total_active_balance(state) * 2:
2019-04-24 11:31:24 -06:00
state.current_justified_epoch = previous_epoch
2019-04-24 14:23:51 +10:00
state.current_justified_root = get_block_root(state, state.current_justified_epoch)
2019-04-16 14:59:35 +10:00
state.justification_bitfield |= (1 < < 1 )
2019-04-22 16:13:46 +10:00
current_epoch_matching_target_balance = get_attesting_balance(state, get_matching_target_attestations(state, current_epoch))
if current_epoch_matching_target_balance * 3 >= get_total_active_balance(state) * 2:
2019-04-24 11:31:24 -06:00
state.current_justified_epoch = current_epoch
2019-04-24 14:23:51 +10:00
state.current_justified_root = get_block_root(state, state.current_justified_epoch)
2019-04-16 14:59:35 +10:00
state.justification_bitfield |= (1 < < 0 )
2019-03-07 12:05:34 -07:00
# Process finalizations
bitfield = state.justification_bitfield
2019-04-16 14:59:35 +10:00
# The 2nd/3rd/4th most recent epochs are justified, the 2nd using the 4th as source
2019-04-18 13:31:19 +10:00
if (bitfield >> 1) % 8 == 0b111 and old_previous_justified_epoch == current_epoch - 3:
state.finalized_epoch = old_previous_justified_epoch
2019-04-24 14:23:51 +10:00
state.finalized_root = get_block_root(state, state.finalized_epoch)
2019-04-16 14:59:35 +10:00
# The 2nd/3rd most recent epochs are justified, the 2nd using the 3rd as source
2019-04-18 13:31:19 +10:00
if (bitfield >> 1) % 4 == 0b11 and old_previous_justified_epoch == current_epoch - 2:
state.finalized_epoch = old_previous_justified_epoch
2019-04-24 14:23:51 +10:00
state.finalized_root = get_block_root(state, state.finalized_epoch)
2019-04-16 14:59:35 +10:00
# The 1st/2nd/3rd most recent epochs are justified, the 1st using the 3rd as source
2019-04-18 13:31:19 +10:00
if (bitfield >> 0) % 8 == 0b111 and old_current_justified_epoch == current_epoch - 2:
state.finalized_epoch = old_current_justified_epoch
2019-04-24 14:23:51 +10:00
state.finalized_root = get_block_root(state, state.finalized_epoch)
2019-04-16 14:59:35 +10:00
# The 1st/2nd most recent epochs are justified, the 1st using the 2nd as source
2019-04-18 13:31:19 +10:00
if (bitfield >> 0) % 4 == 0b11 and old_current_justified_epoch == current_epoch - 1:
state.finalized_epoch = old_current_justified_epoch
2019-04-24 14:23:51 +10:00
state.finalized_root = get_block_root(state, state.finalized_epoch)
2019-02-28 20:54:59 -06:00
```
2018-09-20 13:20:49 +08:00
2019-03-07 12:05:34 -07:00
#### Crosslinks
2018-11-16 07:54:03 -05:00
2019-03-07 12:05:34 -07:00
Run the following function:
2018-11-20 02:40:04 -05:00
2019-03-07 12:05:34 -07:00
```python
def process_crosslinks(state: BeaconState) -> None:
2019-04-18 11:16:50 -06:00
state.previous_crosslinks = [c for c in state.current_crosslinks]
2019-04-18 16:53:02 +08:00
previous_epoch = get_previous_epoch(state)
next_epoch = get_current_epoch(state) + 1
2019-03-07 12:05:34 -07:00
for slot in range(get_epoch_start_slot(previous_epoch), get_epoch_start_slot(next_epoch)):
2019-04-20 15:17:33 +10:00
epoch = slot_to_epoch(slot)
2019-03-07 12:05:34 -07:00
for crosslink_committee, shard in get_crosslink_committees_at_slot(state, slot):
2019-04-24 14:23:51 +10:00
winning_crosslink, attesting_indices = get_winning_crosslink_and_attesting_indices(state, shard, epoch)
2019-04-18 18:53:22 +10:00
if 3 * get_total_balance(state, attesting_indices) >= 2 * get_total_balance(state, crosslink_committee):
2019-04-18 14:20:34 +10:00
state.current_crosslinks[shard] = winning_crosslink
2019-03-01 07:59:52 -06:00
```
2018-12-05 11:22:15 +00:00
2019-01-28 12:15:43 -07:00
#### Rewards and penalties
2018-12-12 09:07:44 -06:00
2019-04-18 19:08:34 +10:00
First, we define additional helpers:
2018-12-12 09:14:44 -06:00
2019-03-01 07:59:52 -06:00
```python
2019-04-22 16:13:46 +10:00
def get_base_reward(state: BeaconState, index: ValidatorIndex) -> Gwei:
adjusted_quotient = integer_squareroot(get_total_active_balance(state)) // BASE_REWARD_QUOTIENT
2019-04-20 16:10:25 +10:00
if adjusted_quotient == 0:
2019-03-04 19:05:00 -07:00
return 0
2019-04-22 16:13:46 +10:00
return state.validator_registry[index].effective_balance // adjusted_quotient // BASE_REWARDS_PER_EPOCH
2019-03-01 07:59:52 -06:00
```
2019-04-17 12:32:50 +10:00
```python
2019-04-20 15:17:33 +10:00
def get_attestation_deltas(state: BeaconState) -> Tuple[List[Gwei], List[Gwei]]:
2019-04-20 16:10:25 +10:00
previous_epoch = get_previous_epoch(state)
2019-04-22 16:13:46 +10:00
total_balance = get_total_active_balance(state)
2019-04-25 16:03:02 +08:00
rewards = [0 for _ in range(len(state.validator_registry))]
penalties = [0 for _ in range(len(state.validator_registry))]
2019-04-24 14:23:51 +10:00
eligible_validator_indices = [
index for index, v in enumerate(state.validator_registry)
if is_active_validator(v, previous_epoch) or (v.slashed and previous_epoch + 1 < v.withdrawable_epoch )
2019-03-21 15:11:05 -05:00
]
2019-04-24 14:23:51 +10:00
# Micro-incentives for matching FFG source, FFG target, and head
matching_source_attestations = get_matching_source_attestations(state, previous_epoch)
matching_target_attestations = get_matching_target_attestations(state, previous_epoch)
matching_head_attestations = get_matching_head_attestations(state, previous_epoch)
for attestations in (matching_source_attestations, matching_target_attestations, matching_head_attestations):
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 // total_balance
2019-04-17 14:06:28 +10:00
else:
2019-04-24 14:23:51 +10:00
penalties[index] += get_base_reward(state, index)
# Proposer and inclusion delay micro-rewards
2019-04-24 15:32:43 +10:00
for index in get_unslashed_attesting_indices(state, matching_source_attestations):
earliest_attestation = get_earliest_attestation(state, matching_source_attestations, index)
rewards[earliest_attestation.proposer_index] += get_base_reward(state, index) // PROPOSER_REWARD_QUOTIENT
inclusion_delay = earliest_attestation.inclusion_slot - earliest_attestation.data.slot
rewards[index] += get_base_reward(state, index) * MIN_ATTESTATION_INCLUSION_DELAY // inclusion_delay
2019-04-24 14:23:51 +10:00
# Inactivity penalty
finality_delay = previous_epoch - state.finalized_epoch
if finality_delay > MIN_EPOCHS_TO_INACTIVITY_PENALTY:
matching_target_attesting_indices = get_unslashed_attesting_indices(state, matching_target_attestations)
for index in eligible_validator_indices:
penalties[index] += BASE_REWARDS_PER_EPOCH * get_base_reward(state, index)
if index not in matching_target_attesting_indices:
2019-04-22 16:13:46 +10:00
penalties[index] += state.validator_registry[index].effective_balance * finality_delay // INACTIVITY_PENALTY_QUOTIENT
2019-04-17 14:06:28 +10:00
2019-04-25 15:37:05 +08:00
return rewards, penalties
2019-03-01 07:59:52 -06:00
```
2018-09-20 13:20:49 +08:00
2019-03-01 07:59:52 -06:00
```python
2019-03-04 09:45:55 -07:00
def get_crosslink_deltas(state: BeaconState) -> Tuple[List[Gwei], List[Gwei]]:
2019-04-25 16:03:02 +08:00
rewards = [0 for _ in range(len(state.validator_registry))]
penalties = [0 for _ in range(len(state.validator_registry))]
2019-04-18 14:20:34 +10:00
for slot in range(get_epoch_start_slot(get_previous_epoch(state)), get_epoch_start_slot(get_current_epoch(state))):
2019-04-20 15:17:33 +10:00
epoch = slot_to_epoch(slot)
2019-03-01 07:59:52 -06:00
for crosslink_committee, shard in get_crosslink_committees_at_slot(state, slot):
2019-04-24 14:23:51 +10:00
winning_crosslink, attesting_indices = get_winning_crosslink_and_attesting_indices(state, shard, epoch)
2019-04-18 14:20:34 +10:00
attesting_balance = get_total_balance(state, attesting_indices)
committee_balance = get_total_balance(state, crosslink_committee)
2019-03-01 07:59:52 -06:00
for index in crosslink_committee:
2019-04-22 16:13:46 +10:00
base_reward = get_base_reward(state, index)
2019-04-18 14:20:34 +10:00
if index in attesting_indices:
2019-04-19 09:37:11 +08:00
rewards[index] += base_reward * attesting_balance // committee_balance
2019-03-01 07:59:52 -06:00
else:
2019-04-17 14:06:28 +10:00
penalties[index] += base_reward
2019-04-25 15:37:05 +08:00
return rewards, penalties
2019-03-01 07:59:52 -06:00
```
2018-10-01 17:19:16 -05:00
2019-04-18 19:08:34 +10:00
Run the following function:
2018-09-20 13:20:49 +08:00
2019-03-01 07:59:52 -06:00
```python
2019-04-18 19:08:34 +10:00
def process_rewards_and_penalties(state: BeaconState) -> None:
2019-04-11 22:28:42 +10:00
if get_current_epoch(state) == GENESIS_EPOCH:
return
2019-04-20 15:17:33 +10:00
rewards1, penalties1 = get_attestation_deltas(state)
2019-03-21 16:29:14 -05:00
rewards2, penalties2 = get_crosslink_deltas(state)
2019-03-01 07:59:52 -06:00
for i in range(len(state.validator_registry)):
2019-04-17 12:32:50 +10:00
increase_balance(state, i, rewards1[i] + rewards2[i])
decrease_balance(state, i, penalties1[i] + penalties2[i])
2019-03-01 07:59:52 -06:00
```
2018-11-23 13:54:11 +09:00
2019-04-18 19:08:34 +10:00
#### Registry updates
2018-12-13 13:45:08 -06:00
2019-04-18 19:08:34 +10:00
Run the following function:
2018-09-20 13:20:49 +08:00
2018-12-05 11:22:15 +00:00
```python
2019-04-18 19:08:34 +10:00
def process_registry_updates(state: BeaconState) -> None:
# Process activation eligibility and ejections
2018-12-10 16:30:25 -06:00
for index, validator in enumerate(state.validator_registry):
2019-04-24 14:46:28 +10:00
if validator.activation_eligibility_epoch == FAR_FUTURE_EPOCH and validator.effective_balance >= MAX_EFFECTIVE_BALANCE:
2019-04-14 22:55:38 +10:00
validator.activation_eligibility_epoch = get_current_epoch(state)
2018-12-04 19:49:26 +00:00
2019-04-20 16:32:41 +10:00
if is_active_validator(validator, get_current_epoch(state)) and validator.effective_balance < = EJECTION_BALANCE:
2019-03-15 13:05:46 +00:00
initiate_validator_exit(state, index)
2019-03-07 12:05:34 -07:00
2019-04-22 15:16:34 +10:00
# Queue validators eligible for activation and not dequeued for activation prior to finalized epoch
2019-04-07 01:24:50 +11:00
activation_queue = sorted([
2019-04-15 07:03:47 +10:00
index for index, validator in enumerate(state.validator_registry) if
2019-04-06 21:07:03 +11:00
validator.activation_eligibility_epoch != FAR_FUTURE_EPOCH and
2019-04-07 08:45:43 +10:00
validator.activation_epoch >= get_delayed_activation_exit_epoch(state.finalized_epoch)
2019-04-06 20:45:11 +11:00
], key=lambda index: state.validator_registry[index].activation_eligibility_epoch)
2019-04-22 15:12:30 +10:00
# Dequeued validators for activation up to churn limit (without resetting activation epoch)
2019-04-13 22:14:05 -05:00
for index in activation_queue[:get_churn_limit(state)]:
2019-04-26 14:43:05 +08:00
validator = state.validator_registry[index]
2019-04-22 20:49:07 +10:00
if validator.activation_epoch == FAR_FUTURE_EPOCH:
2019-04-22 15:12:30 +10:00
validator.activation_epoch = get_delayed_activation_exit_epoch(get_current_epoch(state))
2019-03-07 12:05:34 -07:00
```
2019-04-06 21:07:03 +11:00
#### Slashings
2019-03-07 12:05:34 -07:00
2019-04-18 19:08:34 +10:00
Run the following function:
2019-03-07 12:05:34 -07:00
```python
def process_slashings(state: BeaconState) -> None:
current_epoch = get_current_epoch(state)
2019-04-14 17:28:45 +10:00
active_validator_indices = get_active_validator_indices(state, current_epoch)
2019-03-07 12:05:34 -07:00
total_balance = get_total_balance(state, active_validator_indices)
# Compute `total_penalties`
2019-03-07 23:13:06 +01:00
total_at_start = state.latest_slashed_balances[(current_epoch + 1) % LATEST_SLASHED_EXIT_LENGTH]
total_at_end = state.latest_slashed_balances[current_epoch % LATEST_SLASHED_EXIT_LENGTH]
2019-03-07 12:05:34 -07:00
total_penalties = total_at_end - total_at_start
for index, validator in enumerate(state.validator_registry):
if validator.slashed and current_epoch == validator.withdrawable_epoch - LATEST_SLASHED_EXIT_LENGTH // 2:
penalty = max(
2019-04-22 16:13:46 +10:00
validator.effective_balance * min(total_penalties * 3, total_balance) // total_balance,
2019-04-22 16:34:50 +10:00
validator.effective_balance // MIN_SLASHING_PENALTY_QUOTIENT
2019-03-07 12:05:34 -07:00
)
2019-03-07 01:38:03 -06:00
decrease_balance(state, index, penalty)
2019-03-07 12:05:34 -07:00
```
#### Final updates
Run the following function:
```python
2019-04-18 19:08:34 +10:00
def process_final_updates(state: BeaconState) -> None:
2019-03-07 12:05:34 -07:00
current_epoch = get_current_epoch(state)
next_epoch = current_epoch + 1
2019-04-17 14:30:03 +10:00
# Reset eth1 data votes
2019-04-24 13:15:06 -06:00
if (state.slot + 1) % SLOTS_PER_ETH1_VOTING_PERIOD == 0:
2019-04-17 14:30:03 +10:00
state.eth1_data_votes = []
2019-04-22 16:13:46 +10:00
# Update effective balances with hysteresis
2019-04-20 15:17:33 +10:00
for index, validator in enumerate(state.validator_registry):
2019-04-26 18:46:35 +04:00
balance = state.balances[index]
2019-04-22 16:13:46 +10:00
HALF_INCREMENT = EFFECTIVE_BALANCE_INCREMENT // 2
if balance < validator.effective_balance or validator . effective_balance + 3 * HALF_INCREMENT < balance:
2019-04-26 18:46:35 +04:00
validator.effective_balance = min(balance - balance % EFFECTIVE_BALANCE_INCREMENT, MAX_EFFECTIVE_BALANCE)
2019-04-18 19:08:34 +10:00
# Update start shard
state.latest_start_shard = (state.latest_start_shard + get_shard_delta(state, current_epoch)) % SHARD_COUNT
2019-03-07 12:05:34 -07:00
# Set active index root
index_root_position = (next_epoch + ACTIVATION_EXIT_DELAY) % LATEST_ACTIVE_INDEX_ROOTS_LENGTH
state.latest_active_index_roots[index_root_position] = hash_tree_root(
2019-04-14 17:28:45 +10:00
get_active_validator_indices(state, next_epoch + ACTIVATION_EXIT_DELAY)
2019-03-07 12:05:34 -07:00
)
# Set total slashed balances
state.latest_slashed_balances[next_epoch % LATEST_SLASHED_EXIT_LENGTH] = (
state.latest_slashed_balances[current_epoch % LATEST_SLASHED_EXIT_LENGTH]
)
# Set randao mix
state.latest_randao_mixes[next_epoch % LATEST_RANDAO_MIXES_LENGTH] = get_randao_mix(state, current_epoch)
# Set historical root accumulator
if next_epoch % (SLOTS_PER_HISTORICAL_ROOT // SLOTS_PER_EPOCH) == 0:
2019-03-08 18:13:05 +01:00
historical_batch = HistoricalBatch(
block_roots=state.latest_block_roots,
state_roots=state.latest_state_roots,
)
state.historical_roots.append(hash_tree_root(historical_batch))
2019-03-07 12:05:34 -07:00
# Rotate current/previous epoch attestations
state.previous_epoch_attestations = state.current_epoch_attestations
state.current_epoch_attestations = []
```
### Per-slot processing
At every `slot > GENESIS_SLOT` run the following function:
```python
def advance_slot(state: BeaconState) -> None:
2019-03-08 09:08:30 +01:00
state.slot += 1
2019-03-07 12:05:34 -07:00
```
### Per-block processing
For every `block` except the genesis block, run `process_block_header(state, block)` , `process_randao(state, block)` and `process_eth1_data(state, block)` .
#### Block header
```python
def process_block_header(state: BeaconState, block: BeaconBlock) -> None:
# Verify that the slots match
assert block.slot == state.slot
# Verify that the parent matches
2019-04-08 09:51:13 +08:00
assert block.previous_block_root == signing_root(state.latest_block_header)
2019-03-07 12:05:34 -07:00
# Save current block as the new latest block
2019-04-19 18:26:54 +10:00
state.latest_block_header = BeaconBlockHeader(
slot=block.slot,
previous_block_root=block.previous_block_root,
block_body_root=hash_tree_root(block.body),
)
2019-03-17 11:48:47 +00:00
# Verify proposer is not slashed
2019-04-17 12:32:50 +10:00
proposer = state.validator_registry[get_beacon_proposer_index(state)]
2019-03-11 22:07:34 +00:00
assert not proposer.slashed
# Verify proposer signature
2019-04-18 21:17:29 -06:00
assert bls_verify(proposer.pubkey, signing_root(block), block.signature, get_domain(state, DOMAIN_BEACON_PROPOSER))
2019-03-07 12:05:34 -07:00
```
#### RANDAO
```python
def process_randao(state: BeaconState, block: BeaconBlock) -> None:
2019-04-17 12:32:50 +10:00
proposer = state.validator_registry[get_beacon_proposer_index(state)]
2019-03-07 12:05:34 -07:00
# Verify that the provided randao value is valid
2019-03-31 04:55:24 -05:00
assert bls_verify(proposer.pubkey, hash_tree_root(get_current_epoch(state)), block.body.randao_reveal, get_domain(state, DOMAIN_RANDAO))
2019-03-07 12:05:34 -07:00
# Mix it in
state.latest_randao_mixes[get_current_epoch(state) % LATEST_RANDAO_MIXES_LENGTH] = (
xor(get_randao_mix(state, get_current_epoch(state)),
hash(block.body.randao_reveal))
)
```
#### Eth1 data
```python
def process_eth1_data(state: BeaconState, block: BeaconBlock) -> None:
2019-04-17 14:30:03 +10:00
state.eth1_data_votes.append(block.body.eth1_data)
if state.eth1_data_votes.count(block.body.eth1_data) * 2 > SLOTS_PER_ETH1_VOTING_PERIOD:
state.latest_eth1_data = block.body.eth1_data
2019-03-07 12:05:34 -07:00
```
2019-03-31 09:02:10 +04:00
#### Operations
2019-03-07 12:05:34 -07:00
##### Proposer slashings
Verify that `len(block.body.proposer_slashings) <= MAX_PROPOSER_SLASHINGS` .
For each `proposer_slashing` in `block.body.proposer_slashings` , run the following function:
```python
def process_proposer_slashing(state: BeaconState,
proposer_slashing: ProposerSlashing) -> None:
"""
2019-03-31 09:02:10 +04:00
Process ``ProposerSlashing` ` operation.
2019-03-07 12:05:34 -07:00
Note that this function mutates ``state` `.
"""
proposer = state.validator_registry[proposer_slashing.proposer_index]
2019-03-12 16:49:04 +01:00
# Verify that the epoch is the same
2019-03-12 12:24:37 +00:00
assert slot_to_epoch(proposer_slashing.header_1.slot) == slot_to_epoch(proposer_slashing.header_2.slot)
2019-03-08 18:34:51 +01:00
# But the headers are different
assert proposer_slashing.header_1 != proposer_slashing.header_2
2019-03-19 11:08:17 +00:00
# Check proposer is slashable
2019-03-19 11:12:50 +00:00
assert is_slashable_validator(proposer, get_current_epoch(state))
2019-03-07 12:05:34 -07:00
# Signatures are valid
for header in (proposer_slashing.header_1, proposer_slashing.header_2):
2019-03-31 20:48:44 +04:00
domain = get_domain(state, DOMAIN_BEACON_PROPOSER, slot_to_epoch(header.slot))
2019-04-18 21:17:29 -06:00
assert bls_verify(proposer.pubkey, signing_root(header), header.signature, domain)
2019-03-07 12:05:34 -07:00
slash_validator(state, proposer_slashing.proposer_index)
```
##### Attester slashings
2018-12-10 15:16:06 -06:00
2019-03-07 12:05:34 -07:00
Verify that `len(block.body.attester_slashings) <= MAX_ATTESTER_SLASHINGS` .
2018-10-01 17:19:16 -05:00
2019-03-07 12:05:34 -07:00
For each `attester_slashing` in `block.body.attester_slashings` , run the following function:
2019-01-07 18:53:33 -06:00
2019-03-01 19:49:28 -06:00
```python
2019-03-07 12:05:34 -07:00
def process_attester_slashing(state: BeaconState,
attester_slashing: AttesterSlashing) -> None:
"""
2019-03-31 09:02:10 +04:00
Process ``AttesterSlashing` ` operation.
2019-03-07 12:05:34 -07:00
Note that this function mutates ``state` `.
"""
2019-03-22 06:10:44 -05:00
attestation1 = attester_slashing.attestation_1
attestation2 = attester_slashing.attestation_2
2019-03-07 12:05:34 -07:00
# Check that the attestations are conflicting
assert attestation1.data != attestation2.data
assert (
is_double_vote(attestation1.data, attestation2.data) or
is_surround_vote(attestation1.data, attestation2.data)
)
2019-03-27 08:31:56 -06:00
2019-03-26 13:40:19 -06:00
assert verify_indexed_attestation(state, attestation1)
assert verify_indexed_attestation(state, attestation2)
2019-04-03 11:04:12 +11:00
attesting_indices_1 = attestation1.custody_bit_0_indices + attestation1.custody_bit_1_indices
attesting_indices_2 = attestation2.custody_bit_0_indices + attestation2.custody_bit_1_indices
2019-03-07 12:05:34 -07:00
slashable_indices = [
2019-04-03 11:04:12 +11:00
index for index in attesting_indices_1
2019-03-07 12:05:34 -07:00
if (
2019-04-03 11:04:12 +11:00
index in attesting_indices_2 and
2019-03-19 11:12:50 +00:00
is_slashable_validator(state.validator_registry[index], get_current_epoch(state))
2019-03-01 19:49:28 -06:00
)
2019-03-07 12:05:34 -07:00
]
assert len(slashable_indices) >= 1
for index in slashable_indices:
slash_validator(state, index)
2019-03-01 19:49:28 -06:00
```
2019-01-18 21:06:21 -06:00
2019-03-07 12:05:34 -07:00
##### Attestations
2019-01-07 18:53:33 -06:00
2019-03-07 12:05:34 -07:00
Verify that `len(block.body.attestations) <= MAX_ATTESTATIONS` .
2019-03-01 19:49:28 -06:00
2019-03-07 12:05:34 -07:00
For each `attestation` in `block.body.attestations` , run the following function:
2018-11-05 07:37:07 -05:00
2018-12-28 12:04:03 -06:00
```python
2019-03-07 12:05:34 -07:00
def process_attestation(state: BeaconState, attestation: Attestation) -> None:
2019-01-30 17:25:39 +08:00
"""
2019-03-31 09:02:10 +04:00
Process ``Attestation` ` operation.
2019-01-30 17:25:39 +08:00
Note that this function mutates ``state` `.
"""
2019-04-03 23:40:54 +04:00
data = attestation.data
2019-04-17 20:27:25 -06:00
min_slot = state.slot - SLOTS_PER_EPOCH if get_current_epoch(state) > GENESIS_EPOCH else GENESIS_SLOT
assert min_slot < = data.slot < = state.slot - MIN_ATTESTATION_INCLUSION_DELAY
2019-03-15 12:40:52 +00:00
2019-04-03 23:40:54 +04:00
# Check target epoch, source epoch, source root, and source crosslink
target_epoch = slot_to_epoch(data.slot)
2019-04-08 12:59:42 +10:00
assert (target_epoch, data.source_epoch, data.source_root, data.previous_crosslink_root) in {
2019-04-07 17:55:38 +10:00
(get_current_epoch(state), state.current_justified_epoch, state.current_justified_root, hash_tree_root(state.current_crosslinks[data.shard])),
(get_previous_epoch(state), state.previous_justified_epoch, state.previous_justified_root, hash_tree_root(state.previous_crosslinks[data.shard])),
2019-03-15 12:40:52 +00:00
}
2019-04-03 23:40:54 +04:00
# Check crosslink data root
assert data.crosslink_data_root == ZERO_HASH # [to be removed in phase 1]
2019-03-15 12:40:52 +00:00
2019-03-22 06:10:44 -05:00
# Check signature and bitfields
2019-03-26 13:40:19 -06:00
assert verify_indexed_attestation(state, convert_to_indexed(state, attestation))
2019-03-15 12:40:52 +00:00
# Cache pending attestation
2019-03-07 12:05:34 -07:00
pending_attestation = PendingAttestation(
2019-04-03 23:40:54 +04:00
data=data,
2019-03-07 12:05:34 -07:00
aggregation_bitfield=attestation.aggregation_bitfield,
2019-04-24 14:23:51 +10:00
inclusion_slot=state.slot,
proposer_index=get_beacon_proposer_index(state),
2019-03-07 12:05:34 -07:00
)
2019-03-15 12:51:46 +00:00
if target_epoch == get_current_epoch(state):
2019-03-07 12:05:34 -07:00
state.current_epoch_attestations.append(pending_attestation)
2019-03-15 12:40:52 +00:00
else:
2019-03-07 12:05:34 -07:00
state.previous_epoch_attestations.append(pending_attestation)
2019-02-12 12:24:19 +00:00
```
2019-03-07 12:05:34 -07:00
##### Deposits
2019-04-18 00:43:22 -07:00
Verify that `len(block.body.deposits) == min(MAX_DEPOSITS, state.latest_eth1_data.deposit_count - state.deposit_index)` .
2019-03-07 12:05:34 -07:00
2019-04-12 18:57:55 +10:00
For each `deposit` in `block.body.deposits` , run the following function:
```python
def process_deposit(state: BeaconState, deposit: Deposit) -> None:
"""
2019-04-20 16:32:41 +10:00
Process an Eth1 deposit, registering a validator or increasing its balance.
2019-04-12 18:57:55 +10:00
Note that this function mutates ``state` `.
"""
# Verify the Merkle branch
2019-04-20 16:32:41 +10:00
assert verify_merkle_branch(
2019-04-15 07:54:08 +10:00
leaf=hash_tree_root(deposit.data),
2019-04-12 18:57:55 +10:00
proof=deposit.proof,
depth=DEPOSIT_CONTRACT_TREE_DEPTH,
index=deposit.index,
root=state.latest_eth1_data.deposit_root,
)
2019-04-20 16:35:02 +10:00
# Deposits must be processed in order
assert deposit.index == state.deposit_index
2019-04-12 18:57:55 +10:00
state.deposit_index += 1
pubkey = deposit.data.pubkey
amount = deposit.data.amount
2019-04-20 16:36:34 +10:00
validator_pubkeys = [v.pubkey for v in state.validator_registry]
2019-04-12 18:57:55 +10:00
if pubkey not in validator_pubkeys:
2019-04-18 21:17:29 -06:00
# Verify the deposit signature (proof of possession)
if not bls_verify(pubkey, signing_root(deposit.data), deposit.data.signature, get_domain(state, DOMAIN_DEPOSIT)):
2019-04-12 18:57:55 +10:00
return
2019-04-22 16:13:46 +10:00
# Add validator and balance entries
state.validator_registry.append(Validator(
2019-04-12 18:57:55 +10:00
pubkey=pubkey,
withdrawal_credentials=deposit.data.withdrawal_credentials,
2019-04-14 08:30:13 +10:00
activation_eligibility_epoch=FAR_FUTURE_EPOCH,
2019-04-12 18:57:55 +10:00
activation_epoch=FAR_FUTURE_EPOCH,
exit_epoch=FAR_FUTURE_EPOCH,
withdrawable_epoch=FAR_FUTURE_EPOCH,
2019-04-26 18:46:35 +04:00
effective_balance=min(amount - amount % EFFECTIVE_BALANCE_INCREMENT, MAX_EFFECTIVE_BALANCE)
2019-04-22 16:13:46 +10:00
))
2019-04-20 15:17:33 +10:00
state.balances.append(amount)
2019-04-12 18:57:55 +10:00
else:
# Increase balance by deposit amount
index = validator_pubkeys.index(pubkey)
increase_balance(state, index, amount)
```
2019-03-07 12:05:34 -07:00
##### Voluntary exits
Verify that `len(block.body.voluntary_exits) <= MAX_VOLUNTARY_EXITS` .
For each `exit` in `block.body.voluntary_exits` , run the following function:
2019-02-12 12:24:19 +00:00
```python
2019-03-10 13:50:28 +01:00
def process_voluntary_exit(state: BeaconState, exit: VoluntaryExit) -> None:
2019-02-12 12:24:19 +00:00
"""
2019-03-31 09:02:10 +04:00
Process ``VoluntaryExit` ` operation.
2019-02-12 12:24:19 +00:00
Note that this function mutates ``state` `.
"""
2019-03-07 12:05:34 -07:00
validator = state.validator_registry[exit.validator_index]
2019-03-19 15:49:01 -06:00
# Verify the validator is active
assert is_active_validator(validator, get_current_epoch(state))
2019-03-07 12:05:34 -07:00
# Verify the validator has not yet exited
2019-03-10 13:49:37 +01:00
assert validator.exit_epoch == FAR_FUTURE_EPOCH
2019-03-07 12:05:34 -07:00
# Exits must specify an epoch when they become valid; they are not valid before then
assert get_current_epoch(state) >= exit.epoch
2019-03-19 20:43:05 +00:00
# Verify the validator has been active long enough
2019-03-08 15:16:06 -07:00
assert get_current_epoch(state) - validator.activation_epoch >= PERSISTENT_COMMITTEE_PERIOD
2019-03-07 12:05:34 -07:00
# Verify signature
2019-03-31 04:55:24 -05:00
domain = get_domain(state, DOMAIN_VOLUNTARY_EXIT, exit.epoch)
2019-04-18 21:17:29 -06:00
assert bls_verify(validator.pubkey, signing_root(exit), exit.signature, domain)
2019-03-19 20:43:05 +00:00
# Initiate exit
2019-03-07 12:05:34 -07:00
initiate_validator_exit(state, exit.validator_index)
2018-10-01 17:19:16 -05:00
```
2018-11-07 17:47:27 +01:00
2019-03-07 12:05:34 -07:00
##### Transfers
2018-09-20 13:20:49 +08:00
2019-03-07 12:05:34 -07:00
Verify that `len(block.body.transfers) <= MAX_TRANSFERS` and that all transfers are distinct.
For each `transfer` in `block.body.transfers` , run the following function:
2019-03-01 19:49:28 -06:00
```python
2019-03-07 12:05:34 -07:00
def process_transfer(state: BeaconState, transfer: Transfer) -> None:
"""
2019-03-31 09:02:10 +04:00
Process ``Transfer` ` operation.
2019-03-07 12:05:34 -07:00
Note that this function mutates ``state` `.
"""
2019-04-20 15:17:33 +10:00
# Verify the amount and fee are not individually too big (for anti-overflow purposes)
assert state.balances[transfer.sender] >= max(transfer.amount, transfer.fee)
2019-03-07 12:05:34 -07:00
# A transfer is valid in only one slot
assert state.slot == transfer.slot
2019-04-18 17:51:50 +10:00
# Sender must be not yet eligible for activation, withdrawn, or transfer balance over MAX_EFFECTIVE_BALANCE
2019-03-07 12:05:34 -07:00
assert (
2019-04-18 17:51:50 +10:00
state.validator_registry[transfer.sender].activation_eligibility_epoch == FAR_FUTURE_EPOCH or
2019-03-07 12:05:34 -07:00
get_current_epoch(state) >= state.validator_registry[transfer.sender].withdrawable_epoch or
2019-04-24 15:17:25 +10:00
transfer.amount + transfer.fee + MAX_EFFECTIVE_BALANCE < = state.balances[transfer.sender]
2019-03-01 19:49:28 -06:00
)
2019-03-07 12:05:34 -07:00
# Verify that the pubkey is valid
assert (
state.validator_registry[transfer.sender].withdrawal_credentials ==
BLS_WITHDRAWAL_PREFIX_BYTE + hash(transfer.pubkey)[1:]
)
# Verify that the signature is valid
2019-04-18 21:17:29 -06:00
assert bls_verify(transfer.pubkey, signing_root(transfer), transfer.signature, get_domain(state, DOMAIN_TRANSFER))
2019-03-07 12:05:34 -07:00
# Process the transfer
2019-03-11 10:44:18 -06:00
decrease_balance(state, transfer.sender, transfer.amount + transfer.fee)
increase_balance(state, transfer.recipient, transfer.amount)
2019-04-17 12:32:50 +10:00
increase_balance(state, get_beacon_proposer_index(state), transfer.fee)
2019-04-14 21:53:32 +10:00
# Verify balances are not dust
2019-04-20 15:17:33 +10:00
assert not (0 < state.balances [ transfer . sender ] < MIN_DEPOSIT_AMOUNT )
assert not (0 < state.balances [ transfer . recipient ] < MIN_DEPOSIT_AMOUNT )
2019-03-01 19:49:28 -06:00
```
2018-10-18 13:54:57 -04:00
2019-03-07 12:05:34 -07:00
#### State root verification
Verify the block's `state_root` by running the following function:
2018-12-05 13:03:24 -06:00
2019-03-07 12:05:34 -07:00
```python
def verify_block_state_root(state: BeaconState, block: BeaconBlock) -> None:
assert block.state_root == hash_tree_root(state)
```