eth2.0-specs/specs/core/0_fork-choice.md

286 lines
12 KiB
Markdown
Raw Normal View History

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-06-15 18:42:03 -04:00
## Fork choice
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
2019-11-21 23:13:45 +01:00
- `on_block(block)` whenever a block `block: SignedBeaconBlock` is received
2019-06-30 21:25:58 +02:00
- `on_attestation(attestation)` whenever an attestation `attestation` is received
2019-06-15 18:42:03 -04:00
*Notes*:
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.
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
### Configuration
| Name | Value | Unit | Duration |
| - | - | :-: | :-: |
2019-11-05 12:51:47 -07:00
| `SAFE_SLOTS_TO_UPDATE_JUSTIFIED` | `2**3` (= 8) | slots | 96 seconds |
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: Root
2019-06-25 14:42:37 -06:00
```
2019-06-15 18:42:03 -04:00
#### `Store`
```python
@dataclass
class Store(object):
time: uint64
genesis_time: uint64
2019-06-24 21:01:15 -06:00
justified_checkpoint: Checkpoint
finalized_checkpoint: Checkpoint
best_justified_checkpoint: Checkpoint
blocks: Dict[Root, BeaconBlock] = field(default_factory=dict)
block_states: Dict[Root, BeaconState] = field(default_factory=dict)
2019-06-21 12:55:55 -06:00
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))
2019-11-21 23:13:45 +01:00
root = hash_tree_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)
return Store(
time=genesis_state.genesis_time,
genesis_time=genesis_state.genesis_time,
2019-06-21 12:55:55 -06:00
justified_checkpoint=justified_checkpoint,
finalized_checkpoint=finalized_checkpoint,
best_justified_checkpoint=justified_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-15 18:42:03 -04:00
```
2019-04-22 23:20:48 +10:00
2019-11-06 17:20:21 -07:00
#### `get_current_slot`
```python
def get_current_slot(store: Store) -> Slot:
return Slot((store.time - store.genesis_time) // SECONDS_PER_SLOT)
```
2019-11-06 17:20:21 -07:00
#### `compute_slots_since_epoch_start`
```python
def compute_slots_since_epoch_start(slot: Slot) -> int:
return slot - compute_start_slot_at_epoch(compute_epoch_at_slot(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
def get_ancestor(store: Store, root: Root, slot: Slot) -> Root:
2019-06-15 18:42:03 -04:00
block = store.blocks[root]
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
def get_latest_attesting_balance(store: Store, root: Root) -> 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))
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-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
def get_head(store: Store) -> Root:
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:
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
```
#### `should_update_justified_checkpoint`
```python
def should_update_justified_checkpoint(store: Store, new_justified_checkpoint: Checkpoint) -> bool:
2019-11-07 11:51:53 -07:00
"""
To address the bouncing attack, only update conflicting justified
checkpoints in the fork choice if in the early slots of the epoch.
Otherwise, delay incorporation of new justified checkpoint until next epoch boundary.
See https://ethresear.ch/t/prevention-of-bouncing-attack-on-ffg/6114 for more detailed analysis and discussion.
"""
2019-11-06 17:20:21 -07:00
if compute_slots_since_epoch_start(get_current_slot(store)) < SAFE_SLOTS_TO_UPDATE_JUSTIFIED:
return True
new_justified_block = store.blocks[new_justified_checkpoint.root]
if new_justified_block.slot <= compute_start_slot_at_epoch(store.justified_checkpoint.epoch):
return False
if not (
get_ancestor(store, new_justified_checkpoint.root, store.blocks[store.justified_checkpoint.root].slot) ==
store.justified_checkpoint.root
):
return False
return True
```
2019-06-15 18:42:03 -04:00
### Handlers
#### `on_tick`
```python
def on_tick(store: Store, time: uint64) -> None:
previous_slot = get_current_slot(store)
# update store time
2019-06-15 18:42:03 -04:00
store.time = time
current_slot = get_current_slot(store)
# Not a new epoch, return
2019-11-07 11:51:53 -07:00
if not (current_slot > previous_slot and compute_slots_since_epoch_start(current_slot) == 0):
return
2019-11-07 11:51:53 -07:00
# Update store.justified_checkpoint if a better checkpoint is known
if store.best_justified_checkpoint.epoch > store.justified_checkpoint.epoch:
store.justified_checkpoint = store.best_justified_checkpoint
2019-04-22 23:20:48 +10:00
```
2019-06-15 18:42:03 -04:00
#### `on_block`
2019-06-15 18:42:03 -04:00
```python
2019-11-21 23:13:45 +01:00
def on_block(store: Store, signed_block: SignedBeaconBlock) -> None:
block = signed_block.message
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
2019-11-21 23:13:45 +01:00
store.blocks[hash_tree_root(block)] = block
2019-06-15 18:42:03 -04:00
# Check block is a descendant of the finalized block
2019-06-24 21:09:57 -06:00
assert (
2019-11-21 23:13:45 +01:00
get_ancestor(store, hash_tree_root(block), store.blocks[store.finalized_checkpoint.root].slot) ==
2019-06-24 21:09:57 -06:00
store.finalized_checkpoint.root
)
# 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-11-21 23:13:45 +01:00
state = state_transition(pre_state, signed_block, True)
# Add new state for this block to the store
2019-11-21 23:13:45 +01:00
store.block_states[hash_tree_root(block)] = state
2019-06-21 12:55:55 -06:00
# Update justified checkpoint
2019-06-24 21:43:05 -06:00
if state.current_justified_checkpoint.epoch > store.justified_checkpoint.epoch:
store.best_justified_checkpoint = state.current_justified_checkpoint
if should_update_justified_checkpoint(store, state.current_justified_checkpoint):
store.justified_checkpoint = 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-06-15 18:42:03 -04:00
#### `on_attestation`
```python
def on_attestation(store: Store, attestation: Attestation) -> None:
2019-11-12 16:24:33 -07:00
"""
Run ``on_attestation`` upon receiving a new ``attestation`` from either within a block or directly on the wire.
An ``attestation`` that is asserted as invalid may be valid at a later time,
consider scheduling it for later processing in such case.
"""
2019-06-24 21:43:05 -06:00
target = attestation.data.target
2019-06-21 12:55:55 -06:00
# Attestations must be from the current or previous epoch
current_epoch = compute_epoch_at_slot(get_current_slot(store))
# Use GENESIS_EPOCH for previous when genesis to avoid underflow
previous_epoch = current_epoch - 1 if current_epoch > GENESIS_EPOCH else GENESIS_EPOCH
assert target.epoch in [current_epoch, previous_epoch]
2019-06-24 17:18:22 -06:00
# Attestations target be for a known block. If target block is unknown, delay consideration until the block is found
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
# Attestations must be for a known block. If block is unknown, delay consideration until the block is found
assert attestation.data.beacon_block_root in store.blocks
# Attestations must not be for blocks in the future. If not, the attestation should not be considered
assert store.blocks[attestation.data.beacon_block_root].slot <= attestation.data.slot
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-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)
assert is_valid_indexed_attestation(target_state, indexed_attestation)
2019-06-25 14:42:37 -06:00
# Update latest messages
2019-11-01 21:02:53 -06:00
for i in indexed_attestation.attesting_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
```