package attestations import ( "bytes" "context" "errors" "time" "github.com/prysmaticlabs/prysm/shared/copyutil" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/go-bitfield" attaggregation "github.com/prysmaticlabs/prysm/shared/aggregation/attestations" "github.com/prysmaticlabs/prysm/shared/hashutil" "github.com/prysmaticlabs/prysm/shared/slotutil" "go.opencensus.io/trace" ) // Prepare attestations for fork choice three times per slot. var prepareForkChoiceAttsPeriod = slotutil.DivideSlotBy(3 /* times-per-slot */) // This prepares fork choice attestations by running batchForkChoiceAtts // every prepareForkChoiceAttsPeriod. func (s *Service) prepareForkChoiceAtts() { ticker := time.NewTicker(prepareForkChoiceAttsPeriod) defer ticker.Stop() for { select { case <-ticker.C: if err := s.batchForkChoiceAtts(s.ctx); err != nil { log.WithError(err).Error("Could not prepare attestations for fork choice") } 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] = copyutil.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 := hashutil.HashProto(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) || savedBitlist.Contains(incomingBits) { return true, nil } // Update the bit fields by Or'ing them with the new ones. incomingBits = incomingBits.Or(savedBitlist) } } s.forkChoiceProcessedRoots.Add(attRoot, incomingBits) return false, nil }