package monitor import ( "context" "fmt" types "github.com/prysmaticlabs/eth2-types" "github.com/prysmaticlabs/prysm/beacon-chain/core/altair" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/state" "github.com/prysmaticlabs/prysm/config/params" "github.com/prysmaticlabs/prysm/encoding/bytesutil" ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1" "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/attestation" "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1/block" "github.com/prysmaticlabs/prysm/runtime/version" "github.com/prysmaticlabs/prysm/time/slots" "github.com/sirupsen/logrus" ) // updatedPerformanceFromTrackedVal returns true if the validator is tracked and if the // given slot is different than the last attested slot from this validator. // It assumes that a read lock is held on the monitor service. func (s *Service) updatedPerformanceFromTrackedVal(idx types.ValidatorIndex, slot types.Slot) bool { if !s.trackedIndex(idx) { return false } if lp, ok := s.latestPerformance[idx]; ok { return lp.attestedSlot != slot } return false } // attestingIndices returns the indices of validators that appear in the // given aggregated atestation. func attestingIndices(ctx context.Context, state state.BeaconState, att *ethpb.Attestation) ([]uint64, error) { committee, err := helpers.BeaconCommitteeFromState(ctx, state, att.Data.Slot, att.Data.CommitteeIndex) if err != nil { return nil, err } return attestation.AttestingIndices(att.AggregationBits, committee) } // logMessageTimelyFlagsForIndex returns the log message with the basic // performance indicators for the attestation (head, source, target) func logMessageTimelyFlagsForIndex(idx types.ValidatorIndex, data *ethpb.AttestationData) logrus.Fields { return logrus.Fields{ "ValidatorIndex": idx, "Slot": data.Slot, "Source": fmt.Sprintf("%#x", bytesutil.Trunc(data.Source.Root)), "Target": fmt.Sprintf("%#x", bytesutil.Trunc(data.Target.Root)), "Head": fmt.Sprintf("%#x", bytesutil.Trunc(data.BeaconBlockRoot)), } } // processAttestations logs the event that one of our tracked validators' // attestations was included in a block func (s *Service) processAttestations(ctx context.Context, state state.BeaconState, blk block.BeaconBlock) { if blk == nil || blk.Body() == nil { return } for _, attestation := range blk.Body().Attestations() { s.processIncludedAttestation(ctx, state, attestation) } } // processIncludedAttestation logs in the event that one of our tracked validators' // appears in the attesting indices and this included attestation was not included // before. func (s *Service) processIncludedAttestation(ctx context.Context, state state.BeaconState, att *ethpb.Attestation) { attestingIndices, err := attestingIndices(ctx, state, att) if err != nil { log.WithError(err).Error("Could not get attesting indices") return } s.Lock() defer s.Unlock() for _, idx := range attestingIndices { if s.updatedPerformanceFromTrackedVal(types.ValidatorIndex(idx), att.Data.Slot) { logFields := logMessageTimelyFlagsForIndex(types.ValidatorIndex(idx), att.Data) balance, err := state.BalanceAtIndex(types.ValidatorIndex(idx)) if err != nil { log.WithError(err).Error("Could not get balance") return } aggregatedPerf := s.aggregatedPerformance[types.ValidatorIndex(idx)] aggregatedPerf.totalAttestedCount++ aggregatedPerf.totalRequestedCount++ latestPerf := s.latestPerformance[types.ValidatorIndex(idx)] balanceChg := int64(balance - latestPerf.balance) latestPerf.balanceChange = balanceChg latestPerf.balance = balance latestPerf.attestedSlot = att.Data.Slot latestPerf.inclusionSlot = state.Slot() inclusionSlotGauge.WithLabelValues(fmt.Sprintf("%d", idx)).Set(float64(latestPerf.inclusionSlot)) aggregatedPerf.totalDistance += uint64(latestPerf.inclusionSlot - latestPerf.attestedSlot) if state.Version() == version.Altair { targetIdx := params.BeaconConfig().TimelyTargetFlagIndex sourceIdx := params.BeaconConfig().TimelySourceFlagIndex headIdx := params.BeaconConfig().TimelyHeadFlagIndex var participation []byte if slots.ToEpoch(latestPerf.inclusionSlot) == slots.ToEpoch(latestPerf.attestedSlot) { participation, err = state.CurrentEpochParticipation() if err != nil { log.WithError(err).Error("Could not get current epoch participation") return } } else { participation, err = state.PreviousEpochParticipation() if err != nil { log.WithError(err).Error("Could not get previous epoch participation") return } } flags := participation[idx] hasFlag, err := altair.HasValidatorFlag(flags, sourceIdx) if err != nil { log.WithError(err).Error("Could not get timely Source flag") return } latestPerf.timelySource = hasFlag hasFlag, err = altair.HasValidatorFlag(flags, headIdx) if err != nil { log.WithError(err).Error("Could not get timely Head flag") return } latestPerf.timelyHead = hasFlag hasFlag, err = altair.HasValidatorFlag(flags, targetIdx) if err != nil { log.WithError(err).Error("Could not get timely Target flag") return } latestPerf.timelyTarget = hasFlag if latestPerf.timelySource { timelySourceCounter.WithLabelValues(fmt.Sprintf("%d", idx)).Inc() aggregatedPerf.totalCorrectSource++ } if latestPerf.timelyHead { timelyHeadCounter.WithLabelValues(fmt.Sprintf("%d", idx)).Inc() aggregatedPerf.totalCorrectHead++ } if latestPerf.timelyTarget { timelyTargetCounter.WithLabelValues(fmt.Sprintf("%d", idx)).Inc() aggregatedPerf.totalCorrectTarget++ } } logFields["CorrectHead"] = latestPerf.timelyHead logFields["CorrectSource"] = latestPerf.timelySource logFields["CorrectTarget"] = latestPerf.timelyTarget logFields["InclusionSlot"] = latestPerf.inclusionSlot logFields["NewBalance"] = balance logFields["BalanceChange"] = balanceChg s.latestPerformance[types.ValidatorIndex(idx)] = latestPerf s.aggregatedPerformance[types.ValidatorIndex(idx)] = aggregatedPerf log.WithFields(logFields).Info("Attestation included") } } } // processUnaggregatedAttestation logs when the beacon node sees an unaggregated attestation from one of our // tracked validators func (s *Service) processUnaggregatedAttestation(ctx context.Context, att *ethpb.Attestation) { s.RLock() defer s.RUnlock() root := bytesutil.ToBytes32(att.Data.BeaconBlockRoot) state := s.config.StateGen.StateByRootIfCachedNoCopy(root) if state == nil { log.WithField("BeaconBlockRoot", fmt.Sprintf("%#x", bytesutil.Trunc(root[:]))).Debug( "Skipping unaggregated attestation due to state not found in cache") return } attestingIndices, err := attestingIndices(ctx, state, att) if err != nil { log.WithError(err).Error("Could not get attesting indices") return } for _, idx := range attestingIndices { if s.updatedPerformanceFromTrackedVal(types.ValidatorIndex(idx), att.Data.Slot) { logFields := logMessageTimelyFlagsForIndex(types.ValidatorIndex(idx), att.Data) log.WithFields(logFields).Info("Processed unaggregated attestation") } } } // processAggregatedAttestation logs when we see an aggregation from one of our tracked validators or an aggregated // attestation from one of our tracked validators func (s *Service) processAggregatedAttestation(ctx context.Context, att *ethpb.AggregateAttestationAndProof) { s.Lock() defer s.Unlock() if s.trackedIndex(att.AggregatorIndex) { log.WithFields(logrus.Fields{ "AggregatorIndex": att.AggregatorIndex, "Slot": att.Aggregate.Data.Slot, "BeaconBlockRoot": fmt.Sprintf("%#x", bytesutil.Trunc( att.Aggregate.Data.BeaconBlockRoot)), "SourceRoot:": fmt.Sprintf("%#x", bytesutil.Trunc( att.Aggregate.Data.Source.Root)), "TargetRoot:": fmt.Sprintf("%#x", bytesutil.Trunc( att.Aggregate.Data.Target.Root)), }).Info("Processed attestation aggregation") aggregatedPerf := s.aggregatedPerformance[att.AggregatorIndex] aggregatedPerf.totalAggregations++ s.aggregatedPerformance[att.AggregatorIndex] = aggregatedPerf aggregationCounter.WithLabelValues(fmt.Sprintf("%d", att.AggregatorIndex)).Inc() } var root [32]byte copy(root[:], att.Aggregate.Data.BeaconBlockRoot) state := s.config.StateGen.StateByRootIfCachedNoCopy(root) if state == nil { log.WithField("BeaconBlockRoot", fmt.Sprintf("%#x", bytesutil.Trunc(root[:]))).Debug( "Skipping agregated attestation due to state not found in cache") return } attestingIndices, err := attestingIndices(ctx, state, att.Aggregate) if err != nil { log.WithError(err).Error("Could not get attesting indices") return } for _, idx := range attestingIndices { if s.updatedPerformanceFromTrackedVal(types.ValidatorIndex(idx), att.Aggregate.Data.Slot) { logFields := logMessageTimelyFlagsForIndex(types.ValidatorIndex(idx), att.Aggregate.Data) log.WithFields(logFields).Info("Processed aggregated attestation") } } }