2020-02-22 14:57:24 +00:00
|
|
|
package attestations
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
2020-03-10 19:41:55 +00:00
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
2020-02-22 14:57:24 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2020-03-09 18:14:19 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2020-02-22 14:57:24 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-03-05 18:11:54 +00:00
|
|
|
db "github.com/prysmaticlabs/prysm/slasher/db"
|
2020-02-27 04:48:02 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/slasher/detection/attestations/iface"
|
|
|
|
"github.com/prysmaticlabs/prysm/slasher/detection/attestations/types"
|
2020-02-22 14:57:24 +00:00
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
2020-03-10 19:41:55 +00:00
|
|
|
var (
|
|
|
|
latestMinSpanDistanceObserved = promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "latest_min_span_distance_observed",
|
|
|
|
Help: "The latest distance between target - source observed for min spans",
|
|
|
|
})
|
|
|
|
latestMaxSpanDistanceObserved = promauto.NewGauge(prometheus.GaugeOpts{
|
|
|
|
Name: "latest_max_span_distance_observed",
|
|
|
|
Help: "The latest distance between target - source observed for max spans",
|
|
|
|
})
|
|
|
|
)
|
|
|
|
|
2020-03-08 17:56:43 +00:00
|
|
|
// We look back 128 epochs when updating min/max spans
|
|
|
|
// for incoming attestations.
|
|
|
|
// TODO(#5040): Remove lookback and handle min spans properly.
|
|
|
|
const epochLookback = 128
|
|
|
|
|
2020-02-27 04:48:02 +00:00
|
|
|
var _ = iface.SpanDetector(&SpanDetector{})
|
2020-02-22 14:57:24 +00:00
|
|
|
|
|
|
|
// SpanDetector defines a struct which can detect slashable
|
|
|
|
// attestation offenses by tracking validator min-max
|
2020-02-27 22:21:05 +00:00
|
|
|
// spans from validators and attestation data roots.
|
2020-02-22 14:57:24 +00:00
|
|
|
type SpanDetector struct {
|
2020-03-05 18:11:54 +00:00
|
|
|
slasherDB db.Database
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewSpanDetector creates a new instance of a struct tracking
|
|
|
|
// several epochs of min-max spans for each validator in
|
|
|
|
// the beacon state.
|
2020-03-05 18:11:54 +00:00
|
|
|
func NewSpanDetector(db db.Database) *SpanDetector {
|
2020-02-22 14:57:24 +00:00
|
|
|
return &SpanDetector{
|
2020-03-05 18:11:54 +00:00
|
|
|
slasherDB: db,
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
// DetectSlashingsForAttestation uses a validator index and its corresponding
|
2020-02-22 14:57:24 +00:00
|
|
|
// min-max spans during an epoch to detect an epoch in which the validator
|
|
|
|
// committed a slashable attestation.
|
2020-03-09 18:14:19 +00:00
|
|
|
func (s *SpanDetector) DetectSlashingsForAttestation(
|
2020-02-22 14:57:24 +00:00
|
|
|
ctx context.Context,
|
2020-03-09 18:14:19 +00:00
|
|
|
att *ethpb.IndexedAttestation,
|
|
|
|
) ([]*types.DetectionResult, error) {
|
|
|
|
ctx, traceSpan := trace.StartSpan(ctx, "spanner.DetectSlashingsForAttestation")
|
2020-03-03 09:58:24 +00:00
|
|
|
defer traceSpan.End()
|
2020-03-09 18:14:19 +00:00
|
|
|
sourceEpoch := att.Data.Source.Epoch
|
|
|
|
targetEpoch := att.Data.Target.Epoch
|
2020-02-22 14:57:24 +00:00
|
|
|
if (targetEpoch - sourceEpoch) > params.BeaconConfig().WeakSubjectivityPeriod {
|
|
|
|
return nil, fmt.Errorf(
|
|
|
|
"attestation span was greater than weak subjectivity period %d, received: %d",
|
|
|
|
params.BeaconConfig().WeakSubjectivityPeriod,
|
|
|
|
targetEpoch-sourceEpoch,
|
|
|
|
)
|
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
|
|
|
|
spanMap, err := s.slasherDB.EpochSpansMap(ctx, sourceEpoch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
targetSpanMap, err := s.slasherDB.EpochSpansMap(ctx, targetEpoch)
|
2020-03-05 18:11:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-02-22 14:57:24 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
var detections []*types.DetectionResult
|
|
|
|
distance := uint16(targetEpoch - sourceEpoch)
|
|
|
|
for _, idx := range att.AttestingIndices {
|
|
|
|
span := spanMap[idx]
|
|
|
|
minSpan := span.MinSpan
|
|
|
|
if minSpan > 0 && minSpan < distance {
|
|
|
|
slashableEpoch := sourceEpoch + uint64(minSpan)
|
|
|
|
targetSpan, err := s.slasherDB.EpochSpanByValidatorIndex(ctx, idx, slashableEpoch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
detections = append(detections, &types.DetectionResult{
|
|
|
|
Kind: types.SurroundVote,
|
|
|
|
SlashableEpoch: slashableEpoch,
|
|
|
|
SigBytes: targetSpan.SigBytes,
|
|
|
|
})
|
|
|
|
continue
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
2020-02-27 22:21:05 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
maxSpan := span.MaxSpan
|
|
|
|
if maxSpan > distance {
|
|
|
|
slashableEpoch := sourceEpoch + uint64(maxSpan)
|
|
|
|
targetSpan, err := s.slasherDB.EpochSpanByValidatorIndex(ctx, idx, slashableEpoch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
detections = append(detections, &types.DetectionResult{
|
|
|
|
Kind: types.SurroundVote,
|
|
|
|
SlashableEpoch: slashableEpoch,
|
|
|
|
SigBytes: targetSpan.SigBytes,
|
|
|
|
})
|
|
|
|
continue
|
2020-02-27 22:21:05 +00:00
|
|
|
}
|
2020-03-08 17:56:43 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
targetSpan := targetSpanMap[idx]
|
|
|
|
// Check if the validator has attested for this epoch or not.
|
|
|
|
if targetSpan.HasAttested {
|
|
|
|
detections = append(detections, &types.DetectionResult{
|
|
|
|
Kind: types.DoubleVote,
|
|
|
|
SlashableEpoch: targetEpoch,
|
|
|
|
SigBytes: targetSpan.SigBytes,
|
|
|
|
})
|
|
|
|
continue
|
|
|
|
}
|
2020-03-05 18:11:54 +00:00
|
|
|
}
|
2020-02-27 22:21:05 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
// Clear out any duplicate results.
|
|
|
|
keys := make(map[[32]byte]bool)
|
|
|
|
var detectionList []*types.DetectionResult
|
|
|
|
for _, dd := range detections {
|
|
|
|
hash := hashutil.Hash(dd.Marshal())
|
|
|
|
if _, value := keys[hash]; !value {
|
|
|
|
keys[hash] = true
|
|
|
|
detectionList = append(detectionList, dd)
|
|
|
|
}
|
2020-02-27 22:21:05 +00:00
|
|
|
}
|
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
return detectionList, nil
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateSpans given an indexed attestation for all of its attesting indices.
|
|
|
|
func (s *SpanDetector) UpdateSpans(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "spanner.UpdateSpans")
|
2020-02-22 14:57:24 +00:00
|
|
|
defer span.End()
|
2020-03-09 18:14:19 +00:00
|
|
|
// Save the signature for the received attestation so we can have more detail to find it in the DB.
|
|
|
|
if err := s.saveSigBytes(ctx, att); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// Update min and max spans.
|
|
|
|
if err := s.updateMinSpan(ctx, att); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := s.updateMaxSpan(ctx, att); err != nil {
|
|
|
|
return err
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-03-03 09:58:24 +00:00
|
|
|
// saveSigBytes saves the first 2 bytes of the signature for the att we're updating the spans to.
|
|
|
|
// Later used to help us find the violating attestation in the DB.
|
2020-03-09 18:14:19 +00:00
|
|
|
func (s *SpanDetector) saveSigBytes(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, traceSpan := trace.StartSpan(ctx, "spanner.saveSigBytes")
|
|
|
|
defer traceSpan.End()
|
2020-03-03 09:58:24 +00:00
|
|
|
target := att.Data.Target.Epoch
|
2020-03-09 18:14:19 +00:00
|
|
|
spanMap, err := s.slasherDB.EpochSpansMap(ctx, target)
|
2020-03-05 18:11:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-02-27 22:21:05 +00:00
|
|
|
}
|
2020-03-05 18:11:54 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
// We loop through the indices, instead of constantly locking/unlocking the cache for equivalent accesses.
|
|
|
|
for _, idx := range att.AttestingIndices {
|
|
|
|
span := spanMap[idx]
|
|
|
|
// If the validator has already attested for this target epoch,
|
|
|
|
// then we do not need to update the values of the span sig bytes.
|
|
|
|
if span.HasAttested {
|
|
|
|
return nil
|
|
|
|
}
|
2020-02-27 22:21:05 +00:00
|
|
|
|
2020-03-09 18:14:19 +00:00
|
|
|
sigBytes := [2]byte{0, 0}
|
|
|
|
if len(att.Signature) > 1 {
|
|
|
|
sigBytes = [2]byte{att.Signature[0], att.Signature[1]}
|
|
|
|
}
|
|
|
|
// Save the signature bytes into the span for this epoch.
|
|
|
|
span.HasAttested = true
|
|
|
|
span.SigBytes = sigBytes
|
|
|
|
spanMap[idx] = span
|
2020-02-27 22:21:05 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
return s.slasherDB.SaveEpochSpansMap(ctx, target, spanMap)
|
2020-02-27 22:21:05 +00:00
|
|
|
}
|
|
|
|
|
2020-02-22 14:57:24 +00:00
|
|
|
// Updates a min span for a validator index given a source and target epoch
|
2020-02-26 03:35:34 +00:00
|
|
|
// for an attestation produced by the validator. Used for catching surrounding votes.
|
2020-03-09 18:14:19 +00:00
|
|
|
func (s *SpanDetector) updateMinSpan(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, traceSpan := trace.StartSpan(ctx, "spanner.updateMinSpan")
|
|
|
|
defer traceSpan.End()
|
2020-03-09 18:14:19 +00:00
|
|
|
source := att.Data.Source.Epoch
|
|
|
|
target := att.Data.Target.Epoch
|
2020-02-26 03:35:34 +00:00
|
|
|
if source < 1 {
|
2020-03-05 18:11:54 +00:00
|
|
|
return nil
|
2020-02-26 03:35:34 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
valIndices := make([]uint64, len(att.AttestingIndices))
|
|
|
|
copy(valIndices, att.AttestingIndices)
|
2020-03-08 17:56:43 +00:00
|
|
|
lowestEpoch := source - epochLookback
|
|
|
|
if int(lowestEpoch) <= 0 {
|
|
|
|
lowestEpoch = 0
|
|
|
|
}
|
2020-03-10 19:41:55 +00:00
|
|
|
latestMinSpanDistanceObserved.Set(float64(att.Data.Target.Epoch - att.Data.Source.Epoch))
|
2020-03-09 18:14:19 +00:00
|
|
|
|
2020-03-08 17:56:43 +00:00
|
|
|
for epoch := source - 1; epoch >= lowestEpoch; epoch-- {
|
2020-03-09 18:14:19 +00:00
|
|
|
spanMap, err := s.slasherDB.EpochSpansMap(ctx, epoch)
|
2020-03-05 18:11:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-02-26 03:35:34 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
indices := valIndices[:0]
|
|
|
|
for _, idx := range valIndices {
|
|
|
|
span := spanMap[idx]
|
|
|
|
newMinSpan := uint16(target - epoch)
|
|
|
|
if span.MinSpan == 0 || span.MinSpan > newMinSpan {
|
|
|
|
span = types.Span{
|
|
|
|
MinSpan: newMinSpan,
|
|
|
|
MaxSpan: span.MaxSpan,
|
|
|
|
SigBytes: span.SigBytes,
|
|
|
|
HasAttested: span.HasAttested,
|
|
|
|
}
|
|
|
|
spanMap[idx] = span
|
|
|
|
indices = append(indices, idx)
|
2020-03-05 18:11:54 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
}
|
|
|
|
if err := s.slasherDB.SaveEpochSpansMap(ctx, epoch, spanMap); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(indices) == 0 {
|
2020-02-26 03:35:34 +00:00
|
|
|
break
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
2020-03-08 17:56:43 +00:00
|
|
|
if epoch == 0 {
|
|
|
|
break
|
|
|
|
}
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
2020-03-05 18:11:54 +00:00
|
|
|
return nil
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Updates a max span for a validator index given a source and target epoch
|
2020-02-26 03:35:34 +00:00
|
|
|
// for an attestation produced by the validator. Used for catching surrounded votes.
|
2020-03-09 18:14:19 +00:00
|
|
|
func (s *SpanDetector) updateMaxSpan(ctx context.Context, att *ethpb.IndexedAttestation) error {
|
2020-03-08 17:56:43 +00:00
|
|
|
ctx, traceSpan := trace.StartSpan(ctx, "spanner.updateMaxSpan")
|
|
|
|
defer traceSpan.End()
|
2020-03-09 18:14:19 +00:00
|
|
|
source := att.Data.Source.Epoch
|
|
|
|
target := att.Data.Target.Epoch
|
2020-03-10 19:41:55 +00:00
|
|
|
latestMaxSpanDistanceObserved.Set(float64(target-source))
|
2020-03-09 18:14:19 +00:00
|
|
|
valIndices := make([]uint64, len(att.AttestingIndices))
|
|
|
|
copy(valIndices, att.AttestingIndices)
|
2020-02-26 03:35:34 +00:00
|
|
|
for epoch := source + 1; epoch < target; epoch++ {
|
2020-03-09 18:14:19 +00:00
|
|
|
spanMap, err := s.slasherDB.EpochSpansMap(ctx, epoch)
|
2020-03-05 18:11:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
indices := valIndices[:0]
|
|
|
|
for _, idx := range valIndices {
|
|
|
|
span := spanMap[idx]
|
|
|
|
newMaxSpan := uint16(target - epoch)
|
|
|
|
if newMaxSpan > span.MaxSpan {
|
|
|
|
span = types.Span{
|
|
|
|
MinSpan: span.MinSpan,
|
|
|
|
MaxSpan: newMaxSpan,
|
|
|
|
SigBytes: span.SigBytes,
|
|
|
|
HasAttested: span.HasAttested,
|
|
|
|
}
|
|
|
|
spanMap[idx] = span
|
|
|
|
indices = append(indices, idx)
|
2020-03-05 18:11:54 +00:00
|
|
|
}
|
2020-03-09 18:14:19 +00:00
|
|
|
}
|
|
|
|
if err := s.slasherDB.SaveEpochSpansMap(ctx, epoch, spanMap); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if len(indices) == 0 {
|
2020-02-22 14:57:24 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2020-03-05 18:11:54 +00:00
|
|
|
return nil
|
2020-02-22 14:57:24 +00:00
|
|
|
}
|