2020-04-29 21:32:39 +00:00
|
|
|
// Package proposals defines an implementation of a double-propose
|
|
|
|
// detector in the slasher runtime.
|
2020-03-19 11:59:35 +00:00
|
|
|
package proposals
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
|
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/slasher/db"
|
|
|
|
status "github.com/prysmaticlabs/prysm/slasher/db/types"
|
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ProposeDetector defines a struct which can detect slashable
|
|
|
|
// block proposals.
|
|
|
|
type ProposeDetector struct {
|
|
|
|
slasherDB db.Database
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewProposeDetector creates a new instance of a struct.
|
|
|
|
func NewProposeDetector(db db.Database) *ProposeDetector {
|
|
|
|
return &ProposeDetector{
|
|
|
|
slasherDB: db,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DetectDoublePropose detects double proposals given a block by looking in the db.
|
|
|
|
func (dd *ProposeDetector) DetectDoublePropose(
|
|
|
|
ctx context.Context,
|
|
|
|
incomingBlk *ethpb.SignedBeaconBlockHeader,
|
|
|
|
) (*ethpb.ProposerSlashing, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "detector.DetectDoublePropose")
|
|
|
|
defer span.End()
|
2020-05-04 08:17:18 +00:00
|
|
|
headersFromIdx, err := dd.slasherDB.BlockHeaders(ctx, incomingBlk.Header.Slot, incomingBlk.Header.ProposerIndex)
|
2020-03-19 11:59:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-04 08:17:18 +00:00
|
|
|
for _, blockHeader := range headersFromIdx {
|
|
|
|
if bytes.Equal(blockHeader.Signature, incomingBlk.Signature) {
|
2020-03-19 11:59:35 +00:00
|
|
|
continue
|
|
|
|
}
|
2020-05-04 08:17:18 +00:00
|
|
|
ps := ðpb.ProposerSlashing{Header_1: incomingBlk, Header_2: blockHeader}
|
|
|
|
if err := dd.slasherDB.SaveProposerSlashing(ctx, status.Active, ps); err != nil {
|
2020-03-19 11:59:35 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return ps, nil
|
|
|
|
}
|
2020-05-08 05:51:56 +00:00
|
|
|
if err := dd.slasherDB.SaveBlockHeader(ctx, incomingBlk); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-19 11:59:35 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
2020-06-11 18:50:12 +00:00
|
|
|
|
|
|
|
// DetectDoubleProposeNoUpdate detects double proposals for a given block header by db search
|
|
|
|
// without storing the incoming block to db.
|
|
|
|
func (dd *ProposeDetector) DetectDoubleProposeNoUpdate(
|
|
|
|
ctx context.Context,
|
|
|
|
incomingBlk *ethpb.BeaconBlockHeader,
|
|
|
|
) (bool, error) {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "detector.DetectDoubleProposeNoUpdate")
|
|
|
|
defer span.End()
|
|
|
|
headersFromIdx, err := dd.slasherDB.BlockHeaders(ctx, incomingBlk.Slot, incomingBlk.ProposerIndex)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
for _, blockHeader := range headersFromIdx {
|
|
|
|
sameBodyRoot := bytes.Equal(blockHeader.Header.BodyRoot, incomingBlk.BodyRoot)
|
|
|
|
sameStateRoot := bytes.Equal(blockHeader.Header.StateRoot, incomingBlk.StateRoot)
|
|
|
|
sameParentRoot := bytes.Equal(blockHeader.Header.ParentRoot, incomingBlk.ParentRoot)
|
|
|
|
if sameBodyRoot && sameStateRoot && sameParentRoot {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return false, nil
|
|
|
|
}
|