eth2.0-specs/specs/phase1/shard-fork-choice.md

180 lines
6.4 KiB
Markdown
Raw Normal View History

2020-04-28 23:55:46 +08:00
# Ethereum 2.0 Phase 1 -- Beacon Chain + Shard Chain Fork Choice
**Notice**: This document is a work-in-progress for researchers and implementers.
## Table of contents
<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
2020-04-28 23:55:46 +08:00
- [Introduction](#introduction)
- [Fork choice](#fork-choice)
- [Helpers](#helpers)
2020-05-01 10:33:23 +08:00
- [`ShardStore`](#shardstore)
- [`get_forkchoice_shard_store`](#get_forkchoice_shard_store)
2020-04-28 23:55:46 +08:00
- [`get_shard_latest_attesting_balance`](#get_shard_latest_attesting_balance)
- [`get_shard_head`](#get_shard_head)
- [`get_shard_ancestor`](#get_shard_ancestor)
2020-06-04 20:31:54 +08:00
- [`get_pendings_shard_blocks`](#get_pendings_shard_blocks)
2020-04-28 23:55:46 +08:00
- [Handlers](#handlers)
- [`on_shard_block`](#on_shard_block)
<!-- END doctoc generated TOC please keep comment here to allow auto update -->
## Introduction
This document is the shard chain fork choice spec for part of Ethereum 2.0 Phase 1. It assumes the [beacon chain fork choice spec](./fork-choice.md).
2020-04-28 23:55:46 +08:00
## Fork choice
### Helpers
#### `ShardStore`
2020-04-28 23:55:46 +08:00
```python
@dataclass
class ShardStore:
shard: Shard
blocks: Dict[Root, ShardBlock] = field(default_factory=dict)
block_states: Dict[Root, ShardState] = field(default_factory=dict)
2020-04-28 23:55:46 +08:00
```
2020-05-01 10:33:23 +08:00
#### `get_forkchoice_shard_store`
2020-04-28 23:55:46 +08:00
```python
def get_forkchoice_shard_store(anchor_state: BeaconState, shard: Shard) -> ShardStore:
return ShardStore(
shard=shard,
blocks={anchor_state.shard_states[shard].latest_block_root: ShardBlock(slot=anchor_state.slot)},
block_states={anchor_state.shard_states[shard].latest_block_root: anchor_state.copy().shard_states[shard]},
2020-04-28 23:55:46 +08:00
)
```
#### `get_shard_latest_attesting_balance`
```python
def get_shard_latest_attesting_balance(store: Store, shard_store: ShardStore, root: Root) -> Gwei:
2020-04-28 23:55:46 +08:00
state = store.checkpoint_states[store.justified_checkpoint]
active_indices = get_active_validator_indices(state, get_current_epoch(state))
return Gwei(sum(
state.validators[i].effective_balance for i in active_indices
if (
i in store.latest_messages
and store.latest_messages[i].shard == shard_store.shard
and get_shard_ancestor(
store, shard_store, store.latest_messages[i].root, shard_store.blocks[root].slot
2020-04-28 23:55:46 +08:00
) == root
)
))
```
#### `get_shard_head`
```python
def get_shard_head(store: Store, shard_store: ShardStore) -> Root:
2020-04-28 23:55:46 +08:00
# Execute the LMD-GHOST fork choice
2020-06-02 18:08:28 +08:00
shard_blocks = shard_store.blocks
2020-04-28 23:55:46 +08:00
head_beacon_root = get_head(store)
2020-06-02 18:08:28 +08:00
head_shard_state = store.block_states[head_beacon_root].shard_states[shard_store.shard]
2020-06-04 01:00:52 +08:00
shard_head_root = head_shard_state.latest_block_root
2020-04-28 23:55:46 +08:00
while True:
2020-06-02 18:08:28 +08:00
# Find the valid child block roots
2020-04-28 23:55:46 +08:00
children = [
2020-05-01 10:33:23 +08:00
root for root in shard_store.blocks.keys()
2020-06-02 18:08:28 +08:00
if (
2020-06-04 01:00:52 +08:00
shard_blocks[root].shard_parent_root == shard_head_root
2020-06-02 18:08:28 +08:00
and shard_blocks[root].slot > head_shard_state.slot
)
2020-04-28 23:55:46 +08:00
]
if len(children) == 0:
2020-06-04 01:00:52 +08:00
return shard_head_root
2020-04-28 23:55:46 +08:00
# Sort by latest attesting balance with ties broken lexicographically
2020-06-04 01:00:52 +08:00
shard_head_root = max(
children, key=lambda root: (get_shard_latest_attesting_balance(store, shard_store, root), root)
)
2020-04-28 23:55:46 +08:00
```
#### `get_shard_ancestor`
```python
def get_shard_ancestor(store: Store, shard_store: ShardStore, root: Root, slot: Slot) -> Root:
block = shard_store.blocks[root]
2020-04-28 23:55:46 +08:00
if block.slot > slot:
return get_shard_ancestor(store, shard_store, block.shard_parent_root, slot)
2020-04-28 23:55:46 +08:00
elif block.slot == slot:
return root
else:
# root is older than queried slot, thus a skip slot. Return earliest root prior to slot
return root
```
2020-06-04 20:31:54 +08:00
#### `get_pendings_shard_blocks`
```python
def get_pendings_shard_blocks(store: Store, shard_store: ShardStore) -> Sequence[ShardBlock]:
"""
Return the shard blocks branch that from shard head to beacon head.
"""
shard = shard_store.shard
beacon_head_root = get_head(store)
beacon_head_state = store.block_states[beacon_head_root]
latest_shard_block_root = beacon_head_state.shard_states[shard].latest_block_root
shard_head_root = get_shard_head(store, shard_store)
root = shard_head_root
shard_blocks = []
while root != latest_shard_block_root:
shard_block = shard_store.blocks[root]
shard_blocks.append(shard_block)
root = shard_block.shard_parent_root
shard_blocks.reverse()
return shard_blocks
```
2020-04-28 23:55:46 +08:00
### Handlers
#### `on_shard_block`
```python
def on_shard_block(store: Store, shard_store: ShardStore, signed_shard_block: SignedShardBlock) -> None:
2020-04-28 23:55:46 +08:00
shard_block = signed_shard_block.message
shard = shard_store.shard
# Check shard
# TODO: check it in networking spec
assert shard_block.shard == shard
# Check shard parent exists
2020-04-30 20:06:20 +08:00
assert shard_block.shard_parent_root in shard_store.block_states
pre_shard_state = shard_store.block_states[shard_block.shard_parent_root]
2020-04-28 23:55:46 +08:00
# Check beacon parent exists
2020-04-28 23:55:46 +08:00
assert shard_block.beacon_parent_root in store.block_states
# Check that block is later than the finalized shard state slot (optimization to reduce calls to get_ancestor)
2020-06-02 18:08:28 +08:00
finalized_beacon_state = store.block_states[store.finalized_checkpoint.root]
finalized_shard_state = finalized_beacon_state.shard_states[shard]
assert shard_block.slot > finalized_shard_state.slot
2020-04-28 23:55:46 +08:00
# Check block is a descendant of the finalized block at the checkpoint finalized slot
2020-06-02 18:08:28 +08:00
finalized_slot = compute_start_slot_at_epoch(store.finalized_checkpoint.epoch)
2020-04-28 23:55:46 +08:00
assert (
2020-06-02 18:08:28 +08:00
get_ancestor(store, shard_block.beacon_parent_root, finalized_slot) == store.finalized_checkpoint.root
2020-04-28 23:55:46 +08:00
)
# Add new block to the store
2020-04-30 20:06:20 +08:00
shard_store.blocks[hash_tree_root(shard_block)] = shard_block
2020-04-28 23:55:46 +08:00
# Check the block is valid and compute the post-state
beacon_head_root = get_head(store)
beacon_head_state = store.block_states[beacon_head_root]
assert verify_shard_block_message(beacon_head_state, pre_shard_state, shard_block, shard_block.slot, shard)
assert verify_shard_block_signature(beacon_head_state, signed_shard_block)
post_state = get_post_shard_state(beacon_head_state, pre_shard_state, shard_block)
2020-04-28 23:55:46 +08:00
# Add new state for this block to the store
2020-04-30 20:06:20 +08:00
shard_store.block_states[hash_tree_root(shard_block)] = post_state
2020-04-28 23:55:46 +08:00
```