mirror of
https://gitlab.com/pulsechaincom/lighthouse-pulse.git
synced 2025-01-18 09:48:47 +00:00
op-pool: fix bug in attestation_score
The attestation scoring function was looking only at the previous epoch, but should really look at whichever epoch is appropriate for a given attestation. We also avoid including attestations that don't pay us any reward, as they simply bloat the chain.
This commit is contained in:
parent
64507950dd
commit
ddd9654f70
@ -74,15 +74,26 @@ impl AttestationId {
|
|||||||
/// the aggregate attestation introduces, and is proportional to the size of the reward we will
|
/// the aggregate attestation introduces, and is proportional to the size of the reward we will
|
||||||
/// receive for including it in a block.
|
/// receive for including it in a block.
|
||||||
// TODO: this could be optimised with a map from validator index to whether that validator has
|
// TODO: this could be optimised with a map from validator index to whether that validator has
|
||||||
// attested in the *current* epoch. Alternatively, we could cache an index that allows us to
|
// attested in each of the current and previous epochs. Currently quadractic in number of validators.
|
||||||
// quickly look up the attestations in the current epoch for a given shard.
|
fn attestation_score(attestation: &Attestation, state: &BeaconState, spec: &ChainSpec) -> usize {
|
||||||
fn attestation_score(attestation: &Attestation, state: &BeaconState) -> usize {
|
|
||||||
// Bitfield of validators whose attestations are new/fresh.
|
// Bitfield of validators whose attestations are new/fresh.
|
||||||
let mut new_validators = attestation.aggregation_bitfield.clone();
|
let mut new_validators = attestation.aggregation_bitfield.clone();
|
||||||
|
|
||||||
state
|
let attestation_epoch = attestation.data.slot.epoch(spec.slots_per_epoch);
|
||||||
.current_epoch_attestations
|
|
||||||
|
let state_attestations = if attestation_epoch == state.current_epoch(spec) {
|
||||||
|
&state.current_epoch_attestations
|
||||||
|
} else if attestation_epoch == state.previous_epoch(spec) {
|
||||||
|
&state.previous_epoch_attestations
|
||||||
|
} else {
|
||||||
|
return 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
state_attestations
|
||||||
.iter()
|
.iter()
|
||||||
|
// In a single epoch, an attester should only be attesting for one shard.
|
||||||
|
// TODO: we avoid including slashable attestations in the state here,
|
||||||
|
// but maybe we should do something else with them (like construct slashings).
|
||||||
.filter(|current_attestation| current_attestation.data.shard == attestation.data.shard)
|
.filter(|current_attestation| current_attestation.data.shard == attestation.data.shard)
|
||||||
.for_each(|current_attestation| {
|
.for_each(|current_attestation| {
|
||||||
// Remove the validators who have signed the existing attestation (they are not new)
|
// Remove the validators who have signed the existing attestation (they are not new)
|
||||||
@ -176,7 +187,9 @@ impl OperationPool {
|
|||||||
.filter(|attestation| validate_attestation(state, attestation, spec).is_ok())
|
.filter(|attestation| validate_attestation(state, attestation, spec).is_ok())
|
||||||
// Scored by the number of new attestations they introduce (descending)
|
// Scored by the number of new attestations they introduce (descending)
|
||||||
// TODO: need to consider attestations introduced in THIS block
|
// TODO: need to consider attestations introduced in THIS block
|
||||||
.map(|att| (att, attestation_score(att, state)))
|
.map(|att| (att, attestation_score(att, state, spec)))
|
||||||
|
// Don't include any useless attestations (score 0)
|
||||||
|
.filter(|&(_, score)| score != 0)
|
||||||
.sorted_by_key(|&(_, score)| std::cmp::Reverse(score))
|
.sorted_by_key(|&(_, score)| std::cmp::Reverse(score))
|
||||||
// Limited to the maximum number of attestations per block
|
// Limited to the maximum number of attestations per block
|
||||||
.take(spec.max_attestations as usize)
|
.take(spec.max_attestations as usize)
|
||||||
@ -695,7 +708,7 @@ mod tests {
|
|||||||
num_committees * (spec.slots_per_epoch * spec.target_committee_size) as usize;
|
num_committees * (spec.slots_per_epoch * spec.target_committee_size) as usize;
|
||||||
let mut state_builder =
|
let mut state_builder =
|
||||||
TestingBeaconStateBuilder::from_default_keypairs_file_if_exists(num_validators, spec);
|
TestingBeaconStateBuilder::from_default_keypairs_file_if_exists(num_validators, spec);
|
||||||
let slot_offset = 100000;
|
let slot_offset = 1000 * spec.slots_per_epoch + spec.slots_per_epoch / 2;
|
||||||
let slot = spec.genesis_slot + slot_offset;
|
let slot = spec.genesis_slot + slot_offset;
|
||||||
state_builder.teleport_to_slot(slot, spec);
|
state_builder.teleport_to_slot(slot, spec);
|
||||||
state_builder.build_caches(spec).unwrap();
|
state_builder.build_caches(spec).unwrap();
|
||||||
@ -726,7 +739,7 @@ mod tests {
|
|||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
att1.aggregation_bitfield.num_set_bits(),
|
att1.aggregation_bitfield.num_set_bits(),
|
||||||
attestation_score(&att1, state)
|
attestation_score(&att1, state, spec)
|
||||||
);
|
);
|
||||||
|
|
||||||
state
|
state
|
||||||
@ -735,7 +748,7 @@ mod tests {
|
|||||||
|
|
||||||
assert_eq!(
|
assert_eq!(
|
||||||
committee.committee.len() - 2,
|
committee.committee.len() - 2,
|
||||||
attestation_score(&att2, &state)
|
attestation_score(&att2, state, spec)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user