94 lines
3.3 KiB
Python
Raw Normal View History

2019-05-15 18:36:32 +02:00
from copy import deepcopy
from eth2spec.test.helpers.keys import privkeys
from eth2spec.utils.bls import bls_sign, only_with_bls
2019-11-21 23:13:45 +01:00
from eth2spec.utils.ssz.ssz_impl import hash_tree_root
2019-05-15 18:36:32 +02:00
2019-11-21 23:13:45 +01:00
def get_proposer_index_maybe(spec, state, slot, proposer_index=None):
if proposer_index is None:
2019-11-21 23:13:45 +01:00
assert state.slot <= slot
if slot == state.slot:
2019-06-30 12:42:24 -05:00
proposer_index = spec.get_beacon_proposer_index(state)
else:
2019-11-21 23:13:45 +01:00
if spec.compute_epoch_at_slot(state.slot) + 1 > spec.compute_epoch_at_slot(slot):
print("warning: block slot far away, and no proposer index manually given."
" Signing block is slow due to transition for proposer index calculation.")
# use stub state to get proposer index of future slot
stub_state = deepcopy(state)
2019-11-21 23:13:45 +01:00
spec.process_slots(stub_state, slot)
2019-06-30 12:42:24 -05:00
proposer_index = spec.get_beacon_proposer_index(stub_state)
2019-11-21 23:13:45 +01:00
return proposer_index
2019-05-15 18:36:32 +02:00
2019-11-21 23:13:45 +01:00
@only_with_bls()
def apply_randao_reveal(spec, state, block, proposer_index=None):
assert state.slot <= block.slot
proposer_index = get_proposer_index_maybe(spec, state, block.slot, proposer_index)
2019-05-15 18:36:32 +02:00
privkey = privkeys[proposer_index]
block.body.randao_reveal = bls_sign(
2019-05-15 18:36:32 +02:00
privkey=privkey,
2019-10-18 12:05:43 +09:00
message_hash=hash_tree_root(spec.compute_epoch_at_slot(block.slot)),
2019-05-30 22:57:18 +02:00
domain=spec.get_domain(
2019-05-15 18:36:32 +02:00
state,
2019-10-18 12:05:43 +09:00
message_epoch=spec.compute_epoch_at_slot(block.slot),
2019-05-15 18:36:32 +02:00
domain_type=spec.DOMAIN_RANDAO,
)
)
2019-11-21 23:13:45 +01:00
# Fully ignore the function if BLS is off, beacon-proposer index calculation is slow.
@only_with_bls()
def apply_sig(spec, state, signed_block, proposer_index=None):
block = signed_block.message
proposer_index = get_proposer_index_maybe(spec, state, block.slot, proposer_index)
privkey = privkeys[proposer_index]
signed_block.signature = bls_sign(
message_hash=hash_tree_root(block),
2019-05-15 18:36:32 +02:00
privkey=privkey,
2019-05-30 22:57:18 +02:00
domain=spec.get_domain(
2019-05-15 18:36:32 +02:00
state,
spec.DOMAIN_BEACON_PROPOSER,
2019-10-18 12:05:43 +09:00
spec.compute_epoch_at_slot(block.slot)))
2019-05-15 18:36:32 +02:00
2019-11-21 23:13:45 +01:00
def sign_block(spec, state, block, proposer_index=None):
signed_block = spec.SignedBeaconBlock(message=block)
apply_sig(spec, state, signed_block, proposer_index)
return signed_block
def transition_unsigned_block(spec, state, block):
spec.process_slots(state, block.slot)
spec.process_block(state, block)
2019-05-30 22:57:18 +02:00
def apply_empty_block(spec, state):
"""
Transition via an empty block (on current slot, assuming no block has been applied yet).
"""
2019-11-21 23:13:45 +01:00
block = build_empty_block(spec, state)
transition_unsigned_block(spec, state, block)
2019-11-21 23:13:45 +01:00
def build_empty_block(spec, state, slot=None):
2019-05-15 18:36:32 +02:00
if slot is None:
slot = state.slot
2019-05-30 22:57:18 +02:00
empty_block = spec.BeaconBlock()
2019-05-15 18:36:32 +02:00
empty_block.slot = slot
2019-06-09 20:41:21 +01:00
empty_block.body.eth1_data.deposit_count = state.eth1_deposit_index
2019-06-15 15:09:50 +01:00
previous_block_header = deepcopy(state.latest_block_header)
if previous_block_header.state_root == spec.Root():
2019-05-15 18:36:32 +02:00
previous_block_header.state_root = state.hash_tree_root()
2019-11-21 23:13:45 +01:00
empty_block.parent_root = hash_tree_root(previous_block_header)
apply_randao_reveal(spec, state, empty_block)
2019-05-15 18:36:32 +02:00
return empty_block
2019-11-21 23:13:45 +01:00
def build_empty_block_for_next_slot(spec, state):
return build_empty_block(spec, state, state.slot + 1)