mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-22 04:24:05 +00:00
Updates for finalizing chain! (#162)
* set epoch_boundary_root - chain finalizes! * fix slotStart to offset GENESIS_SLOT * work around bug that will be fixed by https://github.com/ethereum/eth2.0-specs/pull/732 * compile with debug info (there was a GC-related crash in C land)
This commit is contained in:
parent
0b47acae37
commit
aec31b1126
@ -265,31 +265,23 @@ proc makeAttestation(node: BeaconNode,
|
|||||||
var state = node.state.data
|
var state = node.state.data
|
||||||
skipSlots(state, node.state.blck.root, slot)
|
skipSlots(state, node.state.blck.root, slot)
|
||||||
|
|
||||||
|
# If we call makeAttestation too late, we must advance head only to `slot`
|
||||||
|
doAssert state.slot == slot,
|
||||||
|
"Corner case: head advanced beyond sheduled attestation slot"
|
||||||
|
|
||||||
let
|
let
|
||||||
justifiedBlockRoot =
|
attestationData = makeAttestationData(state, shard, node.state.blck.root)
|
||||||
get_block_root(state, get_epoch_start_slot(state.justified_epoch))
|
|
||||||
|
|
||||||
attestationData = AttestationData(
|
|
||||||
slot: slot,
|
|
||||||
shard: shard,
|
|
||||||
beacon_block_root: node.state.blck.root,
|
|
||||||
epoch_boundary_root: Eth2Digest(), # TODO
|
|
||||||
crosslink_data_root: Eth2Digest(), # TODO
|
|
||||||
latest_crosslink: state.latest_crosslinks[shard],
|
|
||||||
justified_epoch: state.justified_epoch,
|
|
||||||
justified_block_root: justifiedBlockRoot)
|
|
||||||
|
|
||||||
validatorSignature = await validator.signAttestation(attestationData)
|
validatorSignature = await validator.signAttestation(attestationData)
|
||||||
|
|
||||||
var participationBitfield = repeat(0'u8, ceil_div8(committeeLen))
|
var aggregationBitfield = repeat(0'u8, ceil_div8(committeeLen))
|
||||||
bitSet(participationBitfield, indexInCommittee)
|
bitSet(aggregationBitfield, indexInCommittee)
|
||||||
|
|
||||||
var attestation = Attestation(
|
var attestation = Attestation(
|
||||||
data: attestationData,
|
data: attestationData,
|
||||||
aggregate_signature: validatorSignature,
|
aggregate_signature: validatorSignature,
|
||||||
aggregation_bitfield: participationBitfield,
|
aggregation_bitfield: aggregationBitfield,
|
||||||
# Stub in phase0
|
# Stub in phase0
|
||||||
custody_bitfield: newSeq[byte](participationBitfield.len)
|
custody_bitfield: newSeq[byte](aggregationBitfield.len)
|
||||||
)
|
)
|
||||||
|
|
||||||
# TODO what are we waiting for here? broadcast should never block, and never
|
# TODO what are we waiting for here? broadcast should never block, and never
|
||||||
|
@ -247,6 +247,15 @@ proc add*(
|
|||||||
return true
|
return true
|
||||||
|
|
||||||
# This is an unresolved block - put it on the unresolved list for now...
|
# This is an unresolved block - put it on the unresolved list for now...
|
||||||
|
# TODO if we receive spam blocks, one heurestic to implement might be to wait
|
||||||
|
# for a couple of attestations to appear before fetching parents - this
|
||||||
|
# would help prevent using up network resources for spam - this serves
|
||||||
|
# two purposes: one is that attestations are likely to appear for the
|
||||||
|
# block only if it's valid / not spam - the other is that malicious
|
||||||
|
# validators that are not proposers can sign invalid blocks and send
|
||||||
|
# them out without penalty - but signing invalid attestations carries
|
||||||
|
# a risk of being slashed, making attestations a more valuable spam
|
||||||
|
# filter.
|
||||||
debug "Unresolved block",
|
debug "Unresolved block",
|
||||||
slot = humaneSlotNum(blck.slot),
|
slot = humaneSlotNum(blck.slot),
|
||||||
stateRoot = shortLog(blck.state_root),
|
stateRoot = shortLog(blck.state_root),
|
||||||
@ -322,7 +331,7 @@ proc maybePutState(pool: BlockPool, state: BeaconState) =
|
|||||||
if state.slot mod SLOTS_PER_EPOCH == 0:
|
if state.slot mod SLOTS_PER_EPOCH == 0:
|
||||||
info "Storing state",
|
info "Storing state",
|
||||||
stateSlot = humaneSlotNum(state.slot),
|
stateSlot = humaneSlotNum(state.slot),
|
||||||
stateRoot = hash_tree_root_final(state) # TODO cache?
|
stateRoot = shortLog(hash_tree_root_final(state)) # TODO cache?
|
||||||
pool.db.putState(state)
|
pool.db.putState(state)
|
||||||
|
|
||||||
proc updateState*(
|
proc updateState*(
|
||||||
|
@ -279,14 +279,15 @@ func get_attestation_participants*(state: BeaconState,
|
|||||||
## ``bitfield``.
|
## ``bitfield``.
|
||||||
let crosslink_committees = get_crosslink_committees_at_slot(
|
let crosslink_committees = get_crosslink_committees_at_slot(
|
||||||
state, attestation_data.slot)
|
state, attestation_data.slot)
|
||||||
|
doAssert anyIt(
|
||||||
assert anyIt(
|
|
||||||
crosslink_committees,
|
crosslink_committees,
|
||||||
it[1] == attestation_data.shard)
|
it[1] == attestation_data.shard)
|
||||||
let crosslink_committee = mapIt(
|
let crosslink_committee = mapIt(
|
||||||
filterIt(crosslink_committees, it.shard == attestation_data.shard),
|
filterIt(crosslink_committees, it.shard == attestation_data.shard),
|
||||||
it.committee)[0]
|
it.committee)[0]
|
||||||
|
|
||||||
|
# TODO this and other attestation-based fields need validation so we don't
|
||||||
|
# crash on a malicious attestation!
|
||||||
doAssert verify_bitfield(bitfield, len(crosslink_committee))
|
doAssert verify_bitfield(bitfield, len(crosslink_committee))
|
||||||
|
|
||||||
# Find the participating attesters in the committee
|
# Find the participating attesters in the committee
|
||||||
@ -493,3 +494,31 @@ func prepare_validator_for_withdrawal*(state: var BeaconState, index: ValidatorI
|
|||||||
# Bug in 0.3.0 spec; constant got renamed. Use 0.3.0 name.
|
# Bug in 0.3.0 spec; constant got renamed. Use 0.3.0 name.
|
||||||
validator.withdrawable_epoch = get_current_epoch(state) +
|
validator.withdrawable_epoch = get_current_epoch(state) +
|
||||||
MIN_VALIDATOR_WITHDRAWABILITY_DELAY
|
MIN_VALIDATOR_WITHDRAWABILITY_DELAY
|
||||||
|
|
||||||
|
# https://github.com/ethereum/eth2.0-specs/blob/0.4.0/specs/validator/0_beacon-chain-validator.md#attestations-1
|
||||||
|
proc makeAttestationData*(
|
||||||
|
state: BeaconState, shard: uint64,
|
||||||
|
beacon_block_root: Eth2Digest): AttestationData =
|
||||||
|
## Fine points:
|
||||||
|
## Head must be the head state during the slot that validator is
|
||||||
|
## part of committee - notably, it can't be a newer or older state (!)
|
||||||
|
|
||||||
|
# TODO update when https://github.com/ethereum/eth2.0-specs/issues/742
|
||||||
|
let
|
||||||
|
epoch_start_slot = get_epoch_start_slot(slot_to_epoch(state.slot))
|
||||||
|
epoch_boundary_root =
|
||||||
|
if epoch_start_slot == state.slot: beacon_block_root
|
||||||
|
else: get_block_root(state, epoch_start_slot)
|
||||||
|
justified_slot = get_epoch_start_slot(state.justified_epoch)
|
||||||
|
justified_block_root = get_block_root(state, justified_slot)
|
||||||
|
|
||||||
|
AttestationData(
|
||||||
|
slot: state.slot,
|
||||||
|
shard: shard,
|
||||||
|
beacon_block_root: beacon_block_root,
|
||||||
|
epoch_boundary_root: epoch_boundary_root,
|
||||||
|
crosslink_data_root: Eth2Digest(), # Stub in phase0
|
||||||
|
latest_crosslink: state.latest_crosslinks[shard],
|
||||||
|
justified_epoch: state.justified_epoch,
|
||||||
|
justified_block_root: justified_block_root,
|
||||||
|
)
|
||||||
|
@ -180,7 +180,7 @@ func get_current_epoch_committee_count*(state: BeaconState): uint64 =
|
|||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.3.0/specs/core/0_beacon-chain.md#get_current_epoch
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.3.0/specs/core/0_beacon-chain.md#get_current_epoch
|
||||||
func get_current_epoch*(state: BeaconState): Epoch =
|
func get_current_epoch*(state: BeaconState): Epoch =
|
||||||
# Return the current epoch of the given ``state``.
|
# Return the current epoch of the given ``state``.
|
||||||
doAssert state.slot >= GENESIS_SLOT, $state.slot
|
doAssert state.slot >= GENESIS_SLOT, $state.slot
|
||||||
slot_to_epoch(state.slot)
|
slot_to_epoch(state.slot)
|
||||||
|
|
||||||
# https://github.com/ethereum/eth2.0-specs/blob/v0.3.0/specs/core/0_beacon-chain.md#get_randao_mix
|
# https://github.com/ethereum/eth2.0-specs/blob/v0.3.0/specs/core/0_beacon-chain.md#get_randao_mix
|
||||||
|
@ -134,17 +134,20 @@ func get_crosslink_committees_at_slot*(state: BeaconState, slot: Slot,
|
|||||||
## ``slot`` in the next epoch -- with and without a `registry_change`
|
## ``slot`` in the next epoch -- with and without a `registry_change`
|
||||||
|
|
||||||
let
|
let
|
||||||
epoch = slot_to_epoch(slot)
|
# TODO: the + 1 here works around a bug, remove when upgrading to
|
||||||
|
# some more recent version:
|
||||||
|
# https://github.com/ethereum/eth2.0-specs/pull/732
|
||||||
|
epoch = slot_to_epoch(slot + 1)
|
||||||
current_epoch = get_current_epoch(state)
|
current_epoch = get_current_epoch(state)
|
||||||
previous_epoch = get_previous_epoch(state)
|
previous_epoch = get_previous_epoch(state)
|
||||||
next_epoch = current_epoch + 1
|
next_epoch = current_epoch + 1
|
||||||
|
|
||||||
assert previous_epoch <= epoch,
|
doAssert previous_epoch <= epoch,
|
||||||
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
||||||
", epoch: " & $humaneEpochNum(epoch) &
|
", epoch: " & $humaneEpochNum(epoch) &
|
||||||
", Next epoch: " & $humaneEpochNum(next_epoch)
|
", Next epoch: " & $humaneEpochNum(next_epoch)
|
||||||
|
|
||||||
assert epoch <= next_epoch,
|
doAssert epoch <= next_epoch,
|
||||||
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
"Previous epoch: " & $humaneEpochNum(previous_epoch) &
|
||||||
", epoch: " & $humaneEpochNum(epoch) &
|
", epoch: " & $humaneEpochNum(epoch) &
|
||||||
", Next epoch: " & $humaneEpochNum(next_epoch)
|
", Next epoch: " & $humaneEpochNum(next_epoch)
|
||||||
@ -170,7 +173,7 @@ func get_crosslink_committees_at_slot*(state: BeaconState, slot: Slot,
|
|||||||
shuffling_start_shard = state.current_shuffling_start_shard
|
shuffling_start_shard = state.current_shuffling_start_shard
|
||||||
(committees_per_epoch, seed, shuffling_epoch, shuffling_start_shard)
|
(committees_per_epoch, seed, shuffling_epoch, shuffling_start_shard)
|
||||||
else:
|
else:
|
||||||
assert epoch == next_epoch
|
doAssert epoch == next_epoch
|
||||||
|
|
||||||
let
|
let
|
||||||
current_committees_per_epoch = get_current_epoch_committee_count(state)
|
current_committees_per_epoch = get_current_epoch_committee_count(state)
|
||||||
|
@ -21,7 +21,7 @@ proc getSlotFromTime*(s: BeaconState, t = now()): Slot =
|
|||||||
int64(SECONDS_PER_SLOT * 1000))
|
int64(SECONDS_PER_SLOT * 1000))
|
||||||
|
|
||||||
func slotStart*(s: BeaconState, slot: Slot): Timestamp =
|
func slotStart*(s: BeaconState, slot: Slot): Timestamp =
|
||||||
(s.genesis_time + (slot * SECONDS_PER_SLOT)) * 1000
|
(s.genesis_time + ((slot - GENESIS_SLOT) * SECONDS_PER_SLOT)) * 1000
|
||||||
|
|
||||||
func slotMiddle*(s: BeaconState, slot: Slot): Timestamp =
|
func slotMiddle*(s: BeaconState, slot: Slot): Timestamp =
|
||||||
s.slotStart(slot) + SECONDS_PER_SLOT * 500
|
s.slotStart(slot) + SECONDS_PER_SLOT * 500
|
||||||
|
@ -37,7 +37,7 @@ if [ ! -f $LAST_VALIDATOR ]; then
|
|||||||
fi
|
fi
|
||||||
|
|
||||||
if [[ -z "$SKIP_BUILDS" ]]; then
|
if [[ -z "$SKIP_BUILDS" ]]; then
|
||||||
nim c -o:"$BEACON_NODE_BIN" $DEFS --opt:speed beacon_chain/beacon_node
|
nim c -o:"$BEACON_NODE_BIN" $DEFS --opt:speed --debuginfo beacon_chain/beacon_node
|
||||||
fi
|
fi
|
||||||
|
|
||||||
if [ ! -f $SNAPSHOT_FILE ]; then
|
if [ ! -f $SNAPSHOT_FILE ]; then
|
||||||
|
@ -168,17 +168,7 @@ proc makeAttestation*(
|
|||||||
get_crosslink_committees_at_slot(state, state.slot), validator_index)
|
get_crosslink_committees_at_slot(state, state.slot), validator_index)
|
||||||
validator = state.validator_registry[validator_index]
|
validator = state.validator_registry[validator_index]
|
||||||
sac_index = sac.committee.find(validator_index)
|
sac_index = sac.committee.find(validator_index)
|
||||||
|
data = makeAttestationData(state, sac.shard, beacon_block_root)
|
||||||
data = AttestationData(
|
|
||||||
slot: state.slot,
|
|
||||||
shard: sac.shard,
|
|
||||||
beacon_block_root: beacon_block_root,
|
|
||||||
epoch_boundary_root: Eth2Digest(), # TODO
|
|
||||||
latest_crosslink: state.latest_crosslinks[sac.shard],
|
|
||||||
crosslink_data_root: Eth2Digest(), # TODO
|
|
||||||
justified_epoch: state.justified_epoch,
|
|
||||||
justified_block_root: get_block_root(state, get_epoch_start_slot(state.justified_epoch)),
|
|
||||||
)
|
|
||||||
|
|
||||||
assert sac_index != -1, "find_shard_committe should guarantee this"
|
assert sac_index != -1, "find_shard_committe should guarantee this"
|
||||||
|
|
||||||
@ -187,7 +177,8 @@ proc makeAttestation*(
|
|||||||
bitSet(aggregation_bitfield, sac_index)
|
bitSet(aggregation_bitfield, sac_index)
|
||||||
|
|
||||||
let
|
let
|
||||||
msg = hash_tree_root_final(AttestationDataAndCustodyBit(data: data, custody_bit: false))
|
msg = hash_tree_root_final(
|
||||||
|
AttestationDataAndCustodyBit(data: data, custody_bit: false))
|
||||||
sig =
|
sig =
|
||||||
if skipValidation notin flags:
|
if skipValidation notin flags:
|
||||||
bls_sign(
|
bls_sign(
|
||||||
@ -203,7 +194,7 @@ proc makeAttestation*(
|
|||||||
data: data,
|
data: data,
|
||||||
aggregation_bitfield: aggregation_bitfield,
|
aggregation_bitfield: aggregation_bitfield,
|
||||||
aggregate_signature: sig,
|
aggregate_signature: sig,
|
||||||
custody_bitfield: repeat(0'u8, ceil_div8(sac.committee.len))
|
custody_bitfield: repeat(0'u8, aggregation_bitfield.len)
|
||||||
)
|
)
|
||||||
|
|
||||||
proc makeTestDB*(tailState: BeaconState, tailBlock: BeaconBlock): BeaconChainDB =
|
proc makeTestDB*(tailState: BeaconState, tailBlock: BeaconBlock): BeaconChainDB =
|
||||||
|
Loading…
x
Reference in New Issue
Block a user