2020-09-18 17:16:31 +00:00
|
|
|
package validator
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sort"
|
|
|
|
|
2021-03-08 18:46:24 +00:00
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
2020-09-18 17:16:31 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2021-03-08 18:46:24 +00:00
|
|
|
"github.com/prysmaticlabs/go-bitfield"
|
2020-09-18 17:16:31 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
2021-03-08 22:37:33 +00:00
|
|
|
iface "github.com/prysmaticlabs/prysm/beacon-chain/state/interface"
|
2021-03-08 18:46:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/aggregation"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
2020-09-18 17:16:31 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
)
|
|
|
|
|
|
|
|
type proposerAtts []*ethpb.Attestation
|
|
|
|
|
|
|
|
// filter separates attestation list into two groups: valid and invalid attestations.
|
|
|
|
// The first group passes the all the required checks for attestation to be considered for proposing.
|
|
|
|
// And attestations from the second group should be deleted.
|
2021-03-08 22:37:33 +00:00
|
|
|
func (a proposerAtts) filter(ctx context.Context, state iface.BeaconState) (proposerAtts, proposerAtts) {
|
2021-01-19 13:21:32 +00:00
|
|
|
validAtts := make([]*ethpb.Attestation, 0, len(a))
|
|
|
|
invalidAtts := make([]*ethpb.Attestation, 0, len(a))
|
|
|
|
for _, att := range a {
|
2020-12-16 18:06:49 +00:00
|
|
|
if _, err := blocks.ProcessAttestationNoVerifySignature(ctx, state, att); err == nil {
|
2020-09-18 17:16:31 +00:00
|
|
|
validAtts = append(validAtts, att)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
invalidAtts = append(invalidAtts, att)
|
|
|
|
}
|
|
|
|
return validAtts, invalidAtts
|
|
|
|
}
|
|
|
|
|
|
|
|
// sortByProfitability orders attestations by highest slot and by highest aggregation bit count.
|
2021-01-19 13:21:32 +00:00
|
|
|
func (a proposerAtts) sortByProfitability() proposerAtts {
|
|
|
|
if len(a) < 2 {
|
|
|
|
return a
|
2020-09-18 17:16:31 +00:00
|
|
|
}
|
2021-03-08 18:46:24 +00:00
|
|
|
if featureconfig.Get().ProposerAttsSelectionUsingMaxCover {
|
|
|
|
return a.sortByProfitabilityUsingMaxCover()
|
|
|
|
}
|
2021-01-19 13:21:32 +00:00
|
|
|
sort.Slice(a, func(i, j int) bool {
|
|
|
|
if a[i].Data.Slot == a[j].Data.Slot {
|
|
|
|
return a[i].AggregationBits.Count() > a[j].AggregationBits.Count()
|
2020-09-18 17:16:31 +00:00
|
|
|
}
|
2021-01-19 13:21:32 +00:00
|
|
|
return a[i].Data.Slot > a[j].Data.Slot
|
2020-09-18 17:16:31 +00:00
|
|
|
})
|
2021-01-19 13:21:32 +00:00
|
|
|
return a
|
2020-09-18 17:16:31 +00:00
|
|
|
}
|
|
|
|
|
2021-03-08 18:46:24 +00:00
|
|
|
// sortByProfitabilityUsingMaxCover orders attestations by highest slot and by highest aggregation bit count.
|
|
|
|
// Duplicate bits are counted only once, using max-cover algorithm.
|
|
|
|
func (a proposerAtts) sortByProfitabilityUsingMaxCover() proposerAtts {
|
|
|
|
// Separate attestations by slot, as slot number takes higher precedence when sorting.
|
|
|
|
var slots []types.Slot
|
|
|
|
attsBySlot := map[types.Slot]proposerAtts{}
|
|
|
|
for _, att := range a {
|
|
|
|
if _, ok := attsBySlot[att.Data.Slot]; !ok {
|
|
|
|
slots = append(slots, att.Data.Slot)
|
|
|
|
}
|
|
|
|
attsBySlot[att.Data.Slot] = append(attsBySlot[att.Data.Slot], att)
|
|
|
|
}
|
|
|
|
|
|
|
|
selectAtts := func(atts proposerAtts) proposerAtts {
|
|
|
|
if len(atts) < 2 {
|
|
|
|
return atts
|
|
|
|
}
|
|
|
|
candidates := make([]*bitfield.Bitlist64, len(atts))
|
|
|
|
for i := 0; i < len(atts); i++ {
|
|
|
|
candidates[i] = atts[i].AggregationBits.ToBitlist64()
|
|
|
|
}
|
|
|
|
// Add selected candidates on top, those that are not selected - append at bottom.
|
|
|
|
selectedKeys, _, err := aggregation.MaxCover(candidates, len(candidates), true /* allowOverlaps */)
|
|
|
|
if err == nil {
|
|
|
|
// Pick selected attestations first, leftover attestations will be appended at the end.
|
|
|
|
// Both lists will be sorted by number of bits set.
|
|
|
|
selectedAtts := make(proposerAtts, selectedKeys.Count())
|
|
|
|
leftoverAtts := make(proposerAtts, selectedKeys.Not().Count())
|
|
|
|
for i, key := range selectedKeys.BitIndices() {
|
|
|
|
selectedAtts[i] = atts[key]
|
|
|
|
}
|
|
|
|
for i, key := range selectedKeys.Not().BitIndices() {
|
|
|
|
leftoverAtts[i] = atts[key]
|
|
|
|
}
|
|
|
|
sort.Slice(selectedAtts, func(i, j int) bool {
|
|
|
|
return selectedAtts[i].AggregationBits.Count() > selectedAtts[j].AggregationBits.Count()
|
|
|
|
})
|
|
|
|
sort.Slice(leftoverAtts, func(i, j int) bool {
|
|
|
|
return leftoverAtts[i].AggregationBits.Count() > leftoverAtts[j].AggregationBits.Count()
|
|
|
|
})
|
|
|
|
return append(selectedAtts, leftoverAtts...)
|
|
|
|
}
|
|
|
|
return atts
|
|
|
|
}
|
|
|
|
|
|
|
|
// Select attestations. Slots are sorted from higher to lower at this point. Within slots attestations
|
|
|
|
// are sorted to maximize profitability (greedily selected, with previous attestations' bits
|
|
|
|
// evaluated before including any new attestation).
|
|
|
|
var sortedAtts proposerAtts
|
|
|
|
sort.Slice(slots, func(i, j int) bool {
|
|
|
|
return slots[i] > slots[j]
|
|
|
|
})
|
|
|
|
for _, slot := range slots {
|
|
|
|
sortedAtts = append(sortedAtts, selectAtts(attsBySlot[slot])...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return sortedAtts
|
|
|
|
}
|
|
|
|
|
2020-09-18 17:16:31 +00:00
|
|
|
// limitToMaxAttestations limits attestations to maximum attestations per block.
|
2021-01-19 13:21:32 +00:00
|
|
|
func (a proposerAtts) limitToMaxAttestations() proposerAtts {
|
|
|
|
if uint64(len(a)) > params.BeaconConfig().MaxAttestations {
|
|
|
|
return a[:params.BeaconConfig().MaxAttestations]
|
2020-09-18 17:16:31 +00:00
|
|
|
}
|
2021-01-19 13:21:32 +00:00
|
|
|
return a
|
2020-09-18 17:16:31 +00:00
|
|
|
}
|
2020-12-11 08:13:14 +00:00
|
|
|
|
|
|
|
// dedup removes duplicate attestations (ones with the same bits set on).
|
|
|
|
// Important: not only exact duplicates are removed, but proper subsets are removed too
|
|
|
|
// (their known bits are redundant and are already contained in their supersets).
|
2021-01-19 13:21:32 +00:00
|
|
|
func (a proposerAtts) dedup() proposerAtts {
|
|
|
|
if len(a) < 2 {
|
|
|
|
return a
|
2020-12-11 08:13:14 +00:00
|
|
|
}
|
2021-01-19 13:21:32 +00:00
|
|
|
attsByDataRoot := make(map[[32]byte][]*ethpb.Attestation, len(a))
|
|
|
|
for _, att := range a {
|
2020-12-11 08:13:14 +00:00
|
|
|
attDataRoot, err := att.Data.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
attsByDataRoot[attDataRoot] = append(attsByDataRoot[attDataRoot], att)
|
|
|
|
}
|
|
|
|
|
2021-01-19 13:21:32 +00:00
|
|
|
uniqAtts := make([]*ethpb.Attestation, 0, len(a))
|
2020-12-11 08:13:14 +00:00
|
|
|
for _, atts := range attsByDataRoot {
|
|
|
|
for i := 0; i < len(atts); i++ {
|
|
|
|
a := atts[i]
|
|
|
|
for j := i + 1; j < len(atts); j++ {
|
|
|
|
b := atts[j]
|
|
|
|
if a.AggregationBits.Contains(b.AggregationBits) {
|
|
|
|
// a contains b, b is redundant.
|
|
|
|
atts[j] = atts[len(atts)-1]
|
|
|
|
atts[len(atts)-1] = nil
|
|
|
|
atts = atts[:len(atts)-1]
|
|
|
|
j--
|
|
|
|
} else if b.AggregationBits.Contains(a.AggregationBits) {
|
|
|
|
// b contains a, a is redundant.
|
|
|
|
atts[i] = atts[len(atts)-1]
|
|
|
|
atts[len(atts)-1] = nil
|
|
|
|
atts = atts[:len(atts)-1]
|
|
|
|
i--
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
uniqAtts = append(uniqAtts, atts...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return uniqAtts
|
|
|
|
}
|