2018-10-05 17:14:50 +00:00
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
2019-03-17 02:56:05 +00:00
|
|
|
"context"
|
2018-10-17 06:11:24 +00:00
|
|
|
"fmt"
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
"github.com/boltdb/bolt"
|
2018-10-05 17:14:50 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-18 23:34:49 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2019-03-17 02:56:05 +00:00
|
|
|
"go.opencensus.io/trace"
|
2018-10-05 17:14:50 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// SaveAttestation puts the attestation record into the beacon chain db.
|
2019-03-17 02:56:05 +00:00
|
|
|
func (db *BeaconDB) SaveAttestation(ctx context.Context, attestation *pb.Attestation) error {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "beaconDB.SaveAttestation")
|
|
|
|
defer span.End()
|
|
|
|
|
2018-12-22 20:30:59 +00:00
|
|
|
encodedState, err := proto.Marshal(attestation)
|
2018-10-05 17:14:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-02-18 23:34:49 +00:00
|
|
|
hash := hashutil.Hash(encodedState)
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
return db.update(func(tx *bolt.Tx) error {
|
|
|
|
a := tx.Bucket(attestationBucket)
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
return a.Put(hash[:], encodedState)
|
|
|
|
})
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|
|
|
|
|
2019-02-15 19:27:45 +00:00
|
|
|
// DeleteAttestation deletes the attestation record into the beacon chain db.
|
|
|
|
func (db *BeaconDB) DeleteAttestation(attestation *pb.Attestation) error {
|
2019-02-18 23:34:49 +00:00
|
|
|
hash, err := hashutil.HashProto(attestation)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-02-15 19:27:45 +00:00
|
|
|
|
|
|
|
return db.update(func(tx *bolt.Tx) error {
|
|
|
|
a := tx.Bucket(attestationBucket)
|
|
|
|
|
|
|
|
return a.Delete(hash[:])
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-01-21 09:34:11 +00:00
|
|
|
// Attestation retrieves an attestation record from the db using its hash.
|
|
|
|
func (db *BeaconDB) Attestation(hash [32]byte) (*pb.Attestation, error) {
|
2018-12-22 20:30:59 +00:00
|
|
|
var attestation *pb.Attestation
|
2018-10-17 06:11:24 +00:00
|
|
|
err := db.view(func(tx *bolt.Tx) error {
|
|
|
|
a := tx.Bucket(attestationBucket)
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
enc := a.Get(hash[:])
|
|
|
|
if enc == nil {
|
|
|
|
return nil
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
var err error
|
|
|
|
attestation, err = createAttestation(enc)
|
|
|
|
return err
|
|
|
|
})
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
return attestation, err
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|
|
|
|
|
2019-02-15 19:27:45 +00:00
|
|
|
// Attestations retrieves all the attestation records from the db.
|
|
|
|
// These are the attestations that have not been seen on the beacon chain.
|
|
|
|
func (db *BeaconDB) Attestations() ([]*pb.Attestation, error) {
|
|
|
|
var attestations []*pb.Attestation
|
|
|
|
err := db.view(func(tx *bolt.Tx) error {
|
|
|
|
a := tx.Bucket(attestationBucket)
|
|
|
|
|
|
|
|
if err := a.ForEach(func(k, v []byte) error {
|
|
|
|
attestation, err := createAttestation(v)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
attestations = append(attestations, attestation)
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
|
|
|
|
return attestations, err
|
|
|
|
}
|
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
// HasAttestation checks if the attestation exists.
|
|
|
|
func (db *BeaconDB) HasAttestation(hash [32]byte) bool {
|
|
|
|
exists := false
|
2018-10-30 16:00:20 +00:00
|
|
|
// #nosec G104
|
2018-10-17 06:11:24 +00:00
|
|
|
db.view(func(tx *bolt.Tx) error {
|
|
|
|
a := tx.Bucket(attestationBucket)
|
2018-10-05 17:14:50 +00:00
|
|
|
|
2018-10-17 06:11:24 +00:00
|
|
|
exists = a.Get(hash[:]) != nil
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return exists
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|
|
|
|
|
2018-12-22 20:30:59 +00:00
|
|
|
func createAttestation(enc []byte) (*pb.Attestation, error) {
|
|
|
|
protoAttestation := &pb.Attestation{}
|
|
|
|
if err := proto.Unmarshal(enc, protoAttestation); err != nil {
|
2018-10-17 06:11:24 +00:00
|
|
|
return nil, fmt.Errorf("failed to unmarshal encoding: %v", err)
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|
2018-12-22 20:30:59 +00:00
|
|
|
return protoAttestation, nil
|
2018-10-05 17:14:50 +00:00
|
|
|
}
|