2021-09-28 18:13:16 +00:00
|
|
|
package slasher
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
|
|
slashertypes "github.com/prysmaticlabs/prysm/beacon-chain/slasher/types"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/proto/prysm/v1alpha1"
|
2021-10-01 20:17:57 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/time/slots"
|
2021-09-28 18:13:16 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Receive indexed attestations from some source event feed,
|
|
|
|
// validating their integrity before appending them to an attestation queue
|
|
|
|
// for batch processing in a separate routine.
|
|
|
|
func (s *Service) receiveAttestations(ctx context.Context, indexedAttsChan chan *ethpb.IndexedAttestation) {
|
|
|
|
sub := s.serviceCfg.IndexedAttestationsFeed.Subscribe(indexedAttsChan)
|
|
|
|
defer sub.Unsubscribe()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case att := <-indexedAttsChan:
|
|
|
|
if !validateAttestationIntegrity(att) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
signingRoot, err := att.Data.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not get hash tree root of attestation")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
attWrapper := &slashertypes.IndexedAttestationWrapper{
|
|
|
|
IndexedAttestation: att,
|
|
|
|
SigningRoot: signingRoot,
|
|
|
|
}
|
|
|
|
s.attsQueue.push(attWrapper)
|
|
|
|
case err := <-sub.Err():
|
|
|
|
log.WithError(err).Debug("Subscriber closed with error")
|
|
|
|
return
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Receive beacon blocks from some source event feed,
|
|
|
|
func (s *Service) receiveBlocks(ctx context.Context, beaconBlockHeadersChan chan *ethpb.SignedBeaconBlockHeader) {
|
|
|
|
sub := s.serviceCfg.BeaconBlockHeadersFeed.Subscribe(beaconBlockHeadersChan)
|
|
|
|
defer sub.Unsubscribe()
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case blockHeader := <-beaconBlockHeadersChan:
|
|
|
|
if !validateBlockHeaderIntegrity(blockHeader) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
signingRoot, err := blockHeader.Header.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not get hash tree root of signed block header")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
wrappedProposal := &slashertypes.SignedBlockHeaderWrapper{
|
|
|
|
SignedBeaconBlockHeader: blockHeader,
|
|
|
|
SigningRoot: signingRoot,
|
|
|
|
}
|
|
|
|
s.blksQueue.push(wrappedProposal)
|
|
|
|
case err := <-sub.Err():
|
|
|
|
log.WithError(err).Debug("Subscriber closed with error")
|
|
|
|
return
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-06 21:45:38 +00:00
|
|
|
// Process queued attestations every time a slot ticker fires. We retrieve
|
2021-09-28 18:13:16 +00:00
|
|
|
// these attestations from a queue, then group them all by validator chunk index.
|
|
|
|
// This grouping will allow us to perform detection on batches of attestations
|
|
|
|
// per validator chunk index which can be done concurrently.
|
|
|
|
func (s *Service) processQueuedAttestations(ctx context.Context, slotTicker <-chan types.Slot) {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case currentSlot := <-slotTicker:
|
|
|
|
attestations := s.attsQueue.dequeue()
|
2021-10-01 20:17:57 +00:00
|
|
|
currentEpoch := slots.ToEpoch(currentSlot)
|
2021-09-28 18:13:16 +00:00
|
|
|
// We take all the attestations in the queue and filter out
|
|
|
|
// those which are valid now and valid in the future.
|
|
|
|
validAtts, validInFuture, numDropped := s.filterAttestations(attestations, currentEpoch)
|
|
|
|
|
|
|
|
deferredAttestationsTotal.Add(float64(len(validInFuture)))
|
|
|
|
droppedAttestationsTotal.Add(float64(numDropped))
|
|
|
|
|
|
|
|
// We add back those attestations that are valid in the future to the queue.
|
|
|
|
s.attsQueue.extend(validInFuture)
|
|
|
|
|
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"currentSlot": currentSlot,
|
|
|
|
"currentEpoch": currentEpoch,
|
|
|
|
"numValidAtts": len(validAtts),
|
|
|
|
"numDeferredAtts": len(validInFuture),
|
|
|
|
"numDroppedAtts": numDropped,
|
2021-12-06 21:45:38 +00:00
|
|
|
}).Info("Processing queued attestations for slashing detection")
|
2021-09-28 18:13:16 +00:00
|
|
|
|
|
|
|
// Save the attestation records to our database.
|
|
|
|
if err := s.serviceCfg.Database.SaveAttestationRecordsForValidators(
|
|
|
|
ctx, validAtts,
|
|
|
|
); err != nil {
|
|
|
|
log.WithError(err).Error("Could not save attestation records to DB")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for slashings.
|
2021-12-06 21:45:38 +00:00
|
|
|
slashings, err := s.checkSlashableAttestations(ctx, currentEpoch, validAtts)
|
2021-09-28 20:54:17 +00:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not check slashable attestations")
|
|
|
|
continue
|
|
|
|
}
|
2021-09-28 18:13:16 +00:00
|
|
|
|
|
|
|
// Process attester slashings by verifying their signatures, submitting
|
|
|
|
// to the beacon node's operations pool, and logging them.
|
|
|
|
if err := s.processAttesterSlashings(ctx, slashings); err != nil {
|
|
|
|
log.WithError(err).Error("Could not process attester slashings")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
processedAttestationsTotal.Add(float64(len(validAtts)))
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Process queued blocks every time an epoch ticker fires. We retrieve
|
|
|
|
// these blocks from a queue, then perform double proposal detection.
|
|
|
|
func (s *Service) processQueuedBlocks(ctx context.Context, slotTicker <-chan types.Slot) {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case currentSlot := <-slotTicker:
|
|
|
|
blocks := s.blksQueue.dequeue()
|
2021-10-01 20:17:57 +00:00
|
|
|
currentEpoch := slots.ToEpoch(currentSlot)
|
2021-09-28 18:13:16 +00:00
|
|
|
|
|
|
|
receivedBlocksTotal.Add(float64(len(blocks)))
|
|
|
|
|
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"currentSlot": currentSlot,
|
|
|
|
"currentEpoch": currentEpoch,
|
|
|
|
"numBlocks": len(blocks),
|
2021-12-06 21:45:38 +00:00
|
|
|
}).Info("Processing queued blocks for slashing detection")
|
2021-09-28 18:13:16 +00:00
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
// Check for slashings.
|
2021-09-28 19:33:45 +00:00
|
|
|
slashings, err := s.detectProposerSlashings(ctx, blocks)
|
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Error("Could not detect proposer slashings")
|
|
|
|
continue
|
|
|
|
}
|
2021-09-28 18:13:16 +00:00
|
|
|
|
|
|
|
// Process proposer slashings by verifying their signatures, submitting
|
|
|
|
// to the beacon node's operations pool, and logging them.
|
|
|
|
if err := s.processProposerSlashings(ctx, slashings); err != nil {
|
|
|
|
log.WithError(err).Error("Could not process proposer slashings")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.WithField("elapsed", time.Since(start)).Debug("Done checking slashable blocks")
|
|
|
|
|
|
|
|
processedBlocksTotal.Add(float64(len(blocks)))
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prunes slasher data on each slot tick to prevent unnecessary build-up of disk space usage.
|
|
|
|
func (s *Service) pruneSlasherData(ctx context.Context, slotTicker <-chan types.Slot) {
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-slotTicker:
|
2021-10-01 20:17:57 +00:00
|
|
|
headEpoch := slots.ToEpoch(s.serviceCfg.HeadStateFetcher.HeadSlot())
|
2021-09-28 18:13:16 +00:00
|
|
|
if err := s.pruneSlasherDataWithinSlidingWindow(ctx, headEpoch); err != nil {
|
|
|
|
log.WithError(err).Error("Could not prune slasher data")
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prunes slasher data by using a sliding window of [current_epoch - HISTORY_LENGTH, current_epoch].
|
|
|
|
// All data before that window is unnecessary for slasher, so can be periodically deleted.
|
|
|
|
// Say HISTORY_LENGTH is 4 and we have data for epochs 0, 1, 2, 3. Once we hit epoch 4, the sliding window
|
|
|
|
// we care about is 1, 2, 3, 4, so we can delete data for epoch 0.
|
|
|
|
func (s *Service) pruneSlasherDataWithinSlidingWindow(ctx context.Context, currentEpoch types.Epoch) error {
|
|
|
|
var maxPruningEpoch types.Epoch
|
|
|
|
if currentEpoch >= s.params.historyLength {
|
|
|
|
maxPruningEpoch = currentEpoch - s.params.historyLength
|
|
|
|
} else {
|
|
|
|
// If the current epoch is less than the history length, we should not
|
|
|
|
// attempt to prune at all.
|
|
|
|
return nil
|
|
|
|
}
|
2021-12-06 21:45:38 +00:00
|
|
|
start := time.Now()
|
2021-09-28 18:13:16 +00:00
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"currentEpoch": currentEpoch,
|
|
|
|
"pruningAllBeforeEpoch": maxPruningEpoch,
|
|
|
|
}).Info("Pruning old attestations and proposals for slasher")
|
|
|
|
numPrunedAtts, err := s.serviceCfg.Database.PruneAttestationsAtEpoch(
|
|
|
|
ctx, maxPruningEpoch,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Could not prune attestations")
|
|
|
|
}
|
|
|
|
numPrunedProposals, err := s.serviceCfg.Database.PruneProposalsAtEpoch(
|
|
|
|
ctx, maxPruningEpoch,
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "Could not prune proposals")
|
|
|
|
}
|
2021-12-07 14:38:19 +00:00
|
|
|
fields := logrus.Fields{}
|
|
|
|
if numPrunedAtts > 0 {
|
|
|
|
fields["numPrunedAtts"] = numPrunedAtts
|
|
|
|
}
|
|
|
|
if numPrunedProposals > 0 {
|
|
|
|
fields["numPrunedProposals"] = numPrunedProposals
|
|
|
|
}
|
|
|
|
fields["elapsed"] = time.Since(start)
|
|
|
|
log.WithFields(fields).Info("Done pruning old attestations and proposals for slasher")
|
2021-09-28 18:13:16 +00:00
|
|
|
return nil
|
|
|
|
}
|