2020-06-24 18:08:26 +08:00
# Ethereum 2.0 Phase 1 -- The Beacon Chain with Shards
2019-10-12 12:05:08 +09:00
**Notice**: This document is a work-in-progress for researchers and implementers.
## Table of contents
2020-01-25 00:43:43 +01:00
<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE - RUN doctoc TO UPDATE -->
2020-09-23 22:55:22 +02:00
2020-01-25 00:43:43 +01:00
- [Introduction ](#introduction )
- [Custom types ](#custom-types )
- [Configuration ](#configuration )
2020-12-11 16:10:50 +08:00
- [Misc ](#misc )
- [Shard block configs ](#shard-block-configs )
- [Precomputed size verification points ](#precomputed-size-verification-points )
- [Gwei values ](#gwei-values )
- [Time parameters ](#time-parameters )
- [Domain types ](#domain-types )
2020-01-25 00:43:43 +01:00
- [Updated containers ](#updated-containers )
2020-12-11 16:10:50 +08:00
- [`AttestationData` ](#attestationdata )
- [`BeaconBlock` ](#beaconblock )
- [`BeaconState` ](#beaconstate )
2020-01-25 00:43:43 +01:00
- [New containers ](#new-containers )
2020-12-11 16:10:50 +08:00
- [`DataCommitment` ](#datacommitment )
- [`ShardHeader` ](#shardheader )
2021-01-04 11:17:30 -07:00
- [`SignedShardHeader` ](#signedshardheader )
2020-12-11 16:10:50 +08:00
- [`PendingShardHeader` ](#pendingshardheader )
2020-01-25 00:43:43 +01:00
- [Helper functions ](#helper-functions )
2020-12-11 16:10:50 +08:00
- [Misc ](#misc-1 )
2021-01-04 11:17:30 -07:00
- [`next_power_of_two` ](#next_power_of_two )
- [`reverse_bit_order` ](#reverse_bit_order )
2020-12-11 16:10:50 +08:00
- [`compute_previous_slot` ](#compute_previous_slot )
- [`compute_updated_gasprice` ](#compute_updated_gasprice )
- [`compute_committee_source_epoch` ](#compute_committee_source_epoch )
- [Beacon state accessors ](#beacon-state-accessors )
- [Updated `get_committee_count_per_slot` ](#updated-get_committee_count_per_slot )
- [`get_active_shard_count` ](#get_active_shard_count )
- [`get_shard_committee` ](#get_shard_committee )
2021-01-04 11:17:30 -07:00
- [`compute_proposer_index` ](#compute_proposer_index )
2020-12-11 16:10:50 +08:00
- [`get_shard_proposer_index` ](#get_shard_proposer_index )
- [`get_start_shard` ](#get_start_shard )
2021-01-04 11:17:30 -07:00
- [`compute_shard_from_committee_index` ](#compute_shard_from_committee_index )
- [`compute_committee_index_from_shard` ](#compute_committee_index_from_shard )
2020-12-11 16:10:50 +08:00
- [Block processing ](#block-processing )
- [Operations ](#operations )
- [New Attestation processing ](#new-attestation-processing )
- [Updated `process_attestation` ](#updated-process_attestation )
- [`update_pending_votes` ](#update_pending_votes )
- [`process_shard_header` ](#process_shard_header )
- [Epoch transition ](#epoch-transition )
- [Pending headers ](#pending-headers )
- [Custody game updates ](#custody-game-updates )
2019-10-12 12:05:08 +09:00
2020-01-25 00:43:43 +01:00
<!-- END doctoc generated TOC please keep comment here to allow auto update -->
2019-10-12 12:05:08 +09:00
## Introduction
2020-12-09 15:29:21 +08:00
This document describes the extensions made to the Phase 0 design of The Beacon Chain to support data sharding,
based on the ideas [here ](https://hackmd.io/G-Iy5jqyT7CXWEz8Ssos8g ) and more broadly [here ](https://arxiv.org/abs/1809.09044 ),
using KZG10 commitments to commit to data to remove any need for fraud proofs (and hence, safety-critical synchrony assumptions) in the design.
2019-10-12 12:05:08 +09:00
2019-11-18 16:40:02 -07:00
## Custom types
We define the following Python custom types for type hinting and readability:
| Name | SSZ equivalent | Description |
| - | - | - |
2020-12-10 14:29:09 +08:00
| `Shard` | `uint64` | A shard number |
| `BLSCommitment` | `bytes48` | A G1 curve point |
2021-01-01 16:51:24 +01:00
| `BLSKateProof` | `bytes48` | A G1 curve point |
2019-11-18 16:40:02 -07:00
2019-10-12 12:05:08 +09:00
## Configuration
### Misc
2020-12-10 10:21:21 +08:00
| Name | Value | Notes |
| - | - | - |
| `MAX_SHARDS` | `uint64(2**10)` (= 1024) | Theoretical max shard count (used to determine data structure sizes) |
| `INITIAL_ACTIVE_SHARDS` | `uint64(2**6)` (= 64) | Initial shard count |
2020-12-14 22:57:53 +00:00
| `GASPRICE_ADJUSTMENT_COEFFICIENT` | `uint64(2**3)` (= 8) | Gasprice may decrease/increase by at most exp(1 / this value) *per epoch* |
2020-12-13 08:34:59 +08:00
| `MAX_SHARD_HEADERS_PER_SHARD` | `4` | |
| `MAX_SHARD_HEADERS` | `MAX_SHARDS * MAX_SHARD_HEADERS_PER_SHARD` | |
2020-12-14 22:57:53 +00:00
| `PRIMITIVE_ROOT_OF_UNITY` | `5` | Primitive root of unity of the BLS12_381 (inner) modulus |
2020-12-16 14:14:21 +00:00
| `DATA_AVAILABILITY_INVERSE_CODING_RATE` | `2**1` (=2) | Factor by which samples are extended for data availability encoding |
2020-06-09 13:34:46 -07:00
### Shard block configs
2020-06-09 15:38:40 -07:00
2020-12-09 15:29:21 +08:00
| Name | Value | Notes |
2020-06-15 15:27:37 +08:00
| - | - | - |
2020-12-09 15:29:21 +08:00
| `POINTS_PER_SAMPLE` | `uint64(2**3)` (= 8) | 31 * 8 = 248 bytes |
| `MAX_SAMPLES_PER_BLOCK` | `uint64(2**11)` (= 2,048) | 248 * 2,048 = 507,904 bytes |
| `TARGET_SAMPLES_PER_BLOCK` | `uint64(2**10)` (= 1,024) | 248 * 1,024 = 253,952 bytes |
2020-06-09 13:34:46 -07:00
2020-12-09 15:29:21 +08:00
### Precomputed size verification points
2020-06-09 13:34:46 -07:00
| Name | Value |
| - | - |
2020-12-28 17:38:09 +00:00
| `G1_SETUP` | Type `List[G1]` . The G1-side trusted setup `[G, G*s, G*s**2....]` ; note that the first point is the generator. |
| `G2_SETUP` | Type `List[G2]` . The G2-side trusted setup `[G, G*s, G*s**2....]` |
2020-12-28 16:12:31 +00:00
| `MODULUS` | `0x73eda753299d7d483339d80809a1d80553bda402fffe5bfeffffffff00000001` (curve order of BLS12_381) |
| `ROOT_OF_UNITY` | `pow(PRIMITIVE_ROOT_OF_UNITY, (MODULUS - 1) // (MAX_SAMPLES_PER_BLOCK * POINTS_PER_SAMPLE, MODULUS)` |
2020-06-09 13:34:46 -07:00
2020-12-09 15:29:21 +08:00
### Gwei values
| Name | Value | Unit | Description |
| - | - | - | - |
2020-12-18 13:57:05 +00:00
| `MAX_GASPRICE` | `Gwei(2**33)` (= 8,589,934,592) | Gwei | Max gasprice charged for a TARGET-sized shard block |
2020-12-14 22:57:53 +00:00
| `MIN_GASPRICE` | `Gwei(2**3)` (= 8) | Gwei | Min gasprice charged for a TARGET-sized shard block |
2020-06-09 13:34:46 -07:00
### Time parameters
| Name | Value | Unit | Duration |
| - | - | :-: | :-: |
2020-12-09 15:29:21 +08:00
| `SHARD_COMMITTEE_PERIOD` | `Epoch(2**8)` (= 256) | epochs | ~27 hours |
2020-04-02 16:48:02 -06:00
### Domain types
| Name | Value |
| - | - |
2020-12-10 10:21:21 +08:00
| `DOMAIN_SHARD_HEADER` | `DomainType('0x80000000')` |
2020-12-31 09:35:27 +08:00
| `DOMAIN_SHARD_COMMITTEE` | `DomainType('0x81000000')` |
2019-10-12 12:05:08 +09:00
2019-11-22 15:45:55 -07:00
## Updated containers
2019-10-12 12:05:08 +09:00
2019-11-22 15:45:55 -07:00
The following containers have updated definitions in Phase 1.
2019-11-03 12:06:19 -08:00
2020-12-09 15:29:21 +08:00
### `AttestationData`
2019-10-27 09:01:10 +08:00
2019-10-29 10:43:13 -07:00
```python
2019-11-22 15:45:55 -07:00
class AttestationData(Container):
2019-10-12 12:05:08 +09:00
slot: Slot
2019-10-27 09:01:10 +08:00
index: CommitteeIndex
2019-10-12 12:05:08 +09:00
# LMD GHOST vote
2019-12-05 12:36:48 -07:00
beacon_block_root: Root
2019-10-12 12:05:08 +09:00
# FFG vote
source: Checkpoint
target: Checkpoint
2020-12-09 15:29:21 +08:00
# Shard header root
shard_header_root: Root
2019-11-15 21:11:42 +01:00
```
2020-12-10 11:47:02 +08:00
### `BeaconBlock`
```python
class BeaconBlock(phase0.BeaconBlock):
2021-01-01 21:52:29 +01:00
< < < < < < < HEAD
2020-12-09 15:29:21 +08:00
# insert phase 0 fields
2021-01-01 21:52:29 +01:00
=======
>>>>>>> 3c19069e (more DAS spec work: DAS function signatures, gossip details)
2021-01-04 11:17:30 -07:00
shard_headers: List[SignedShardHeader, MAX_SHARD_HEADERS]
2020-12-10 11:47:02 +08:00
```
2020-12-09 15:29:21 +08:00
### `BeaconState`
2019-11-16 11:13:47 +01:00
2019-11-15 21:11:42 +01:00
```python
2020-12-09 15:29:21 +08:00
class BeaconState(phase0.BeaconState):
2020-12-11 16:17:16 +08:00
# Updated fields
previous_epoch_attestations: List[PendingAttestation, MAX_ATTESTATIONS * SLOTS_PER_EPOCH]
current_epoch_attestations: List[PendingAttestation, MAX_ATTESTATIONS * SLOTS_PER_EPOCH]
# New fields
2020-12-14 22:57:53 +00:00
previous_epoch_pending_shard_headers: List[PendingShardHeader, MAX_SHARD_HEADERS * SLOTS_PER_EPOCH]
2020-12-28 16:57:44 +00:00
current_epoch_pending_shard_headers: List[PendingShardHeader, MAX_SHARD_HEADERS * SLOTS_PER_EPOCH]
2020-12-28 16:53:50 +00:00
grandparent_epoch_confirmed_commitments: Vector[Vector[DataCommitment, SLOTS_PER_EPOCH], MAX_SHARDS]
2020-12-09 15:29:21 +08:00
shard_gasprice: uint64
2020-12-10 11:47:02 +08:00
current_epoch_start_shard: Shard
2019-10-12 23:59:51 +09:00
```
2019-11-22 15:45:55 -07:00
## New containers
The following containers are new in Phase 1.
2020-12-10 14:42:43 +08:00
### `DataCommitment`
```python
class DataCommitment(Container):
2020-12-28 16:24:31 +00:00
# KZG10 commitment to the data
2020-12-10 14:42:43 +08:00
point: BLSCommitment
# Length of the data in samples
2020-12-28 16:01:47 +00:00
length: uint64
2020-12-10 14:42:43 +08:00
```
2020-12-09 15:29:21 +08:00
### `ShardHeader`
2019-11-22 15:45:55 -07:00
```python
2020-12-09 15:29:21 +08:00
class ShardHeader(Container):
# Slot and shard that this header is intended for
2019-11-22 15:45:55 -07:00
slot: Slot
2020-05-28 21:32:27 +08:00
shard: Shard
2020-12-10 14:42:43 +08:00
# The actual data commitment
commitment: DataCommitment
2020-12-28 16:36:28 +00:00
# Proof that the degree < commitment.length
2021-01-01 16:51:24 +01:00
degree_proof: BLSKateProof
2020-03-31 12:37:36 +08:00
```
2021-01-01 21:52:29 +01:00
TODO: add shard-proposer-index to shard headers, similar to optimization done with beacon-blocks.
2021-01-04 11:17:30 -07:00
### `SignedShardHeader`
```python
class SignedShardHeader(Container):
message: ShardHeader
signature: BLSSignature
```
2020-12-09 15:29:21 +08:00
### `PendingShardHeader`
2020-03-31 12:37:36 +08:00
```python
2020-12-09 15:29:21 +08:00
class PendingShardHeader(Container):
# Slot and shard that this header is intended for
2020-12-28 16:14:05 +00:00
slot: Slot
2020-05-28 21:32:27 +08:00
shard: Shard
2020-12-28 16:24:31 +00:00
# KZG10 commitment to the data
2020-12-28 17:21:46 +00:00
commitment: DataCommitment
2021-01-04 11:17:30 -07:00
# hash_tree_root of the ShardHeader (stored so that attestations can be checked against it)
2020-12-28 16:55:06 +00:00
root: Root
2020-12-09 15:29:21 +08:00
# Who voted for the header
2020-12-28 16:56:43 +00:00
votes: Bitlist[MAX_VALIDATORS_PER_COMMITTEE]
2020-12-09 15:29:21 +08:00
# Has this header been confirmed?
confirmed: bool
2019-11-22 15:45:55 -07:00
```
2019-11-15 21:11:42 +01:00
## Helper functions
### Misc
2020-12-18 20:33:34 +00:00
#### `next_power_of_two`
```python
def next_power_of_two(x):
return 2 ** ((x - 1).bit_length())
```
2020-04-28 11:28:21 +08:00
#### `compute_previous_slot`
2020-01-05 20:11:55 +01:00
```python
2020-04-28 11:28:21 +08:00
def compute_previous_slot(slot: Slot) -> Slot:
2020-01-05 20:11:55 +01:00
if slot > 0:
return Slot(slot - 1)
else:
return Slot(0)
```
2020-04-16 19:43:48 +08:00
#### `compute_updated_gasprice`
```python
2020-12-10 10:21:21 +08:00
def compute_updated_gasprice(prev_gasprice: Gwei, shard_block_length: uint64, adjustment_quotient: uint64) -> Gwei:
2020-12-09 15:29:21 +08:00
if shard_block_length > TARGET_SAMPLES_PER_BLOCK:
2020-12-16 11:55:31 +08:00
delta = max(1, prev_gasprice * (shard_block_length - TARGET_SAMPLES_PER_BLOCK)
// TARGET_SAMPLES_PER_BLOCK // adjustment_quotient)
2020-04-16 19:43:48 +08:00
return min(prev_gasprice + delta, MAX_GASPRICE)
else:
2020-12-16 11:55:31 +08:00
delta = max(1, prev_gasprice * (TARGET_SAMPLES_PER_BLOCK - shard_block_length)
// TARGET_SAMPLES_PER_BLOCK // adjustment_quotient)
2020-04-16 19:43:48 +08:00
return max(prev_gasprice, MIN_GASPRICE + delta) - delta
```
2020-05-30 03:09:42 +08:00
#### `compute_committee_source_epoch`
```python
def compute_committee_source_epoch(epoch: Epoch, period: uint64) -> Epoch:
"""
Return the source epoch for computing the committee.
"""
2020-09-03 19:24:17 +08:00
source_epoch = Epoch(epoch - epoch % period)
2020-05-30 03:09:42 +08:00
if source_epoch >= period:
source_epoch -= period # `period` epochs lookahead
return source_epoch
```
2019-11-15 21:11:42 +01:00
### Beacon state accessors
2020-08-10 13:02:22 -06:00
#### Updated `get_committee_count_per_slot`
2020-08-05 12:44:31 -06:00
```python
def get_committee_count_per_slot(state: BeaconState, epoch: Epoch) -> uint64:
"""
Return the number of committees in each slot for the given ``epoch` `.
"""
return max(uint64(1), min(
2020-12-10 11:47:02 +08:00
get_active_shard_count(state, epoch),
2020-08-05 12:44:31 -06:00
uint64(len(get_active_validator_indices(state, epoch))) // SLOTS_PER_EPOCH // TARGET_COMMITTEE_SIZE,
))
```
2020-01-05 20:20:20 +01:00
#### `get_active_shard_count`
```python
2020-12-10 11:47:02 +08:00
def get_active_shard_count(state: BeaconState, epoch: Epoch) -> uint64:
2020-08-10 13:02:22 -06:00
"""
Return the number of active shards.
Note that this puts an upper bound on the number of committees per slot.
"""
2020-08-05 12:44:31 -06:00
return INITIAL_ACTIVE_SHARDS
2020-01-05 20:20:20 +01:00
```
2019-11-15 21:11:42 +01:00
#### `get_shard_committee`
2019-11-12 06:13:47 -08:00
```python
def get_shard_committee(beacon_state: BeaconState, epoch: Epoch, shard: Shard) -> Sequence[ValidatorIndex]:
2020-05-28 22:39:52 +08:00
"""
Return the shard committee of the given ``epoch`` of the given ``shard` `.
"""
2020-05-30 03:09:42 +08:00
source_epoch = compute_committee_source_epoch(epoch, SHARD_COMMITTEE_PERIOD)
2019-11-12 06:13:47 -08:00
active_validator_indices = get_active_validator_indices(beacon_state, source_epoch)
seed = get_seed(beacon_state, source_epoch, DOMAIN_SHARD_COMMITTEE)
2020-04-02 11:20:22 +08:00
return compute_committee(
indices=active_validator_indices,
seed=seed,
index=shard,
2020-12-10 11:47:02 +08:00
count=get_active_shard_count(beacon_state, epoch),
2020-04-02 11:20:22 +08:00
)
2019-11-12 06:13:47 -08:00
```
2020-12-18 13:57:05 +00:00
#### `compute_proposer_index`
2021-01-04 11:17:30 -07:00
Updated version to get a proposer index that will only allow proposers with a certain minimum balance,
ensuring that the balance is always sufficient to cover gas costs.
2020-12-18 13:57:05 +00:00
```python
2021-01-04 11:17:30 -07:00
def compute_proposer_index(beacon_state: BeaconState,
indices: Sequence[ValidatorIndex],
seed: Bytes32,
min_effective_balance: GWei = GWei(0)) -> ValidatorIndex:
2020-12-18 13:57:05 +00:00
"""
Return from ``indices` ` a random index sampled by effective balance.
"""
assert len(indices) > 0
MAX_RANDOM_BYTE = 2**8 - 1
i = uint64(0)
total = uint64(len(indices))
while True:
candidate_index = indices[compute_shuffled_index(i % total, total, seed)]
random_byte = hash(seed + uint_to_bytes(uint64(i // 32)))[i % 32]
2021-01-04 11:17:30 -07:00
effective_balance = beacon_state.validators[candidate_index].effective_balance
if effective_balance < = min_effective_balance:
continue
if effective_balance * MAX_RANDOM_BYTE >= MAX_EFFECTIVE_BALANCE * random_byte:
2020-12-18 13:57:05 +00:00
return candidate_index
i += 1
```
2019-11-15 21:11:42 +01:00
#### `get_shard_proposer_index`
2019-11-12 06:13:47 -08:00
```python
def get_shard_proposer_index(beacon_state: BeaconState, slot: Slot, shard: Shard) -> ValidatorIndex:
2020-06-04 11:19:04 +08:00
"""
Return the proposer's index of shard block at ``slot` `.
"""
2020-06-04 05:06:04 +08:00
epoch = compute_epoch_at_slot(slot)
2020-06-04 11:19:04 +08:00
committee = get_shard_committee(beacon_state, epoch, shard)
2021-01-04 11:17:30 -07:00
seed = hash(get_seed(beacon_state, epoch, DOMAIN_BEACON_PROPOSER) + uint_to_bytes(beacon_state.slot))
2020-12-18 13:57:05 +00:00
2021-01-04 11:17:30 -07:00
# Proposer must have sufficient balance to pay for worst case fee burn
EFFECTIVE_BALANCE_MAX_DOWNWARD_DEVIATION = (
(EFFECTIVE_BALANCE_INCREMENT - EFFECTIVE_BALANCE_INCREMENT)
* HYSTERESIS_DOWNWARD_MULTIPLIER // HYSTERESIS_QUOTIENT
)
min_effective_balance = (
beacon_state.shard_gasprice * MAX_SAMPLES_PER_BLOCK // TARGET_SAMPLES_PER_BLOCK
+ EFFECTIVE_BALANCE_MAX_DOWNWARD_DEVIATION
)
return compute_proposer_index(beacon_state, committee, seed, min_effective_balance)
2020-05-29 23:50:18 +08:00
```
2019-11-18 16:40:02 -07:00
#### `get_start_shard`
```python
def get_start_shard(state: BeaconState, slot: Slot) -> Shard:
2020-05-29 23:50:18 +08:00
"""
Return the start shard at ``slot` `.
"""
current_epoch_start_slot = compute_start_slot_at_epoch(get_current_epoch(state))
2020-12-10 11:47:02 +08:00
shard = state.current_epoch_start_shard
if slot > current_epoch_start_slot:
2020-05-29 23:50:18 +08:00
# Current epoch or the next epoch lookahead
2020-12-10 11:47:02 +08:00
for _slot in range(current_epoch_start_slot, slot):
committee_count = get_committee_count_per_slot(state, compute_epoch_at_slot(Slot(_slot)))
active_shard_count = get_active_shard_count(state, compute_epoch_at_slot(Slot(_slot)))
shard = (shard + committee_count) % active_shard_count
elif slot < current_epoch_start_slot:
2020-05-29 23:50:18 +08:00
# Previous epoch
2020-12-10 11:47:02 +08:00
for _slot in list(range(slot, current_epoch_start_slot))[::-1]:
committee_count = get_committee_count_per_slot(state, compute_epoch_at_slot(Slot(_slot)))
active_shard_count = get_active_shard_count(state, compute_epoch_at_slot(Slot(_slot)))
2020-05-29 23:50:18 +08:00
# Ensure positive
2020-12-10 11:47:02 +08:00
shard = (shard + active_shard_count - committee_count) % active_shard_count
return Shard(shard)
2019-11-18 16:40:02 -07:00
```
2020-12-31 09:56:29 +08:00
#### `compute_shard_from_committee_index`
```python
def compute_shard_from_committee_index(state: BeaconState, slot: Slot, index: CommitteeIndex) -> Shard:
active_shards = get_active_shard_count(state, compute_epoch_at_slot(slot))
return Shard((index + get_start_shard(state, slot)) % active_shards)
```
#### `compute_committee_index_from_shard`
```python
def compute_committee_index_from_shard(state: BeaconState, slot: Slot, shard: Shard) -> CommitteeIndex:
active_shards = get_active_shard_count(state, compute_epoch_at_slot(slot))
return CommitteeIndex((active_shards + shard - get_start_shard(state, slot)) % active_shards)
```
2019-11-15 21:11:42 +01:00
### Block processing
2019-11-06 14:19:00 -08:00
```python
2019-11-15 21:11:42 +01:00
def process_block(state: BeaconState, block: BeaconBlock) -> None:
process_block_header(state, block)
process_randao(state, block.body)
process_eth1_data(state, block.body)
2020-04-01 12:02:56 -06:00
process_light_client_aggregate(state, block.body)
2019-11-15 21:11:42 +01:00
process_operations(state, block.body)
2019-11-12 05:27:34 -08:00
```
2019-11-15 21:11:42 +01:00
#### Operations
2019-10-12 12:05:08 +09:00
2019-10-13 17:42:55 +09:00
```python
2019-11-15 21:11:42 +01:00
def process_operations(state: BeaconState, body: BeaconBlockBody) -> None:
# Verify that outstanding deposits are processed up to the maximum number of deposits
assert len(body.deposits) == min(MAX_DEPOSITS, state.eth1_data.deposit_count - state.eth1_deposit_index)
2020-02-06 12:58:21 -06:00
2019-11-20 04:43:32 +01:00
def for_ops(operations: Sequence[Any], fn: Callable[[BeaconState, Any], None]) -> None:
2019-11-15 21:11:42 +01:00
for operation in operations:
fn(state, operation)
2020-02-06 12:58:21 -06:00
2019-11-16 11:13:47 +01:00
for_ops(body.proposer_slashings, process_proposer_slashing)
for_ops(body.attester_slashings, process_attester_slashing)
2021-01-04 11:17:30 -07:00
# Limit is dynamic based on active shard count
assert len(body.shard_headers) < = MAX_SHARD_HEADERS_PER_SHARD * get_active_shard_count(state, get_current_epoch(state))
for_ops(body.shard_headers, process_shard_header)
2019-11-15 21:11:42 +01:00
# New attestation processing
2020-02-06 12:58:21 -06:00
for_ops(body.attestations, process_attestation)
2019-11-16 11:13:47 +01:00
for_ops(body.deposits, process_deposit)
for_ops(body.voluntary_exits, process_voluntary_exit)
2019-11-15 21:11:42 +01:00
2019-11-15 23:42:28 +01:00
# See custody game spec.
process_custody_game_operations(state, body)
2019-10-12 12:05:08 +09:00
```
2020-12-09 15:29:21 +08:00
### New Attestation processing
2019-11-03 08:17:46 -08:00
2020-12-09 15:29:21 +08:00
#### Updated `process_attestation`
2020-05-29 03:20:36 +08:00
```python
def process_attestation(state: BeaconState, attestation: Attestation) -> None:
2020-12-09 15:29:21 +08:00
phase0.process_attestation(state, attestation)
2020-12-31 09:35:27 +08:00
update_pending_votes(state, attestation)
2020-05-29 03:20:36 +08:00
```
2020-12-09 15:29:21 +08:00
#### `update_pending_votes`
2019-11-03 08:17:46 -08:00
```python
2020-12-09 15:29:21 +08:00
def update_pending_votes(state: BeaconState, attestation: Attestation) -> None:
2021-01-04 11:17:30 -07:00
# Find and update the PendingShardHeader object, invalid block if pending header not in state
if compute_epoch_at_slot(attestation.data.slot) == get_current_epoch(state):
2020-12-13 08:36:45 +08:00
pending_headers = state.current_epoch_pending_shard_headers
2020-12-09 15:29:21 +08:00
else:
2020-12-13 08:36:45 +08:00
pending_headers = state.previous_epoch_pending_shard_headers
2020-12-10 11:47:02 +08:00
pending_header = None
for header in pending_headers:
2020-12-09 15:29:21 +08:00
if header.root == attestation.data.shard_header_root:
2020-12-10 11:47:02 +08:00
pending_header = header
assert pending_header is not None
2020-12-31 09:35:27 +08:00
assert pending_header.slot == attestation.data.slot
2020-12-09 15:29:21 +08:00
assert pending_header.shard == compute_shard_from_committee_index(
state,
2020-12-31 09:56:29 +08:00
attestation.data.slot,
2020-12-09 15:29:21 +08:00
attestation.data.index,
)
2021-01-04 11:17:30 -07:00
for i in range(len(pending_header.votes)):
pending_header.votes[i] = pending_header.votes[i] or attestation.aggregation_bits[i]
2019-11-03 12:06:19 -08:00
2020-12-09 15:29:21 +08:00
# Check if the PendingShardHeader is eligible for expedited confirmation
# Requirement 1: nothing else confirmed
all_candidates = [
c for c in pending_headers if
(c.slot, c.shard) == (pending_header.slot, pending_header.shard)
]
2021-01-04 11:17:30 -07:00
if True in [c.confirmed for c in all_candidates]:
return
# Requirement 2: >= 2/3 of balance attesting
2021-03-17 22:41:35 +01:00
participants = get_attesting_indices(state, attestation.data, pending_header.votes)
2021-01-04 11:17:30 -07:00
participants_balance = get_total_balance(state, participants)
full_committee = get_beacon_committee(state, attestation.data.slot, attestation.data.index)
full_committee_balance = get_total_balance(state, full_committee)
if participants_balance * 3 >= full_committee_balance * 2:
pending_header.confirmed = True
2020-12-09 15:29:21 +08:00
```
2020-12-10 11:47:02 +08:00
#### `process_shard_header`
2020-12-09 15:29:21 +08:00
```python
2020-12-10 11:47:02 +08:00
def process_shard_header(state: BeaconState,
2021-01-04 11:17:30 -07:00
signed_header: SignedShardHeader) -> None:
2020-12-09 15:29:21 +08:00
header = signed_header.message
header_root = hash_tree_root(header)
2021-01-04 11:17:30 -07:00
assert compute_epoch_at_slot(header.slot) in [get_previous_epoch(state), get_current_epoch(state)]
2020-12-09 15:29:21 +08:00
# Verify signature
signer_index = get_shard_proposer_index(state, header.slot, header.shard)
2021-01-04 11:17:30 -07:00
signing_root = compute_signing_root(header, get_domain(state, DOMAIN_SHARD_HEADER))
assert bls.Verify(state.validators[signer_index].pubkey, signing_root, signed_header.signature)
2020-12-28 17:38:09 +00:00
# Verify the length by verifying the degree.
if header.commitment.length == 0:
assert header.degree_proof == G1_SETUP[0]
2019-11-18 15:07:50 -07:00
assert (
2021-01-04 11:17:30 -07:00
bls.Pairing(header.degree_proof, G2_SETUP[0])
== bls.Pairing(header.commitment.point, G2_SETUP[-header.commitment.length]))
2019-11-18 15:07:50 -07:00
)
2021-01-04 11:17:30 -07:00
2020-12-09 15:29:21 +08:00
# Get the correct pending header list
2020-12-10 11:47:02 +08:00
if compute_epoch_at_slot(header.slot) == get_current_epoch(state):
2020-12-13 08:36:45 +08:00
pending_headers = state.current_epoch_pending_shard_headers
2020-12-09 15:29:21 +08:00
else:
2020-12-13 08:36:45 +08:00
pending_headers = state.previous_epoch_pending_shard_headers
2021-01-04 11:17:30 -07:00
2020-12-09 15:29:21 +08:00
# Check that this header is not yet in the pending list
2021-01-04 11:17:30 -07:00
assert header_root not in [pending_header.root for pending_header in pending_headers]
2020-12-09 15:29:21 +08:00
# Include it in the pending list
2020-12-31 09:56:29 +08:00
index = compute_committee_index_from_shard(state, header.slot, header.shard)
committee_length = len(get_beacon_committee(state, header.slot, index))
2020-12-09 15:29:21 +08:00
pending_headers.append(PendingShardHeader(
slot=header.slot,
shard=header.shard,
commitment=header.commitment,
root=header_root,
2020-12-28 16:03:20 +00:00
votes=Bitlist[MAX_VALIDATORS_PER_COMMITTEE ]([0] * committee_length ),
2021-01-04 11:17:30 -07:00
confirmed=False,
2020-12-09 15:29:21 +08:00
))
2020-02-06 12:58:21 -06:00
```
2020-12-09 15:29:21 +08:00
The degree proof works as follows. For a block `B` with length `l` (so `l` values in `[0...l - 1]` , seen as a polynomial `B(X)` which takes these values),
the length proof is the commitment to the polynomial `B(X) * X**(MAX_DEGREE + 1 - l)` ,
where `MAX_DEGREE` is the maximum power of `s` available in the setup, which is `MAX_DEGREE = len(G2_SETUP) - 1` .
The goal is to ensure that a proof can only be constructed if `deg(B) < l` (there are not hidden higher-order terms in the polynomial, which would thwart reconstruction).
2020-02-06 12:58:21 -06:00
2019-10-12 23:59:51 +09:00
### Epoch transition
2019-11-15 21:11:42 +01:00
This epoch transition overrides the phase0 epoch transition:
2019-10-12 23:59:51 +09:00
```python
2019-11-15 21:11:42 +01:00
def process_epoch(state: BeaconState) -> None:
process_justification_and_finalization(state)
process_rewards_and_penalties(state)
process_registry_updates(state)
2020-12-09 15:29:21 +08:00
# Proof of custody
process_reveal_deadlines(state)
process_challenge_deadlines(state)
2019-11-15 21:11:42 +01:00
process_slashings(state)
2020-12-09 15:29:21 +08:00
# Sharding
process_pending_headers(state)
charge_confirmed_header_fees(state)
reset_pending_headers(state)
# Final updates
# Phase 0
2021-01-27 14:42:50 +08:00
process_eth1_data_reset(state)
process_effective_balance_updates(state)
process_slashings_reset(state)
process_randao_mixes_reset(state)
process_historical_roots_update(state)
2021-01-19 21:41:34 +08:00
process_participation_record_updates(state)
2020-12-09 15:29:21 +08:00
# Proof of custody
2019-11-15 23:42:28 +01:00
process_custody_final_updates(state)
2020-05-29 23:50:18 +08:00
# Update current_epoch_start_shard
2020-06-01 23:15:16 +08:00
state.current_epoch_start_shard = get_start_shard(state, Slot(state.slot + 1))
2019-11-15 23:42:28 +01:00
```
2019-11-15 21:11:42 +01:00
2020-12-09 15:29:21 +08:00
#### Pending headers
```python
2021-01-04 11:17:30 -07:00
def process_pending_headers(state: BeaconState) -> None:
# Pending header processing applies to the previous epoch.
# Skip if `GENESIS_EPOCH` because no prior epoch to process.
if get_current_epoch(state) == GENESIS_EPOCH:
return
previous_epoch_start_slot = compute_start_slot_at_epoch(get_previous_epoch(state))
for slot in range(previous_epoch_start_slot, previous_epoch_start_slot + SLOTS_PER_EPOCH):
2020-12-31 09:35:27 +08:00
for shard in range(get_active_shard_count(state)):
2020-12-09 15:29:21 +08:00
# Pending headers for this (slot, shard) combo
candidates = [
2021-01-04 11:17:30 -07:00
c for c in state.previous_epoch_pending_shard_headers
if (c.slot, c.shard) == (slot, shard)
2020-12-09 15:29:21 +08:00
]
2020-12-28 17:39:49 +00:00
# The entire committee (and its balance)
full_committee = get_beacon_committee(state, slot, shard)
full_committee_balance = get_total_balance(state, full_committee)
2021-01-04 11:17:30 -07:00
# If any candidates already confirmed, skip
if True in [c.confirmed for c in candidates]:
continue
# The set of voters who voted for each header (and their total balances)
voting_sets = [
[v for i, v in enumerate(full_committee) if c.votes[i]]
for c in candidates
]
voting_balances = [
get_total_balance(state, voters)
for voters in voting_sets
]
# Get the index with the most total balance voting for them.
# NOTE: if two choices get exactly the same voting balance,
# the candidate earlier in the list wins
if max(voting_balances) > 0:
winning_index = voting_balances.index(max(voting_balances))
else:
# If no votes, zero wins
winning_index = [c.root for c in candidates].index(Root())
candidates[winning_index].confirmed = True
for slot_index in range(SLOTS_PER_EPOCH):
2020-12-10 14:37:00 +08:00
for shard in range(SHARD_COUNT):
2021-01-04 11:17:30 -07:00
state.grandparent_epoch_confirmed_commitments[shard][slot_index] = DataCommitment()
confirmed_headers = [candidate in state.previous_epoch_pending_shard_headers if candidate.confirmed]
for header in confirmed_headers:
state.grandparent_epoch_confirmed_commitments[c.shard][c.slot % SLOTS_PER_EPOCH] = c.commitment
```
2020-12-09 15:29:21 +08:00
```python
def charge_confirmed_header_fees(state: BeaconState) -> None:
new_gasprice = state.shard_gasprice
2021-01-04 11:17:30 -07:00
adjustment_quotient = (
get_active_shard_count(state, get_current_epoch(state))
* SLOTS_PER_EPOCH * GASPRICE_ADJUSTMENT_COEFFICIENT
)
previous_epoch_start_slot = compute_start_slot_at_epoch(get_previous_epoch(state))
for slot in range(previous_epoch_start_slot, previous_epoch_start_slot + SLOTS_PER_EPOCH):
2020-12-09 15:29:21 +08:00
for shard in range(SHARD_COUNT):
confirmed_candidates = [
2021-01-04 11:17:30 -07:00
c for c in state.previous_epoch_pending_shard_headers
if (c.slot, c.shard, c.confirmed) == (slot, shard, True)
2020-12-09 15:29:21 +08:00
]
2021-01-04 11:17:30 -07:00
if not any(confirmed_candidates):
continue
candidate = confirmed_candidates[0]
# Charge EIP 1559 fee
proposer = get_shard_proposer(state, slot, shard)
fee = (
(state.shard_gasprice * candidate.commitment.length)
// TARGET_SAMPLES_PER_BLOCK
)
decrease_balance(state, proposer, fee)
# Track updated gas price
new_gasprice = compute_updated_gasprice(
new_gasprice,
candidate.commitment.length,
adjustment_quotient,
)
2020-12-09 15:29:21 +08:00
state.shard_gasprice = new_gasprice
```
```python
2020-12-09 15:29:21 +08:00
def reset_pending_headers(state: BeaconState) -> None:
2020-12-13 08:36:45 +08:00
state.previous_epoch_pending_shard_headers = state.current_epoch_pending_shard_headers
state.current_epoch_pending_shard_headers = []
2021-01-04 11:17:30 -07:00
# Add dummy "empty" PendingShardHeader (default vote for if no shard header available)
next_epoch = get_current_epoch(state) + 1
next_epoch_start_slot = compute_start_slot_at_epoch(next_epoch)
for slot in range(next_epoch_start_slot, next_epoch_start_slot + SLOTS_IN_EPOCH):
for index in range(get_committee_count_per_slot(next_epoch)
2020-12-31 09:56:29 +08:00
shard = compute_shard_from_committee_index(state, slot, index)
2021-01-04 11:17:30 -07:00
committee_length = len(get_beacon_committee(state, slot, shard))
2020-12-13 08:36:45 +08:00
state.current_epoch_pending_shard_headers.append(PendingShardHeader(
2020-12-09 15:29:21 +08:00
slot=slot,
shard=shard,
2020-12-10 14:42:43 +08:00
commitment=DataCommitment(),
2020-12-09 15:29:21 +08:00
root=Root(),
2020-12-28 16:03:20 +00:00
votes=Bitlist[MAX_VALIDATORS_PER_COMMITTEE ]([0] * committee_length ),
2021-01-04 11:17:30 -07:00
confirmed=False,
2020-12-09 15:29:21 +08:00
))
2019-10-12 23:59:51 +09:00
```
2019-11-15 23:42:28 +01:00
2020-12-09 15:29:21 +08:00
#### Custody game updates
2019-11-15 23:42:28 +01:00
2020-12-09 15:29:21 +08:00
`process_reveal_deadlines` , `process_challenge_deadlines` and `process_custody_final_updates` are defined in [the Custody Game spec ](./custody-game.md ).