2020-01-19 22:05:48 +00:00
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/gogo/protobuf/proto"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
slashpb "github.com/prysmaticlabs/prysm/proto/slashing"
|
2020-04-14 20:27:03 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-03-24 20:00:54 +00:00
|
|
|
bolt "go.etcd.io/bbolt"
|
2020-01-19 22:05:48 +00:00
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
|
|
|
func unmarshalAttestationHistory(enc []byte) (*slashpb.AttestationHistory, error) {
|
|
|
|
history := &slashpb.AttestationHistory{}
|
|
|
|
err := proto.Unmarshal(enc, history)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to unmarshal encoding")
|
|
|
|
}
|
|
|
|
return history, nil
|
|
|
|
}
|
|
|
|
|
2020-05-08 15:16:53 +00:00
|
|
|
// AttestationHistoryForPubKeys accepts an array of validator public keys and returns a mapping of corresponding attestation history.
|
|
|
|
func (db *Store) AttestationHistoryForPubKeys(ctx context.Context, publicKeys [][48]byte) (map[[48]byte]*slashpb.AttestationHistory, error) {
|
2020-01-19 22:05:48 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "Validator.AttestationHistory")
|
|
|
|
defer span.End()
|
|
|
|
|
2020-05-08 15:16:53 +00:00
|
|
|
if len(publicKeys) == 0 {
|
|
|
|
return make(map[[48]byte]*slashpb.AttestationHistory), nil
|
|
|
|
}
|
|
|
|
|
2020-01-19 22:05:48 +00:00
|
|
|
var err error
|
2020-05-08 15:16:53 +00:00
|
|
|
attestationHistoryForVals := make(map[[48]byte]*slashpb.AttestationHistory)
|
2020-01-19 22:05:48 +00:00
|
|
|
err = db.view(func(tx *bolt.Tx) error {
|
|
|
|
bucket := tx.Bucket(historicAttestationsBucket)
|
2020-05-08 15:16:53 +00:00
|
|
|
for _, key := range publicKeys {
|
|
|
|
enc := bucket.Get(key[:])
|
|
|
|
var attestationHistory *slashpb.AttestationHistory
|
|
|
|
if enc == nil {
|
|
|
|
newMap := make(map[uint64]uint64)
|
|
|
|
newMap[0] = params.BeaconConfig().FarFutureEpoch
|
|
|
|
attestationHistory = &slashpb.AttestationHistory{
|
|
|
|
TargetToSource: newMap,
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
attestationHistory, err = unmarshalAttestationHistory(enc)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-04-14 20:27:03 +00:00
|
|
|
}
|
2020-05-08 15:16:53 +00:00
|
|
|
attestationHistoryForVals[key] = attestationHistory
|
2020-01-19 22:05:48 +00:00
|
|
|
}
|
2020-05-08 15:16:53 +00:00
|
|
|
return nil
|
2020-01-19 22:05:48 +00:00
|
|
|
})
|
2020-05-08 15:16:53 +00:00
|
|
|
return attestationHistoryForVals, err
|
2020-01-19 22:05:48 +00:00
|
|
|
}
|
|
|
|
|
2020-05-08 15:16:53 +00:00
|
|
|
// SaveAttestationHistoryForPubKeys returns the attestation histories for the requested validator public keys.
|
|
|
|
func (db *Store) SaveAttestationHistoryForPubKeys(ctx context.Context, historyByPubKeys map[[48]byte]*slashpb.AttestationHistory) error {
|
2020-01-19 22:05:48 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "Validator.SaveAttestationHistory")
|
|
|
|
defer span.End()
|
|
|
|
|
2020-05-08 15:16:53 +00:00
|
|
|
encoded := make(map[[48]byte][]byte)
|
|
|
|
for pubKey, history := range historyByPubKeys {
|
|
|
|
enc, err := proto.Marshal(history)
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "failed to encode attestation history")
|
|
|
|
}
|
|
|
|
encoded[pubKey] = enc
|
2020-01-19 22:05:48 +00:00
|
|
|
}
|
|
|
|
|
2020-05-08 15:16:53 +00:00
|
|
|
err := db.update(func(tx *bolt.Tx) error {
|
2020-01-19 22:05:48 +00:00
|
|
|
bucket := tx.Bucket(historicAttestationsBucket)
|
2020-05-08 15:16:53 +00:00
|
|
|
for pubKey, encodedHistory := range encoded {
|
|
|
|
if err := bucket.Put(pubKey[:], encodedHistory); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2020-01-19 22:05:48 +00:00
|
|
|
})
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteAttestationHistory deletes the attestation history for the corresponding validator public key.
|
|
|
|
func (db *Store) DeleteAttestationHistory(ctx context.Context, pubkey []byte) error {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "Validator.DeleteAttestationHistory")
|
|
|
|
defer span.End()
|
|
|
|
|
|
|
|
return db.update(func(tx *bolt.Tx) error {
|
|
|
|
bucket := tx.Bucket(historicAttestationsBucket)
|
|
|
|
if err := bucket.Delete(pubkey); err != nil {
|
|
|
|
return errors.Wrap(err, "failed to delete the attestation history")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|