2020-02-27 17:22:39 +00:00
|
|
|
package detection
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2020-03-03 18:08:21 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
2020-02-27 17:22:39 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2020-03-03 18:08:21 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2020-02-27 17:22:39 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/sliceutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/slasher/detection/attestations/types"
|
2020-03-08 17:56:43 +00:00
|
|
|
"go.opencensus.io/trace"
|
2020-02-27 17:22:39 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func (ds *Service) detectAttesterSlashings(
|
|
|
|
ctx context.Context,
|
|
|
|
att *ethpb.IndexedAttestation,
|
|
|
|
) ([]*ethpb.AttesterSlashing, error) {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "detection.detectAttesterSlashings")
|
|
|
|
defer span.End()
|
2020-02-27 17:22:39 +00:00
|
|
|
slashings := make([]*ethpb.AttesterSlashing, 0)
|
|
|
|
for i := 0; i < len(att.AttestingIndices); i++ {
|
|
|
|
valIdx := att.AttestingIndices[i]
|
2020-03-03 18:08:21 +00:00
|
|
|
result, err := ds.minMaxSpanDetector.DetectSlashingForValidator(ctx, valIdx, att.Data)
|
2020-02-27 17:22:39 +00:00
|
|
|
if err != nil {
|
2020-03-03 18:08:21 +00:00
|
|
|
return nil, err
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
2020-03-03 18:08:21 +00:00
|
|
|
// If the response is nil, there was no slashing detected.
|
|
|
|
if result == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var slashing *ethpb.AttesterSlashing
|
|
|
|
switch result.Kind {
|
|
|
|
case types.DoubleVote:
|
|
|
|
slashing, err = ds.detectDoubleVote(ctx, 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, att, result)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "could not detect surround votes on attestation")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
slashings = append(slashings, slashing)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Clear out any duplicate slashings.
|
|
|
|
keys := make(map[[32]byte]bool)
|
|
|
|
var slashingList []*ethpb.AttesterSlashing
|
|
|
|
for _, ss := range slashings {
|
|
|
|
hash, err := hashutil.HashProto(ss)
|
2020-02-27 17:22:39 +00:00
|
|
|
if err != nil {
|
2020-03-03 18:08:21 +00:00
|
|
|
return nil, err
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
2020-03-03 18:08:21 +00:00
|
|
|
if _, value := keys[hash]; !value {
|
|
|
|
keys[hash] = true
|
|
|
|
slashingList = append(slashingList, ss)
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-03 18:08:21 +00:00
|
|
|
|
|
|
|
return slashingList, nil
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
|
|
|
|
2020-03-03 18:08:21 +00:00
|
|
|
// 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(
|
2020-02-27 17:22:39 +00:00
|
|
|
ctx context.Context,
|
2020-03-03 18:08:21 +00:00
|
|
|
incomingAtt *ethpb.IndexedAttestation,
|
|
|
|
detectionResult *types.DetectionResult,
|
|
|
|
) (*ethpb.AttesterSlashing, error) {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "detection.detectDoubleVote")
|
|
|
|
defer span.End()
|
2020-03-03 18:08:21 +00:00
|
|
|
if detectionResult == nil || detectionResult.Kind != types.DoubleVote {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
otherAtts, err := ds.slasherDB.IndexedAttestationsWithPrefix(ctx, detectionResult.SlashableEpoch, detectionResult.SigBytes[:])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, att := range otherAtts {
|
|
|
|
if att.Data == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// If there are no shared indices, there is no validator to slash.
|
|
|
|
if len(sliceutil.IntersectionUint64(att.AttestingIndices, incomingAtt.AttestingIndices)) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if isDoubleVote(incomingAtt, att) {
|
|
|
|
return ðpb.AttesterSlashing{
|
|
|
|
Attestation_1: incomingAtt,
|
|
|
|
Attestation_2: att,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
}
|
2020-03-08 17:56:43 +00:00
|
|
|
return nil, nil
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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,
|
|
|
|
incomingAtt *ethpb.IndexedAttestation,
|
2020-03-03 18:08:21 +00:00
|
|
|
detectionResult *types.DetectionResult,
|
|
|
|
) (*ethpb.AttesterSlashing, error) {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "detection.detectSurroundVotes")
|
|
|
|
defer span.End()
|
2020-03-03 18:08:21 +00:00
|
|
|
if detectionResult == nil || detectionResult.Kind != types.SurroundVote {
|
2020-02-27 17:22:39 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2020-03-03 18:08:21 +00:00
|
|
|
otherAtts, err := ds.slasherDB.IndexedAttestationsWithPrefix(ctx, detectionResult.SlashableEpoch, detectionResult.SigBytes[:])
|
2020-02-27 17:22:39 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for _, att := range otherAtts {
|
|
|
|
if att.Data == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
// If there are no shared indices, there is no validator to slash.
|
|
|
|
if len(sliceutil.IntersectionUint64(att.AttestingIndices, incomingAtt.AttestingIndices)) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2020-03-03 18:08:21 +00:00
|
|
|
// Slashings must be submitted as the incoming attestation surrounding the saved attestation.
|
|
|
|
// So we swap the order if needed.
|
|
|
|
if isSurrounding(incomingAtt, att) {
|
|
|
|
return ðpb.AttesterSlashing{
|
|
|
|
Attestation_1: incomingAtt,
|
|
|
|
Attestation_2: att,
|
|
|
|
}, nil
|
|
|
|
} else if isSurrounded(incomingAtt, att) {
|
|
|
|
return ðpb.AttesterSlashing{
|
2020-02-27 17:22:39 +00:00
|
|
|
Attestation_1: att,
|
|
|
|
Attestation_2: incomingAtt,
|
2020-03-03 18:08:21 +00:00
|
|
|
}, nil
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-03 18:08:21 +00:00
|
|
|
return nil, errors.New("unexpected false positive in surround vote detection")
|
|
|
|
}
|
|
|
|
|
|
|
|
func isDoubleVote(incomingAtt *ethpb.IndexedAttestation, prevAtt *ethpb.IndexedAttestation) bool {
|
|
|
|
return !proto.Equal(incomingAtt.Data, prevAtt.Data) && incomingAtt.Data.Target.Epoch == prevAtt.Data.Target.Epoch
|
2020-02-27 17:22:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func isSurrounding(incomingAtt *ethpb.IndexedAttestation, prevAtt *ethpb.IndexedAttestation) bool {
|
|
|
|
return incomingAtt.Data.Source.Epoch < prevAtt.Data.Source.Epoch &&
|
|
|
|
incomingAtt.Data.Target.Epoch > prevAtt.Data.Target.Epoch
|
|
|
|
}
|
|
|
|
|
|
|
|
func isSurrounded(incomingAtt *ethpb.IndexedAttestation, prevAtt *ethpb.IndexedAttestation) bool {
|
|
|
|
return incomingAtt.Data.Source.Epoch > prevAtt.Data.Source.Epoch &&
|
|
|
|
incomingAtt.Data.Target.Epoch < prevAtt.Data.Target.Epoch
|
|
|
|
}
|