mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-07 02:02:18 +00:00
0b261cba5e
* unify fields * fix tests
155 lines
4.6 KiB
Go
155 lines
4.6 KiB
Go
package attestations
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"time"
|
|
|
|
"github.com/prysmaticlabs/go-bitfield"
|
|
"github.com/prysmaticlabs/prysm/v5/config/features"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
"github.com/prysmaticlabs/prysm/v5/crypto/hash"
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
attaggregation "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1/attestation/aggregation/attestations"
|
|
"github.com/prysmaticlabs/prysm/v5/time/slots"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// This prepares fork choice attestations by running batchForkChoiceAtts
|
|
// every prepareForkChoiceAttsPeriod.
|
|
func (s *Service) prepareForkChoiceAtts() {
|
|
intervals := features.Get().AggregateIntervals
|
|
slotDuration := time.Duration(params.BeaconConfig().SecondsPerSlot) * time.Second
|
|
// Adjust intervals for networks with a lower slot duration (Hive, e2e, etc)
|
|
for {
|
|
if intervals[len(intervals)-1] >= slotDuration {
|
|
for i, offset := range intervals {
|
|
intervals[i] = offset / 2
|
|
}
|
|
} else {
|
|
break
|
|
}
|
|
}
|
|
ticker := slots.NewSlotTickerWithIntervals(time.Unix(int64(s.genesisTime), 0), intervals[:])
|
|
for {
|
|
select {
|
|
case slotInterval := <-ticker.C():
|
|
t := time.Now()
|
|
if err := s.batchForkChoiceAtts(s.ctx); err != nil {
|
|
log.WithError(err).Error("Could not prepare attestations for fork choice")
|
|
}
|
|
switch slotInterval.Interval {
|
|
case 0:
|
|
duration := time.Since(t)
|
|
log.WithField("duration", duration).Debug("Aggregated unaggregated attestations")
|
|
batchForkChoiceAttsT1.Observe(float64(duration.Milliseconds()))
|
|
case 1:
|
|
batchForkChoiceAttsT2.Observe(float64(time.Since(t).Milliseconds()))
|
|
}
|
|
case <-s.ctx.Done():
|
|
log.Debug("Context closed, exiting routine")
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
// This gets the attestations from the unaggregated, aggregated and block
|
|
// pool. Then finds the common data, aggregate and batch them for fork choice.
|
|
// The resulting attestations are saved in the fork choice pool.
|
|
func (s *Service) batchForkChoiceAtts(ctx context.Context) error {
|
|
ctx, span := trace.StartSpan(ctx, "Operations.attestations.batchForkChoiceAtts")
|
|
defer span.End()
|
|
|
|
if err := s.cfg.Pool.AggregateUnaggregatedAttestations(ctx); err != nil {
|
|
return err
|
|
}
|
|
atts := append(s.cfg.Pool.AggregatedAttestations(), s.cfg.Pool.BlockAttestations()...)
|
|
atts = append(atts, s.cfg.Pool.ForkchoiceAttestations()...)
|
|
|
|
attsByDataRoot := make(map[[32]byte][]*ethpb.Attestation, len(atts))
|
|
|
|
// Consolidate attestations by aggregating them by similar data root.
|
|
for _, att := range atts {
|
|
seen, err := s.seen(att)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if seen {
|
|
continue
|
|
}
|
|
|
|
attDataRoot, err := att.Data.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
attsByDataRoot[attDataRoot] = append(attsByDataRoot[attDataRoot], att)
|
|
}
|
|
|
|
for _, atts := range attsByDataRoot {
|
|
if err := s.aggregateAndSaveForkChoiceAtts(atts); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
for _, a := range s.cfg.Pool.BlockAttestations() {
|
|
if err := s.cfg.Pool.DeleteBlockAttestation(a); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// This aggregates a list of attestations using the aggregation algorithm defined in AggregateAttestations
|
|
// and saves the attestations for fork choice.
|
|
func (s *Service) aggregateAndSaveForkChoiceAtts(atts []*ethpb.Attestation) error {
|
|
clonedAtts := make([]*ethpb.Attestation, len(atts))
|
|
for i, a := range atts {
|
|
clonedAtts[i] = ethpb.CopyAttestation(a)
|
|
}
|
|
aggregatedAtts, err := attaggregation.Aggregate(clonedAtts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.cfg.Pool.SaveForkchoiceAttestations(aggregatedAtts)
|
|
}
|
|
|
|
// This checks if the attestation has previously been aggregated for fork choice
|
|
// return true if yes, false if no.
|
|
func (s *Service) seen(att *ethpb.Attestation) (bool, error) {
|
|
attRoot, err := hash.Proto(att.Data)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
incomingBits := att.AggregationBits
|
|
savedBits, ok := s.forkChoiceProcessedRoots.Get(attRoot)
|
|
if ok {
|
|
savedBitlist, ok := savedBits.(bitfield.Bitlist)
|
|
if !ok {
|
|
return false, errors.New("not a bit field")
|
|
}
|
|
if savedBitlist.Len() == incomingBits.Len() {
|
|
// Returns true if the node has seen all the bits in the new bit field of the incoming attestation.
|
|
if bytes.Equal(savedBitlist, incomingBits) {
|
|
return true, nil
|
|
}
|
|
if c, err := savedBitlist.Contains(incomingBits); err != nil {
|
|
return false, err
|
|
} else if c {
|
|
return true, nil
|
|
}
|
|
var err error
|
|
// Update the bit fields by Or'ing them with the new ones.
|
|
incomingBits, err = incomingBits.Or(savedBitlist)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
}
|
|
|
|
s.forkChoiceProcessedRoots.Add(attRoot, incomingBits)
|
|
return false, nil
|
|
}
|