2019-04-22 23:20:48 +10:00
# Ethereum 2.0 Phase 0 -- Beacon Chain Fork Choice
2019-05-06 10:30:32 -05:00
**Notice**: This document is a work-in-progress for researchers and implementers.
2019-04-22 23:20:48 +10:00
## Table of contents
<!-- TOC -->
- [Ethereum 2.0 Phase 0 -- Beacon Chain Fork Choice ](#ethereum-20-phase-0----beacon-chain-fork-choice )
- [Table of contents ](#table-of-contents )
- [Introduction ](#introduction )
2019-06-15 18:42:03 -04:00
- [Fork choice ](#fork-choice )
2019-06-20 11:58:05 +01:00
- [Helpers ](#helpers )
2019-07-01 04:57:42 +08:00
- [`LatestMessage` ](#latestmessage )
2019-06-15 18:42:03 -04:00
- [`Store` ](#store )
- [`get_genesis_store` ](#get_genesis_store )
- [`get_ancestor` ](#get_ancestor )
2019-06-20 11:58:05 +01:00
- [`get_latest_attesting_balance` ](#get_latest_attesting_balance )
2019-06-15 18:42:03 -04:00
- [`get_head` ](#get_head )
- [Handlers ](#handlers )
- [`on_tick` ](#on_tick )
- [`on_block` ](#on_block )
- [`on_attestation` ](#on_attestation )
2019-04-22 23:20:48 +10:00
<!-- /TOC -->
## Introduction
2019-06-15 18:42:03 -04:00
This document is the beacon chain fork choice spec, part of Ethereum 2.0 Phase 0. It assumes the [beacon chain state transition function spec ](./0_beacon-chain.md ).
2019-04-24 13:37:50 -06:00
2019-06-15 18:42:03 -04:00
## Fork choice
2019-06-25 08:32:56 -05:00
The head block root associated with a `store` is defined as `get_head(store)` . At genesis, let `store = get_genesis_store(genesis_state)` and update `store` by running:
2019-06-15 18:42:03 -04:00
2019-06-30 21:25:58 +02:00
- `on_tick(time)` whenever `time > store.time` where `time` is the current Unix time
- `on_block(block)` whenever a block `block` is received
- `on_attestation(attestation)` whenever an attestation `attestation` is received
2019-06-15 18:42:03 -04:00
*Notes*:
2019-06-21 12:13:22 +02:00
1) **Leap seconds** : Slots will last `SECONDS_PER_SLOT + 1` or `SECONDS_PER_SLOT - 1` seconds around leap seconds. This is automatically handled by [UNIX time ](https://en.wikipedia.org/wiki/Unix_time ).
2019-06-15 18:42:03 -04:00
2) **Honest clocks** : Honest nodes are assumed to have clocks synchronized within `SECONDS_PER_SLOT` seconds of each other.
2019-06-25 08:32:56 -05:00
3) **Eth1 data** : The large `ETH1_FOLLOW_DISTANCE` specified in the [honest validator document ](../validator/0_beacon-chain-validator.md ) should ensure that `state.latest_eth1_data` of the canonical Ethereum 2.0 chain remains consistent with the canonical Ethereum 1.0 chain. If not, emergency manual intervention will be required.
2019-06-20 11:58:05 +01:00
4) **Manual forks** : Manual forks may arbitrarily change the fork choice rule but are expected to be enacted at epoch transitions, with the fork details reflected in `state.fork` .
2019-06-21 11:18:24 -06:00
5) **Implementation** : The implementation found in this specification is constructed for ease of understanding rather than for optimization in computation, space, or any other resource. A number of optimized alternatives can be found [here ](https://github.com/protolambda/lmd-ghost ).
2019-06-15 18:42:03 -04:00
2019-11-05 08:55:34 -07:00
### Configuration
| Name | Value | Unit | Duration |
| - | - | :-: | :-: |
2019-11-05 11:02:58 -07:00
| `SAFE_SLOTS_TO_UPDATE_JUSTIFIED` | `2**3 (= 8)` | slots | 96 seconds |
2019-11-05 08:55:34 -07:00
2019-06-20 11:58:05 +01:00
### Helpers
2019-04-22 23:20:48 +10:00
2019-06-25 14:42:37 -06:00
#### `LatestMessage`
```python
@dataclass (eq=True, frozen=True)
class LatestMessage(object):
epoch: Epoch
root: Hash
```
2019-06-15 18:42:03 -04:00
#### `Store`
```python
2019-06-17 10:48:33 -04:00
@dataclass
class Store(object):
2019-06-30 13:00:22 -05:00
time: uint64
2019-11-05 08:55:34 -07:00
genesis_time: uint64
2019-06-24 21:01:15 -06:00
justified_checkpoint: Checkpoint
finalized_checkpoint: Checkpoint
2019-11-05 08:55:34 -07:00
queued_justified_checkpoints: List[Checkpoint, 2**40] = field(default_factory=list)
2019-06-20 11:58:05 +01:00
blocks: Dict[Hash, BeaconBlock] = field(default_factory=dict)
2019-06-21 12:55:55 -06:00
block_states: Dict[Hash, BeaconState] = field(default_factory=dict)
checkpoint_states: Dict[Checkpoint, BeaconState] = field(default_factory=dict)
2019-06-25 14:42:37 -06:00
latest_messages: Dict[ValidatorIndex, LatestMessage] = field(default_factory=dict)
2019-06-15 18:42:03 -04:00
```
2019-04-22 23:20:48 +10:00
2019-06-15 18:42:03 -04:00
#### `get_genesis_store`
2019-04-22 23:20:48 +10:00
2019-06-15 18:42:03 -04:00
```python
def get_genesis_store(genesis_state: BeaconState) -> Store:
genesis_block = BeaconBlock(state_root=hash_tree_root(genesis_state))
root = signing_root(genesis_block)
2019-06-24 21:43:05 -06:00
justified_checkpoint = Checkpoint(epoch=GENESIS_EPOCH, root=root)
finalized_checkpoint = Checkpoint(epoch=GENESIS_EPOCH, root=root)
2019-06-20 20:50:17 +02:00
return Store(
time=genesis_state.genesis_time,
2019-11-05 08:55:34 -07:00
genesis_time=genesis_state.genesis_time,
2019-06-21 12:55:55 -06:00
justified_checkpoint=justified_checkpoint,
finalized_checkpoint=finalized_checkpoint,
2019-06-24 21:01:15 -06:00
blocks={root: genesis_block},
2019-06-25 11:48:55 -06:00
block_states={root: genesis_state.copy()},
2019-06-24 21:01:15 -06:00
checkpoint_states={justified_checkpoint: genesis_state.copy()},
2019-06-20 20:50:17 +02:00
)
2019-06-15 18:42:03 -04:00
```
2019-04-22 23:20:48 +10:00
2019-11-05 08:55:34 -07:00
```python
def get_current_slot(store: Store) -> Slot:
return Slot((store.time - store.genesis_time) // SECONDS_PER_SLOT)
```
2019-06-15 18:42:03 -04:00
#### `get_ancestor`
2019-04-22 23:20:48 +10:00
2019-06-15 18:42:03 -04:00
```python
2019-06-20 11:58:05 +01:00
def get_ancestor(store: Store, root: Hash, slot: Slot) -> Hash:
2019-06-15 18:42:03 -04:00
block = store.blocks[root]
2019-07-20 02:13:31 +02:00
if block.slot > slot:
return get_ancestor(store, block.parent_root, slot)
elif block.slot == slot:
return root
else:
return Bytes32() # root is older than queried slot: no results.
2019-06-15 18:42:03 -04:00
```
2019-04-22 23:20:48 +10:00
2019-06-20 11:58:05 +01:00
#### `get_latest_attesting_balance`
2019-04-22 23:20:48 +10:00
```python
2019-06-20 11:58:05 +01:00
def get_latest_attesting_balance(store: Store, root: Hash) -> Gwei:
2019-06-21 12:55:55 -06:00
state = store.checkpoint_states[store.justified_checkpoint]
2019-06-25 11:48:55 -06:00
active_indices = get_active_validator_indices(state, get_current_epoch(state))
2019-06-19 20:27:54 +02:00
return Gwei(sum(
2019-06-25 11:48:55 -06:00
state.validators[i].effective_balance for i in active_indices
2019-06-30 22:59:12 +02:00
if (i in store.latest_messages
and get_ancestor(store, store.latest_messages[i].root, store.blocks[root].slot) == root)
2019-06-19 20:27:54 +02:00
))
2019-04-22 23:20:48 +10:00
```
2019-06-15 18:42:03 -04:00
#### `get_head`
2019-04-22 23:20:48 +10:00
```python
2019-06-20 11:58:05 +01:00
def get_head(store: Store) -> Hash:
2019-06-15 18:42:03 -04:00
# Execute the LMD-GHOST fork choice
2019-06-21 12:55:55 -06:00
head = store.justified_checkpoint.root
2019-10-23 09:37:15 +09:00
justified_slot = compute_start_slot_at_epoch(store.justified_checkpoint.epoch)
2019-06-15 18:42:03 -04:00
while True:
2019-06-20 14:48:10 -06:00
children = [
root for root in store.blocks.keys()
if store.blocks[root].parent_root == head and store.blocks[root].slot > justified_slot
]
2019-04-22 23:20:48 +10:00
if len(children) == 0:
return head
2019-06-20 11:58:05 +01:00
# Sort by latest attesting balance with ties broken lexicographically
head = max(children, key=lambda root: (get_latest_attesting_balance(store, root), root))
2019-06-15 18:42:03 -04:00
```
2019-11-05 08:55:34 -07:00
#### `should_update_justified_checkpoint`
```python
def should_update_justified_checkpoint(store: Store, justified_checkpoint: Checkpoint) -> bool:
current_epoch = compute_epoch_at_slot(get_current_slot(store))
2019-11-05 10:58:45 -07:00
if get_current_slot(store) % SLOTS_PER_EPOCH < SAFE_SLOTS_TO_UPDATE_JUSTIFIED:
2019-11-05 08:55:34 -07:00
return True
justified_block = store.blocks[justified_checkpoint.root]
if justified_block.slot < = compute_start_slot_at_epoch(store.justified_checkpoint.epoch):
return False
if not get_ancestor(store, justified_checkpoint.root, store.blocks[justified_checkpoint.root].slot):
return False
return True
```
2019-06-15 18:42:03 -04:00
### Handlers
#### `on_tick`
```python
2019-06-30 13:00:22 -05:00
def on_tick(store: Store, time: uint64) -> None:
2019-11-05 08:55:34 -07:00
previous_slot = get_current_slot(store)
# update store time
2019-06-15 18:42:03 -04:00
store.time = time
2019-11-05 08:55:34 -07:00
current_slot = get_current_slot(store)
2019-11-05 11:02:58 -07:00
# Not a new epoch, return
2019-11-05 08:55:34 -07:00
if not (current_slot > previous_slot and current_slot % SLOTS_PER_EPOCH == 0):
return
2019-11-05 11:02:58 -07:00
# If new epoch and there are queued_justified_checkpoints, update if any is better than the best in store
2019-11-05 08:55:34 -07:00
if any(store.queued_justified_checkpoints):
best_justified_checkpoint = max(store.queued_justified_checkpoints, key=lambda checkpoint: checkpoint.epoch)
if best_justified_checkpoint.epoch > store.justified_checkpoint.epoch:
store.justified_checkpoint = best_justified_checkpoint
2019-11-05 11:02:58 -07:00
store.queued_justified_checkpoints = []
2019-04-22 23:20:48 +10:00
```
2019-05-01 16:10:01 -07:00
2019-06-15 18:42:03 -04:00
#### `on_block`
2019-05-01 16:10:01 -07:00
2019-06-15 18:42:03 -04:00
```python
def on_block(store: Store, block: BeaconBlock) -> None:
2019-06-21 13:00:42 +02:00
# Make a copy of the state to avoid mutability issues
2019-06-24 21:09:57 -06:00
assert block.parent_root in store.block_states
2019-06-24 21:01:15 -06:00
pre_state = store.block_states[block.parent_root].copy()
2019-06-21 12:19:08 +02:00
# Blocks cannot be in the future. If they are, their consideration must be delayed until the are in the past.
assert store.time >= pre_state.genesis_time + block.slot * SECONDS_PER_SLOT
2019-06-15 18:42:03 -04:00
# Add new block to the store
store.blocks[signing_root(block)] = block
# Check block is a descendant of the finalized block
2019-06-24 21:09:57 -06:00
assert (
get_ancestor(store, signing_root(block), store.blocks[store.finalized_checkpoint.root].slot) ==
store.finalized_checkpoint.root
)
2019-06-20 14:48:10 -06:00
# Check that block is later than the finalized epoch slot
2019-10-23 09:37:15 +09:00
assert block.slot > compute_start_slot_at_epoch(store.finalized_checkpoint.epoch)
2019-06-15 18:42:03 -04:00
# Check the block is valid and compute the post-state
2019-07-24 19:26:39 -07:00
state = state_transition(pre_state, block, True)
2019-06-20 14:48:10 -06:00
# Add new state for this block to the store
2019-06-21 12:55:55 -06:00
store.block_states[signing_root(block)] = state
# Update justified checkpoint
2019-06-24 21:43:05 -06:00
if state.current_justified_checkpoint.epoch > store.justified_checkpoint.epoch:
2019-11-05 08:55:34 -07:00
if should_update_justified_checkpoint(store, state.current_justified_checkpoint):
store.justified_checkpoint = state.current_justified_checkpoint
else:
store.queued_justified_checkpoints.append(state.current_justified_checkpoint)
2019-06-21 12:55:55 -06:00
# Update finalized checkpoint
2019-06-24 21:43:05 -06:00
if state.finalized_checkpoint.epoch > store.finalized_checkpoint.epoch:
store.finalized_checkpoint = state.finalized_checkpoint
2019-06-15 18:42:03 -04:00
```
2019-05-01 16:10:01 -07:00
2019-06-15 18:42:03 -04:00
#### `on_attestation`
```python
def on_attestation(store: Store, attestation: Attestation) -> None:
2019-06-24 21:43:05 -06:00
target = attestation.data.target
2019-06-21 12:55:55 -06:00
# Cannot calculate the current shuffling if have not seen the target
2019-06-24 17:18:22 -06:00
assert target.root in store.blocks
2019-06-30 21:25:58 +02:00
# Attestations cannot be from future epochs. If they are, delay consideration until the epoch arrives
2019-06-24 21:01:15 -06:00
base_state = store.block_states[target.root].copy()
2019-10-23 09:37:15 +09:00
assert store.time >= base_state.genesis_time + compute_start_slot_at_epoch(target.epoch) * SECONDS_PER_SLOT
2019-06-20 14:48:10 -06:00
2019-06-21 12:55:55 -06:00
# Store target checkpoint state if not yet seen
2019-06-24 17:18:22 -06:00
if target not in store.checkpoint_states:
2019-10-23 09:37:15 +09:00
process_slots(base_state, compute_start_slot_at_epoch(target.epoch))
2019-06-24 21:09:57 -06:00
store.checkpoint_states[target] = base_state
2019-06-24 17:18:22 -06:00
target_state = store.checkpoint_states[target]
# Attestations can only affect the fork choice of subsequent slots.
# Delay consideration in the fork choice until their slot is in the past.
2019-10-18 12:10:36 +09:00
assert store.time >= (attestation.data.slot + 1) * SECONDS_PER_SLOT
2019-06-20 14:48:10 -06:00
2019-06-24 17:18:22 -06:00
# Get state at the `target` to validate attestation and calculate the committees
2019-06-30 10:02:18 +01:00
indexed_attestation = get_indexed_attestation(target_state, attestation)
2019-07-01 00:10:28 +02:00
assert is_valid_indexed_attestation(target_state, indexed_attestation)
2019-06-20 14:48:10 -06:00
2019-06-25 14:42:37 -06:00
# Update latest messages
2019-06-15 18:42:03 -04:00
for i in indexed_attestation.custody_bit_0_indices + indexed_attestation.custody_bit_1_indices:
2019-06-25 14:42:37 -06:00
if i not in store.latest_messages or target.epoch > store.latest_messages[i].epoch:
store.latest_messages[i] = LatestMessage(epoch=target.epoch, root=attestation.data.beacon_block_root)
2019-06-15 18:42:03 -04:00
```