2019-05-15 16:36:32 +00:00
|
|
|
from typing import List
|
|
|
|
|
2019-05-15 17:31:02 +00:00
|
|
|
from eth2spec.test.helpers.block import build_empty_block_for_next_slot, sign_block
|
2019-05-15 16:36:32 +00:00
|
|
|
from eth2spec.test.helpers.keys import privkeys
|
|
|
|
from eth2spec.utils.bls import bls_sign, bls_aggregate_signatures
|
2019-06-27 10:45:35 +00:00
|
|
|
from eth2spec.utils.ssz.ssz_typing import Bitlist
|
2019-05-15 16:36:32 +00:00
|
|
|
|
|
|
|
|
2019-10-12 03:16:13 +00:00
|
|
|
def build_attestation_data(spec, state, slot, index):
|
2019-05-15 16:36:32 +00:00
|
|
|
assert state.slot >= slot
|
|
|
|
|
|
|
|
if slot == state.slot:
|
2019-05-30 20:57:18 +00:00
|
|
|
block_root = build_empty_block_for_next_slot(spec, state).parent_root
|
2019-05-15 16:36:32 +00:00
|
|
|
else:
|
2019-05-30 20:57:18 +00:00
|
|
|
block_root = spec.get_block_root_at_slot(state, slot)
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-10-23 00:37:15 +00:00
|
|
|
current_epoch_start_slot = spec.compute_start_slot_at_epoch(spec.get_current_epoch(state))
|
2019-06-30 17:42:24 +00:00
|
|
|
if slot < current_epoch_start_slot:
|
2019-05-30 20:57:18 +00:00
|
|
|
epoch_boundary_root = spec.get_block_root(state, spec.get_previous_epoch(state))
|
2019-06-30 17:42:24 +00:00
|
|
|
elif slot == current_epoch_start_slot:
|
2019-05-15 16:36:32 +00:00
|
|
|
epoch_boundary_root = block_root
|
|
|
|
else:
|
2019-05-30 20:57:18 +00:00
|
|
|
epoch_boundary_root = spec.get_block_root(state, spec.get_current_epoch(state))
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-06-30 17:42:24 +00:00
|
|
|
if slot < current_epoch_start_slot:
|
2019-06-22 18:00:26 +00:00
|
|
|
source_epoch = state.previous_justified_checkpoint.epoch
|
|
|
|
source_root = state.previous_justified_checkpoint.root
|
2019-05-15 16:36:32 +00:00
|
|
|
else:
|
2019-06-22 18:00:26 +00:00
|
|
|
source_epoch = state.current_justified_checkpoint.epoch
|
|
|
|
source_root = state.current_justified_checkpoint.root
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-05-30 20:57:18 +00:00
|
|
|
return spec.AttestationData(
|
2019-10-12 03:16:13 +00:00
|
|
|
slot=slot,
|
2019-10-17 08:47:51 +00:00
|
|
|
index=index,
|
2019-05-15 16:36:32 +00:00
|
|
|
beacon_block_root=block_root,
|
2019-06-22 20:49:53 +00:00
|
|
|
source=spec.Checkpoint(epoch=source_epoch, root=source_root),
|
2019-10-17 08:47:51 +00:00
|
|
|
target=spec.Checkpoint(epoch=spec.compute_epoch_at_slot(slot), root=epoch_boundary_root),
|
2019-05-15 16:36:32 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-10-12 03:16:13 +00:00
|
|
|
def get_valid_attestation(spec, state, slot=None, index=None, signed=False):
|
2019-05-15 16:36:32 +00:00
|
|
|
if slot is None:
|
|
|
|
slot = state.slot
|
2019-10-12 03:16:13 +00:00
|
|
|
if index is None:
|
2019-10-16 09:47:19 +00:00
|
|
|
index = 0
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-10-12 03:16:13 +00:00
|
|
|
attestation_data = build_attestation_data(spec, state, slot, index)
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-10-17 01:45:07 +00:00
|
|
|
beacon_committee = spec.get_beacon_committee(
|
2019-05-27 17:09:52 +00:00
|
|
|
state,
|
2019-10-12 04:06:52 +00:00
|
|
|
attestation_data.slot,
|
2019-10-12 03:16:13 +00:00
|
|
|
attestation_data.index,
|
2019-05-27 17:09:52 +00:00
|
|
|
)
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-10-17 01:45:07 +00:00
|
|
|
committee_size = len(beacon_committee)
|
2019-06-29 17:21:13 +00:00
|
|
|
aggregation_bits = Bitlist[spec.MAX_VALIDATORS_PER_COMMITTEE](*([0] * committee_size))
|
2019-05-30 20:57:18 +00:00
|
|
|
attestation = spec.Attestation(
|
2019-06-28 11:23:22 +00:00
|
|
|
aggregation_bits=aggregation_bits,
|
2019-05-15 16:36:32 +00:00
|
|
|
data=attestation_data,
|
|
|
|
)
|
2019-05-30 20:57:18 +00:00
|
|
|
fill_aggregate_attestation(spec, state, attestation)
|
2019-05-15 17:47:50 +00:00
|
|
|
if signed:
|
2019-05-30 20:57:18 +00:00
|
|
|
sign_attestation(spec, state, attestation)
|
2019-05-15 16:36:32 +00:00
|
|
|
return attestation
|
|
|
|
|
|
|
|
|
2019-05-30 20:57:18 +00:00
|
|
|
def sign_aggregate_attestation(spec, state, attestation_data, participants: List[int]):
|
2019-05-15 16:36:32 +00:00
|
|
|
signatures = []
|
|
|
|
for validator_index in participants:
|
|
|
|
privkey = privkeys[validator_index]
|
|
|
|
signatures.append(
|
|
|
|
get_attestation_signature(
|
2019-05-31 08:41:39 +00:00
|
|
|
spec,
|
2019-05-15 16:36:32 +00:00
|
|
|
state,
|
2019-05-30 20:57:18 +00:00
|
|
|
attestation_data,
|
2019-05-15 16:36:32 +00:00
|
|
|
privkey
|
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
return bls_aggregate_signatures(signatures)
|
|
|
|
|
|
|
|
|
2019-05-30 20:57:18 +00:00
|
|
|
def sign_indexed_attestation(spec, state, indexed_attestation):
|
2019-11-02 03:02:53 +00:00
|
|
|
participants = indexed_attestation.attesting_indices
|
2019-05-30 20:57:18 +00:00
|
|
|
indexed_attestation.signature = sign_aggregate_attestation(spec, state, indexed_attestation.data, participants)
|
2019-05-15 16:36:32 +00:00
|
|
|
|
|
|
|
|
2019-05-30 20:57:18 +00:00
|
|
|
def sign_attestation(spec, state, attestation):
|
|
|
|
participants = spec.get_attesting_indices(
|
2019-05-15 16:36:32 +00:00
|
|
|
state,
|
|
|
|
attestation.data,
|
2019-06-28 11:23:22 +00:00
|
|
|
attestation.aggregation_bits,
|
2019-05-15 16:36:32 +00:00
|
|
|
)
|
|
|
|
|
2019-05-30 20:57:18 +00:00
|
|
|
attestation.signature = sign_aggregate_attestation(spec, state, attestation.data, participants)
|
2019-05-15 16:36:32 +00:00
|
|
|
|
|
|
|
|
2019-11-02 03:12:32 +00:00
|
|
|
def get_attestation_signature(spec, state, attestation_data, privkey):
|
2019-05-15 16:36:32 +00:00
|
|
|
return bls_sign(
|
2019-11-02 03:12:32 +00:00
|
|
|
message_hash=attestation_data.hash_tree_root(),
|
2019-05-15 16:36:32 +00:00
|
|
|
privkey=privkey,
|
2019-05-30 20:57:18 +00:00
|
|
|
domain=spec.get_domain(
|
2019-05-15 16:36:32 +00:00
|
|
|
state=state,
|
2019-09-22 19:27:42 +00:00
|
|
|
domain_type=spec.DOMAIN_BEACON_ATTESTER,
|
2019-06-22 20:49:53 +00:00
|
|
|
message_epoch=attestation_data.target.epoch,
|
2019-05-15 16:36:32 +00:00
|
|
|
)
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-09-20 21:05:10 +00:00
|
|
|
def fill_aggregate_attestation(spec, state, attestation, signed=False):
|
2019-10-28 06:27:41 +00:00
|
|
|
|
2019-10-17 01:45:07 +00:00
|
|
|
beacon_committee = spec.get_beacon_committee(
|
2019-05-27 17:09:52 +00:00
|
|
|
state,
|
2019-10-12 04:06:52 +00:00
|
|
|
attestation.data.slot,
|
2019-10-12 03:16:13 +00:00
|
|
|
attestation.data.index,
|
2019-05-27 17:09:52 +00:00
|
|
|
)
|
2019-10-17 01:45:07 +00:00
|
|
|
for i in range(len(beacon_committee)):
|
2019-06-28 11:23:22 +00:00
|
|
|
attestation.aggregation_bits[i] = True
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-09-20 21:05:10 +00:00
|
|
|
if signed:
|
|
|
|
sign_attestation(spec, state, attestation)
|
|
|
|
|
2019-05-15 16:36:32 +00:00
|
|
|
|
2019-09-20 16:27:42 +00:00
|
|
|
def add_attestations_to_state(spec, state, attestations, slot):
|
2019-05-30 20:57:18 +00:00
|
|
|
block = build_empty_block_for_next_slot(spec, state)
|
2019-05-15 16:36:32 +00:00
|
|
|
block.slot = slot
|
2019-09-20 16:27:42 +00:00
|
|
|
for attestation in attestations:
|
|
|
|
block.body.attestations.append(attestation)
|
2019-05-30 20:57:18 +00:00
|
|
|
spec.process_slots(state, block.slot)
|
|
|
|
sign_block(spec, state, block)
|
|
|
|
spec.state_transition(state, block)
|