mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-13 21:48:19 +00:00
a04b7c2e4f
* WIP - slasher highest attestation start * fixed previous * highest source and target * highest attestation cache * cleanup * persist + fixes * PR fixes and cleanup * slashing proto * highest att. api * cleanup + tests * increased highest att. cache to 300K * removed highest att. api (for a separate PR) * fixed linting * bazel build fix * highest att. kv test * slasher highest att. test + purge + fix on eviction persist performance * cleanup + linting * linting + test fixes * bazel gazelle run * PR fixes * run goimports * go mod tidy * ineffectual assignment fix * run gazelle * bazel gazelle run * test fixes * linter fix * Apply suggestions from code review Co-authored-by: Shay Zluf <thezluf@gmail.com> * goimports run * cache tests * A bunch of small fixes * gazelle fix + gofmt * merge fixes * kv ordering fix * small typos and text fixes * capital letter fix Co-authored-by: Shay Zluf <thezluf@gmail.com>
252 lines
8.3 KiB
Go
252 lines
8.3 KiB
Go
package detection
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
|
|
"github.com/pkg/errors"
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
slashpb "github.com/prysmaticlabs/prysm/proto/slashing"
|
|
"github.com/prysmaticlabs/prysm/shared/attestationutil"
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
|
"github.com/prysmaticlabs/prysm/shared/sliceutil"
|
|
status "github.com/prysmaticlabs/prysm/slasher/db/types"
|
|
"github.com/prysmaticlabs/prysm/slasher/detection/attestations/types"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// DetectAttesterSlashings detects double, surround and surrounding attestation offences given an attestation.
|
|
func (ds *Service) DetectAttesterSlashings(
|
|
ctx context.Context,
|
|
att *ethpb.IndexedAttestation,
|
|
) ([]*ethpb.AttesterSlashing, error) {
|
|
ctx, span := trace.StartSpan(ctx, "detection.DetectAttesterSlashings")
|
|
defer span.End()
|
|
results, err := ds.minMaxSpanDetector.DetectSlashingsForAttestation(ctx, att)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
// If the response is nil, there was no slashing detected.
|
|
if len(results) == 0 {
|
|
return nil, nil
|
|
}
|
|
|
|
resultsToAtts, err := ds.mapResultsToAtts(ctx, results)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var slashings []*ethpb.AttesterSlashing
|
|
for _, result := range results {
|
|
resultKey := resultHash(result)
|
|
var slashing *ethpb.AttesterSlashing
|
|
switch result.Kind {
|
|
case types.DoubleVote:
|
|
slashing, err = ds.detectDoubleVote(ctx, resultsToAtts[resultKey], att, result)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not detect double votes on attestation")
|
|
}
|
|
case types.SurroundVote:
|
|
slashing, err = ds.detectSurroundVotes(ctx, resultsToAtts[resultKey], att, result)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not detect surround votes on attestation")
|
|
}
|
|
}
|
|
if slashing != nil {
|
|
slashings = append(slashings, slashing)
|
|
}
|
|
}
|
|
|
|
// Clear out any duplicate results.
|
|
keys := make(map[[32]byte]bool)
|
|
var slashingList []*ethpb.AttesterSlashing
|
|
for _, ss := range slashings {
|
|
hash, err := hashutil.HashProto(ss)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not hash slashing")
|
|
}
|
|
if _, value := keys[hash]; !value {
|
|
keys[hash] = true
|
|
slashingList = append(slashingList, ss)
|
|
}
|
|
}
|
|
if len(slashings) > 0 {
|
|
if err = ds.slasherDB.SaveAttesterSlashings(ctx, status.Active, slashings); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
return slashingList, nil
|
|
}
|
|
|
|
// UpdateSpans passthrough function that updates span maps given an indexed attestation.
|
|
func (ds *Service) UpdateSpans(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
|
return ds.minMaxSpanDetector.UpdateSpans(ctx, att)
|
|
}
|
|
|
|
// detectDoubleVote cross references the passed in attestation with the bloom filter maintained
|
|
// for every epoch for the validator in order to determine if it is a double vote.
|
|
func (ds *Service) detectDoubleVote(
|
|
_ context.Context,
|
|
possibleAtts []*ethpb.IndexedAttestation,
|
|
incomingAtt *ethpb.IndexedAttestation,
|
|
detectionResult *types.DetectionResult,
|
|
) (*ethpb.AttesterSlashing, error) {
|
|
if detectionResult == nil || detectionResult.Kind != types.DoubleVote {
|
|
return nil, nil
|
|
}
|
|
|
|
for _, att := range possibleAtts {
|
|
if att.Data == nil {
|
|
continue
|
|
}
|
|
|
|
if !isDoubleVote(incomingAtt, att) {
|
|
continue
|
|
}
|
|
|
|
// If there are no shared indices, there is no validator to slash.
|
|
if !sliceutil.IsInUint64(detectionResult.ValidatorIndex, att.AttestingIndices) {
|
|
continue
|
|
}
|
|
|
|
doubleVotesDetected.Inc()
|
|
return ðpb.AttesterSlashing{
|
|
Attestation_1: incomingAtt,
|
|
Attestation_2: att,
|
|
}, nil
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
// detectSurroundVotes cross references the passed in attestation with the requested validator's
|
|
// voting history in order to detect any possible surround votes.
|
|
func (ds *Service) detectSurroundVotes(
|
|
ctx context.Context,
|
|
possibleAtts []*ethpb.IndexedAttestation,
|
|
incomingAtt *ethpb.IndexedAttestation,
|
|
detectionResult *types.DetectionResult,
|
|
) (*ethpb.AttesterSlashing, error) {
|
|
ctx, span := trace.StartSpan(ctx, "detection.detectSurroundVotes")
|
|
defer span.End()
|
|
if detectionResult == nil || detectionResult.Kind != types.SurroundVote {
|
|
return nil, nil
|
|
}
|
|
|
|
for _, att := range possibleAtts {
|
|
if att.Data == nil {
|
|
continue
|
|
}
|
|
isSurround := isSurrounding(incomingAtt, att)
|
|
isSurrounded := isSurrounding(att, incomingAtt)
|
|
if !isSurround && !isSurrounded {
|
|
continue
|
|
}
|
|
// If there are no shared indices, there is no validator to slash.
|
|
if !sliceutil.IsInUint64(detectionResult.ValidatorIndex, att.AttestingIndices) {
|
|
continue
|
|
}
|
|
|
|
// Slashings must be submitted as the incoming attestation surrounding the saved attestation.
|
|
// So we swap the order if needed.
|
|
if isSurround {
|
|
surroundingVotesDetected.Inc()
|
|
return ðpb.AttesterSlashing{
|
|
Attestation_1: incomingAtt,
|
|
Attestation_2: att,
|
|
}, nil
|
|
} else if isSurrounded {
|
|
surroundedVotesDetected.Inc()
|
|
return ðpb.AttesterSlashing{
|
|
Attestation_1: att,
|
|
Attestation_2: incomingAtt,
|
|
}, nil
|
|
}
|
|
}
|
|
return nil, errors.New("unexpected false positive in surround vote detection")
|
|
}
|
|
|
|
// DetectDoubleProposals checks if the given signed beacon block is a slashable offense and returns the slashing.
|
|
func (ds *Service) DetectDoubleProposals(ctx context.Context, incomingBlock *ethpb.SignedBeaconBlockHeader) (*ethpb.ProposerSlashing, error) {
|
|
return ds.proposalsDetector.DetectDoublePropose(ctx, incomingBlock)
|
|
}
|
|
|
|
// DetectDoubleProposeNoUpdate checks if the given beacon block header is a slashable offense.
|
|
func (ds *Service) DetectDoubleProposeNoUpdate(ctx context.Context, incomingBlock *ethpb.BeaconBlockHeader) (bool, error) {
|
|
return ds.proposalsDetector.DetectDoubleProposeNoUpdate(ctx, incomingBlock)
|
|
}
|
|
|
|
// mapResultsToAtts handles any duplicate detections by ensuring they reuse the same pool of attestations, instead of re-checking the DB for the same data.
|
|
func (ds *Service) mapResultsToAtts(ctx context.Context, results []*types.DetectionResult) (map[[32]byte][]*ethpb.IndexedAttestation, error) {
|
|
ctx, span := trace.StartSpan(ctx, "detection.mapResultsToAtts")
|
|
defer span.End()
|
|
resultsToAtts := make(map[[32]byte][]*ethpb.IndexedAttestation)
|
|
for _, result := range results {
|
|
resultKey := resultHash(result)
|
|
if _, ok := resultsToAtts[resultKey]; ok {
|
|
continue
|
|
}
|
|
matchingAtts, err := ds.slasherDB.IndexedAttestationsWithPrefix(ctx, result.SlashableEpoch, result.SigBytes[:])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
resultsToAtts[resultKey] = matchingAtts
|
|
}
|
|
return resultsToAtts, nil
|
|
}
|
|
|
|
func resultHash(result *types.DetectionResult) [32]byte {
|
|
resultBytes := append(bytesutil.Bytes8(result.SlashableEpoch), result.SigBytes[:]...)
|
|
return hashutil.Hash(resultBytes)
|
|
}
|
|
|
|
func isDoublePropose(incomingBlockHeader, prevBlockHeader *ethpb.SignedBeaconBlockHeader) bool {
|
|
return incomingBlockHeader.Header.ProposerIndex == prevBlockHeader.Header.ProposerIndex &&
|
|
!bytes.Equal(incomingBlockHeader.Signature, prevBlockHeader.Signature) &&
|
|
incomingBlockHeader.Header.Slot == prevBlockHeader.Header.Slot
|
|
}
|
|
|
|
func isDoubleVote(incomingAtt, prevAtt *ethpb.IndexedAttestation) bool {
|
|
return !attestationutil.AttDataIsEqual(incomingAtt.Data, prevAtt.Data) && incomingAtt.Data.Target.Epoch == prevAtt.Data.Target.Epoch
|
|
}
|
|
|
|
func isSurrounding(incomingAtt, prevAtt *ethpb.IndexedAttestation) bool {
|
|
return incomingAtt.Data.Source.Epoch < prevAtt.Data.Source.Epoch &&
|
|
incomingAtt.Data.Target.Epoch > prevAtt.Data.Target.Epoch
|
|
}
|
|
|
|
// UpdateHighestAttestation updates to the db the highest source and target attestations for a each validator.
|
|
func (ds *Service) UpdateHighestAttestation(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
|
for _, idx := range att.AttestingIndices {
|
|
h, err := ds.slasherDB.HighestAttestation(ctx, idx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Creates a default instance.
|
|
if h == nil {
|
|
h = &slashpb.HighestAttestation{
|
|
HighestSourceEpoch: 0,
|
|
HighestTargetEpoch: 0,
|
|
ValidatorId: idx,
|
|
}
|
|
}
|
|
update := false
|
|
if h.HighestSourceEpoch < att.Data.Source.Epoch {
|
|
h.HighestSourceEpoch = att.Data.Source.Epoch
|
|
update = true
|
|
}
|
|
if h.HighestTargetEpoch < att.Data.Target.Epoch {
|
|
h.HighestTargetEpoch = att.Data.Target.Epoch
|
|
update = true
|
|
}
|
|
|
|
// If it's not a new instance of HighestAttestation, changing it will also change the cached instance.
|
|
if update {
|
|
if err := ds.slasherDB.SaveHighestAttestation(ctx, h); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|