2020-02-14 19:03:25 +00:00
|
|
|
package attestations
|
2019-11-18 19:49:39 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
2020-02-14 16:46:55 +00:00
|
|
|
"github.com/pkg/errors"
|
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2019-12-21 03:47:00 +00:00
|
|
|
slashpb "github.com/prysmaticlabs/prysm/proto/slashing"
|
2019-11-18 19:49:39 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-02-14 16:46:55 +00:00
|
|
|
"go.opencensus.io/trace"
|
2019-11-18 19:49:39 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Detector is a function type used to implement the slashable surrounding/surrounded
|
|
|
|
// vote detection methods.
|
2019-12-21 03:47:00 +00:00
|
|
|
type detectFn = func(attestationEpochSpan uint64, recorderEpochSpan *slashpb.MinMaxEpochSpan, sourceEpoch uint64) uint64
|
2019-11-18 19:49:39 +00:00
|
|
|
|
|
|
|
// detectMax is a function for maxDetector used to detect surrounding attestations.
|
2020-02-14 16:46:55 +00:00
|
|
|
func detectMax(attEpochSpan uint64, recorderEpochSpan *slashpb.MinMaxEpochSpan, attSourceEpoch uint64) uint64 {
|
2019-11-18 19:49:39 +00:00
|
|
|
maxSpan := uint64(recorderEpochSpan.MaxEpochSpan)
|
2020-02-14 16:46:55 +00:00
|
|
|
if maxSpan > attEpochSpan {
|
|
|
|
return maxSpan + attSourceEpoch
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2020-02-14 16:46:55 +00:00
|
|
|
// detectMin is a function for minDetector used to detect surrounded attestations.
|
|
|
|
func detectMin(attEpochSpan uint64, recorderEpochSpan *slashpb.MinMaxEpochSpan, attSourceEpoch uint64) uint64 {
|
2019-11-18 19:49:39 +00:00
|
|
|
minSpan := uint64(recorderEpochSpan.MinEpochSpan)
|
2020-02-14 16:46:55 +00:00
|
|
|
if minSpan > 0 && minSpan < attEpochSpan {
|
|
|
|
return minSpan + attSourceEpoch
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2020-02-14 16:46:55 +00:00
|
|
|
// DetectAndUpdateSpans runs detection and updating for both min and max epoch spans, this is used for
|
|
|
|
// attestation slashing detection.
|
|
|
|
// Detailed here: https://github.com/protolambda/eth2-surround/blob/master/README.md#min-max-surround
|
|
|
|
func DetectAndUpdateSpans(
|
|
|
|
ctx context.Context,
|
|
|
|
att *ethpb.IndexedAttestation,
|
|
|
|
spanMap *slashpb.EpochSpanMap,
|
|
|
|
) (*slashpb.EpochSpanMap, uint64, uint64, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "Detection.DetectAndUpdateSpans")
|
|
|
|
defer span.End()
|
|
|
|
minTargetEpoch, spanMap, err := detectAndUpdateMinEpochSpan(ctx, att.Data.Source.Epoch, att.Data.Target.Epoch, spanMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, 0, errors.Wrap(err, "failed to update min spans")
|
|
|
|
}
|
|
|
|
maxTargetEpoch, spanMap, err := detectAndUpdateMaxEpochSpan(ctx, att.Data.Source.Epoch, att.Data.Target.Epoch, spanMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, 0, errors.Wrap(err, "failed to update max spans")
|
|
|
|
}
|
|
|
|
return spanMap, minTargetEpoch, maxTargetEpoch, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// detectAndUpdateMaxEpochSpan is used to detect and update the max span of an incoming attestation.
|
2019-11-18 19:49:39 +00:00
|
|
|
// This is used for detecting surrounding votes.
|
|
|
|
// The max span is the span between the current attestation's source epoch and the furthest attestation's
|
|
|
|
// target epoch that has a lower (earlier) source epoch.
|
|
|
|
// Logic for this detection method was designed by https://github.com/protolambda
|
|
|
|
// Detailed here: https://github.com/protolambda/eth2-surround/blob/master/README.md#min-max-surround
|
2020-02-14 16:46:55 +00:00
|
|
|
func detectAndUpdateMaxEpochSpan(
|
2019-12-13 07:31:37 +00:00
|
|
|
ctx context.Context,
|
|
|
|
source uint64,
|
|
|
|
target uint64,
|
2019-12-21 03:47:00 +00:00
|
|
|
spanMap *slashpb.EpochSpanMap,
|
|
|
|
) (uint64, *slashpb.EpochSpanMap, error) {
|
2020-02-14 16:46:55 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "Detection.detectAndUpdateMaxEpochSpan")
|
|
|
|
defer span.End()
|
2019-11-21 07:11:23 +00:00
|
|
|
if target < source {
|
2020-02-14 16:46:55 +00:00
|
|
|
return 0, nil, fmt.Errorf("target: %d < source: %d ", target, source)
|
2019-11-21 07:11:23 +00:00
|
|
|
}
|
2020-02-14 16:46:55 +00:00
|
|
|
targetEpoch, minMaxSpan, spanMap, err := detectSlashingByEpochSpan(ctx, source, target, spanMap, detectMax)
|
2019-11-18 19:49:39 +00:00
|
|
|
if err != nil {
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, nil, err
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
if targetEpoch > 0 {
|
2019-12-04 06:39:38 +00:00
|
|
|
return targetEpoch, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
2020-02-04 18:48:51 +00:00
|
|
|
|
2019-11-18 19:49:39 +00:00
|
|
|
for i := uint64(1); i < target-source; i++ {
|
2020-02-14 16:46:55 +00:00
|
|
|
val := uint32(minMaxSpan - i)
|
2019-11-18 19:49:39 +00:00
|
|
|
if _, ok := spanMap.EpochSpanMap[source+i]; !ok {
|
2019-12-21 03:47:00 +00:00
|
|
|
spanMap.EpochSpanMap[source+i] = &slashpb.MinMaxEpochSpan{}
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
if spanMap.EpochSpanMap[source+i].MaxEpochSpan < val {
|
|
|
|
spanMap.EpochSpanMap[source+i].MaxEpochSpan = val
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
|
2020-02-14 16:46:55 +00:00
|
|
|
// detectAndUpdateMinEpochSpan is used to detect surrounded votes and update the min epoch span
|
2019-11-18 19:49:39 +00:00
|
|
|
// of an incoming attestation.
|
|
|
|
// The min span is the span between the current attestations target epoch and the
|
|
|
|
// closest attestation's target distance.
|
|
|
|
//
|
|
|
|
// Logic is following the detection method designed by https://github.com/protolambda
|
|
|
|
// Detailed here: https://github.com/protolambda/eth2-surround/blob/master/README.md#min-max-surround
|
2020-02-14 16:46:55 +00:00
|
|
|
func detectAndUpdateMinEpochSpan(
|
2019-12-13 07:31:37 +00:00
|
|
|
ctx context.Context,
|
|
|
|
source uint64,
|
|
|
|
target uint64,
|
2019-12-21 03:47:00 +00:00
|
|
|
spanMap *slashpb.EpochSpanMap,
|
|
|
|
) (uint64, *slashpb.EpochSpanMap, error) {
|
2020-02-14 16:46:55 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "Detection.detectAndUpdateMinEpochSpan")
|
|
|
|
defer span.End()
|
2019-11-21 07:11:23 +00:00
|
|
|
if target < source {
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, nil, fmt.Errorf(
|
2019-11-21 07:11:23 +00:00
|
|
|
"target: %d < source: %d ",
|
|
|
|
target,
|
|
|
|
source,
|
|
|
|
)
|
|
|
|
}
|
2020-02-14 16:46:55 +00:00
|
|
|
targetEpoch, _, spanMap, err := detectSlashingByEpochSpan(ctx, source, target, spanMap, detectMin)
|
2019-11-18 19:49:39 +00:00
|
|
|
if err != nil {
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, nil, err
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
if targetEpoch > 0 {
|
2019-12-04 06:39:38 +00:00
|
|
|
return targetEpoch, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
if source == 0 {
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
2020-02-04 18:48:51 +00:00
|
|
|
|
2019-11-18 19:49:39 +00:00
|
|
|
for i := source - 1; i > 0; i-- {
|
|
|
|
val := uint32(target - (i))
|
|
|
|
if _, ok := spanMap.EpochSpanMap[i]; !ok {
|
2019-12-21 03:47:00 +00:00
|
|
|
spanMap.EpochSpanMap[i] = &slashpb.MinMaxEpochSpan{}
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
if spanMap.EpochSpanMap[i].MinEpochSpan == 0 || spanMap.EpochSpanMap[i].MinEpochSpan > val {
|
|
|
|
spanMap.EpochSpanMap[i].MinEpochSpan = val
|
|
|
|
} else {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
2019-12-04 06:39:38 +00:00
|
|
|
return 0, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// detectSlashingByEpochSpan is used to detect if a slashable event is present
|
|
|
|
// in the db by checking either the closest attestation target or the furthest
|
|
|
|
// attestation target. This method receives a detector function in order to be used
|
|
|
|
// for both surrounding and surrounded vote cases.
|
2020-02-14 16:46:55 +00:00
|
|
|
func detectSlashingByEpochSpan(
|
|
|
|
ctx context.Context,
|
|
|
|
source uint64,
|
2019-12-13 07:31:37 +00:00
|
|
|
target uint64,
|
2019-12-21 03:47:00 +00:00
|
|
|
spanMap *slashpb.EpochSpanMap,
|
2019-12-13 07:31:37 +00:00
|
|
|
detector detectFn,
|
2019-12-21 03:47:00 +00:00
|
|
|
) (uint64, uint64, *slashpb.EpochSpanMap, error) {
|
2020-02-14 16:46:55 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "Detection.detectSlashingByEpochSpan")
|
|
|
|
defer span.End()
|
|
|
|
minMaxSpan := target - source
|
|
|
|
if minMaxSpan > params.BeaconConfig().WeakSubjectivityPeriod {
|
|
|
|
return 0, minMaxSpan, nil, fmt.Errorf(
|
|
|
|
"target: %d - source: %d > weakSubjectivityPeriod",
|
2019-11-18 19:49:39 +00:00
|
|
|
params.BeaconConfig().WeakSubjectivityPeriod,
|
2020-02-14 16:46:55 +00:00
|
|
|
minMaxSpan,
|
2019-11-18 19:49:39 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
if _, ok := spanMap.EpochSpanMap[source]; ok {
|
2020-02-14 16:46:55 +00:00
|
|
|
return detector(minMaxSpan, spanMap.EpochSpanMap[source], source), minMaxSpan, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|
2020-02-14 16:46:55 +00:00
|
|
|
return 0, minMaxSpan, spanMap, nil
|
2019-11-18 19:49:39 +00:00
|
|
|
}
|