mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-31 23:41:22 +00:00
3839f577ec
* Change database *Index() to use slice * Remove underscore from helper name
240 lines
7.4 KiB
Go
240 lines
7.4 KiB
Go
package aggregator
|
|
|
|
import (
|
|
"context"
|
|
"encoding/binary"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
|
"github.com/prysmaticlabs/go-bitfield"
|
|
"github.com/prysmaticlabs/go-ssz"
|
|
mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
dbutil "github.com/prysmaticlabs/prysm/beacon-chain/db/testing"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/operations/attestations"
|
|
mockp2p "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing"
|
|
mockSync "github.com/prysmaticlabs/prysm/beacon-chain/sync/initial-sync/testing"
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
|
"github.com/prysmaticlabs/prysm/shared/bls"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
"github.com/prysmaticlabs/prysm/shared/testutil"
|
|
)
|
|
|
|
func init() {
|
|
// Use minimal config to reduce test setup time.
|
|
params.OverrideBeaconConfig(params.MinimalSpecConfig())
|
|
}
|
|
|
|
// pubKey is a helper to generate a well-formed public key.
|
|
func pubKey(i uint64) []byte {
|
|
pubKey := make([]byte, params.BeaconConfig().BLSPubkeyLength)
|
|
binary.LittleEndian.PutUint64(pubKey, uint64(i))
|
|
return pubKey
|
|
}
|
|
|
|
func TestSubmitAggregateAndProof_Syncing(t *testing.T) {
|
|
db := dbutil.SetupDB(t)
|
|
defer dbutil.TeardownDB(t, db)
|
|
ctx := context.Background()
|
|
|
|
s := &pbp2p.BeaconState{}
|
|
|
|
aggregatorServer := &Server{
|
|
HeadFetcher: &mock.ChainService{State: s},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: true},
|
|
BeaconDB: db,
|
|
}
|
|
|
|
req := &pb.AggregationRequest{CommitteeIndex: 1}
|
|
wanted := "Syncing to latest head, not ready to respond"
|
|
if _, err := aggregatorServer.SubmitAggregateAndProof(ctx, req); !strings.Contains(err.Error(), wanted) {
|
|
t.Error("Did not receive wanted error")
|
|
}
|
|
}
|
|
|
|
func TestSubmitAggregateAndProof_CantFindValidatorIndex(t *testing.T) {
|
|
db := dbutil.SetupDB(t)
|
|
defer dbutil.TeardownDB(t, db)
|
|
ctx := context.Background()
|
|
|
|
s := &pbp2p.BeaconState{
|
|
RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector),
|
|
}
|
|
|
|
aggregatorServer := &Server{
|
|
HeadFetcher: &mock.ChainService{State: s},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
BeaconDB: db,
|
|
}
|
|
|
|
priv := bls.RandKey()
|
|
sig := priv.Sign([]byte{'A'}, 0)
|
|
req := &pb.AggregationRequest{CommitteeIndex: 1, SlotSignature: sig.Marshal(), PublicKey: pubKey(3)}
|
|
wanted := "Could not locate validator index in DB"
|
|
if _, err := aggregatorServer.SubmitAggregateAndProof(ctx, req); !strings.Contains(err.Error(), wanted) {
|
|
t.Errorf("Did not receive wanted error: expected %v, received %v", wanted, err.Error())
|
|
}
|
|
}
|
|
|
|
func TestSubmitAggregateAndProof_IsAggregator(t *testing.T) {
|
|
db := dbutil.SetupDB(t)
|
|
defer dbutil.TeardownDB(t, db)
|
|
ctx := context.Background()
|
|
|
|
s := &pbp2p.BeaconState{
|
|
RandaoMixes: make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector),
|
|
}
|
|
|
|
aggregatorServer := &Server{
|
|
HeadFetcher: &mock.ChainService{State: s},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
BeaconDB: db,
|
|
AttPool: attestations.NewPool(),
|
|
}
|
|
|
|
priv := bls.RandKey()
|
|
sig := priv.Sign([]byte{'A'}, 0)
|
|
pubKey := pubKey(1)
|
|
req := &pb.AggregationRequest{CommitteeIndex: 1, SlotSignature: sig.Marshal(), PublicKey: pubKey}
|
|
if err := aggregatorServer.BeaconDB.SaveValidatorIndex(ctx, pubKey, 100); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if _, err := aggregatorServer.SubmitAggregateAndProof(ctx, req); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
}
|
|
|
|
func TestSubmitAggregateAndProof_AggregateOk(t *testing.T) {
|
|
params.UseMinimalConfig()
|
|
c := params.MinimalSpecConfig()
|
|
c.TargetAggregatorsPerCommittee = 16
|
|
params.OverrideBeaconConfig(c)
|
|
defer params.UseMainnetConfig()
|
|
|
|
db := dbutil.SetupDB(t)
|
|
defer dbutil.TeardownDB(t, db)
|
|
ctx := context.Background()
|
|
|
|
beaconState, privKeys := testutil.DeterministicGenesisState(t, 32)
|
|
att0 := generateAtt(beaconState, 0, privKeys)
|
|
att1 := generateAtt(beaconState, 1, privKeys)
|
|
|
|
beaconState.Slot += params.BeaconConfig().MinAttestationInclusionDelay
|
|
|
|
aggregatorServer := &Server{
|
|
HeadFetcher: &mock.ChainService{State: beaconState},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
BeaconDB: db,
|
|
AttPool: attestations.NewPool(),
|
|
P2p: &mockp2p.MockBroadcaster{},
|
|
}
|
|
|
|
priv := bls.RandKey()
|
|
sig := priv.Sign([]byte{'B'}, 0)
|
|
pubKey := pubKey(2)
|
|
req := &pb.AggregationRequest{CommitteeIndex: 1, SlotSignature: sig.Marshal(), PublicKey: pubKey}
|
|
if err := aggregatorServer.BeaconDB.SaveValidatorIndex(ctx, pubKey, 100); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if err := aggregatorServer.AttPool.SaveUnaggregatedAttestation(att0); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if err := aggregatorServer.AttPool.SaveUnaggregatedAttestation(att1); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if _, err := aggregatorServer.SubmitAggregateAndProof(ctx, req); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
aggregatedAtts := aggregatorServer.AttPool.AggregatedAttestations()
|
|
wanted, err := helpers.AggregateAttestation(att0, att1)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if reflect.DeepEqual(aggregatedAtts, wanted) {
|
|
t.Error("Did not receive wanted attestation")
|
|
}
|
|
}
|
|
|
|
func TestSubmitAggregateAndProof_AggregateNotOk(t *testing.T) {
|
|
params.UseMinimalConfig()
|
|
c := params.MinimalSpecConfig()
|
|
c.TargetAggregatorsPerCommittee = 16
|
|
params.OverrideBeaconConfig(c)
|
|
defer params.UseMainnetConfig()
|
|
|
|
db := dbutil.SetupDB(t)
|
|
defer dbutil.TeardownDB(t, db)
|
|
ctx := context.Background()
|
|
|
|
beaconState, privKeys := testutil.DeterministicGenesisState(t, 32)
|
|
att0 := generateAtt(beaconState, 0, privKeys)
|
|
|
|
beaconState.Slot += params.BeaconConfig().MinAttestationInclusionDelay
|
|
|
|
aggregatorServer := &Server{
|
|
HeadFetcher: &mock.ChainService{State: beaconState},
|
|
SyncChecker: &mockSync.Sync{IsSyncing: false},
|
|
BeaconDB: db,
|
|
AttPool: attestations.NewPool(),
|
|
P2p: &mockp2p.MockBroadcaster{},
|
|
}
|
|
|
|
priv := bls.RandKey()
|
|
sig := priv.Sign([]byte{'B'}, 0)
|
|
pubKey := pubKey(2)
|
|
req := &pb.AggregationRequest{CommitteeIndex: 1, SlotSignature: sig.Marshal(), PublicKey: pubKey}
|
|
if err := aggregatorServer.BeaconDB.SaveValidatorIndex(ctx, pubKey, 100); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if err := aggregatorServer.AttPool.SaveUnaggregatedAttestation(att0); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
if _, err := aggregatorServer.SubmitAggregateAndProof(ctx, req); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
aggregatedAtts := aggregatorServer.AttPool.AggregatedAttestations()
|
|
if len(aggregatedAtts) != 0 {
|
|
t.Errorf("Wanted aggregated attestation 0, got %d", len(aggregatedAtts))
|
|
}
|
|
}
|
|
|
|
func generateAtt(state *pbp2p.BeaconState, index uint64, privKeys []*bls.SecretKey) *ethpb.Attestation {
|
|
aggBits := bitfield.NewBitlist(4)
|
|
aggBits.SetBitAt(index, true)
|
|
att := ðpb.Attestation{
|
|
Data: ðpb.AttestationData{
|
|
CommitteeIndex: 1,
|
|
Source: ðpb.Checkpoint{Epoch: 0, Root: params.BeaconConfig().ZeroHash[:]},
|
|
Target: ðpb.Checkpoint{Epoch: 0},
|
|
},
|
|
AggregationBits: aggBits,
|
|
}
|
|
committee, _ := helpers.BeaconCommitteeFromState(state, att.Data.Slot, att.Data.CommitteeIndex)
|
|
attestingIndices, _ := helpers.AttestingIndices(att.AggregationBits, committee)
|
|
domain := helpers.Domain(state.Fork, 0, params.BeaconConfig().DomainBeaconAttester)
|
|
|
|
sigs := make([]*bls.Signature, len(attestingIndices))
|
|
zeroSig := [96]byte{}
|
|
att.Signature = zeroSig[:]
|
|
|
|
for i, indice := range attestingIndices {
|
|
hashTreeRoot, _ := ssz.HashTreeRoot(att.Data)
|
|
sig := privKeys[indice].Sign(hashTreeRoot[:], domain)
|
|
sigs[i] = sig
|
|
}
|
|
|
|
att.Signature = bls.AggregateSignatures(sigs).Marshal()[:]
|
|
|
|
return att
|
|
}
|