2019-01-28 15:40:40 +00:00
|
|
|
package rpc
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-05-13 18:42:57 +00:00
|
|
|
"sync"
|
2019-01-28 15:40:40 +00:00
|
|
|
"testing"
|
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
"github.com/gogo/protobuf/proto"
|
2019-05-09 00:27:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache"
|
2019-04-03 22:50:35 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
2019-02-11 16:15:25 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/internal"
|
2019-01-28 15:40:40 +00:00
|
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-11 16:15:25 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/rpc/v1"
|
2019-03-03 17:31:29 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/hashutil"
|
2019-02-23 06:06:20 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2019-01-28 15:40:40 +00:00
|
|
|
)
|
|
|
|
|
2019-05-06 20:30:29 +00:00
|
|
|
type mockBroadcaster struct{}
|
|
|
|
|
|
|
|
func (m *mockBroadcaster) Broadcast(ctx context.Context, msg proto.Message) {
|
|
|
|
}
|
|
|
|
|
2019-02-22 15:11:26 +00:00
|
|
|
func TestAttestHead_OK(t *testing.T) {
|
2019-05-07 22:31:06 +00:00
|
|
|
db := internal.SetupDB(t)
|
|
|
|
defer internal.TeardownDB(t, db)
|
2019-01-31 18:54:24 +00:00
|
|
|
mockOperationService := &mockOperationService{}
|
2019-01-28 15:40:40 +00:00
|
|
|
attesterServer := &AttesterServer{
|
2019-01-31 18:54:24 +00:00
|
|
|
operationService: mockOperationService,
|
2019-05-06 20:30:29 +00:00
|
|
|
p2p: &mockBroadcaster{},
|
2019-05-07 22:31:06 +00:00
|
|
|
beaconDB: db,
|
2019-05-09 00:27:29 +00:00
|
|
|
cache: cache.NewAttestationCache(),
|
2019-05-07 22:31:06 +00:00
|
|
|
}
|
|
|
|
head := &pbp2p.BeaconBlock{
|
|
|
|
Slot: 999,
|
|
|
|
ParentRootHash32: []byte{'a'},
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(head); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
root, err := hashutil.HashBeaconBlock(head)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
2019-01-28 15:40:40 +00:00
|
|
|
}
|
2019-02-06 16:20:38 +00:00
|
|
|
req := &pbp2p.Attestation{
|
|
|
|
Data: &pbp2p.AttestationData{
|
2019-02-27 23:07:39 +00:00
|
|
|
Slot: 999,
|
|
|
|
Shard: 1,
|
|
|
|
CrosslinkDataRootHash32: []byte{'a'},
|
2019-05-07 22:31:06 +00:00
|
|
|
BeaconBlockRootHash32: root[:],
|
2019-01-28 15:40:40 +00:00
|
|
|
},
|
|
|
|
}
|
|
|
|
if _, err := attesterServer.AttestHead(context.Background(), req); err != nil {
|
|
|
|
t.Errorf("Could not attest head correctly: %v", err)
|
|
|
|
}
|
|
|
|
}
|
2019-02-11 16:15:25 +00:00
|
|
|
|
2019-02-27 20:21:15 +00:00
|
|
|
func TestAttestationDataAtSlot_OK(t *testing.T) {
|
2019-02-11 16:15:25 +00:00
|
|
|
db := internal.SetupDB(t)
|
|
|
|
defer internal.TeardownDB(t, db)
|
2019-04-02 08:49:45 +00:00
|
|
|
ctx := context.Background()
|
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
block := &pbp2p.BeaconBlock{
|
2019-02-15 23:19:36 +00:00
|
|
|
Slot: 1 + params.BeaconConfig().GenesisSlot,
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
|
|
|
epochBoundaryBlock := &pbp2p.BeaconBlock{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: 1*params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot,
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
|
|
|
justifiedBlock := &pbp2p.BeaconBlock{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: 2*params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot,
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
blockRoot, err := hashutil.HashBeaconBlock(block)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash beacon block: %v", err)
|
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
justifiedBlockRoot, err := hashutil.HashBeaconBlock(justifiedBlock)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash justified block: %v", err)
|
|
|
|
}
|
2019-02-26 03:42:31 +00:00
|
|
|
epochBoundaryRoot, err := hashutil.HashBeaconBlock(epochBoundaryBlock)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash justified block: %v", err)
|
|
|
|
}
|
2019-04-05 14:41:49 +00:00
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
beaconState := &pbp2p.BeaconState{
|
2019-02-18 16:52:16 +00:00
|
|
|
Slot: 3*params.BeaconConfig().SlotsPerEpoch + params.BeaconConfig().GenesisSlot + 1,
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: 2 + params.BeaconConfig().GenesisEpoch,
|
|
|
|
LatestBlockRootHash32S: make([][]byte, params.BeaconConfig().LatestBlockRootsLength),
|
2019-02-11 16:15:25 +00:00
|
|
|
LatestCrosslinks: []*pbp2p.Crosslink{
|
|
|
|
{
|
2019-02-27 23:07:39 +00:00
|
|
|
CrosslinkDataRootHash32: []byte("A"),
|
2019-02-11 16:15:25 +00:00
|
|
|
},
|
|
|
|
},
|
2019-04-10 06:52:06 +00:00
|
|
|
JustifiedRoot: justifiedBlockRoot[:],
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
|
|
|
beaconState.LatestBlockRootHash32S[1] = blockRoot[:]
|
2019-02-18 16:52:16 +00:00
|
|
|
beaconState.LatestBlockRootHash32S[1*params.BeaconConfig().SlotsPerEpoch] = epochBoundaryRoot[:]
|
|
|
|
beaconState.LatestBlockRootHash32S[2*params.BeaconConfig().SlotsPerEpoch] = justifiedBlockRoot[:]
|
2019-02-11 16:15:25 +00:00
|
|
|
attesterServer := &AttesterServer{
|
|
|
|
beaconDB: db,
|
2019-05-06 20:30:29 +00:00
|
|
|
p2p: &mockBroadcaster{},
|
2019-05-09 00:27:29 +00:00
|
|
|
cache: cache.NewAttestationCache(),
|
2019-02-11 16:15:25 +00:00
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(epochBoundaryBlock); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
2019-04-02 08:49:45 +00:00
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, epochBoundaryBlock, beaconState); err != nil {
|
2019-02-11 16:15:25 +00:00
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(justifiedBlock); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
2019-04-02 08:49:45 +00:00
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, justifiedBlock, beaconState); err != nil {
|
2019-02-11 16:15:25 +00:00
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
2019-02-19 05:49:56 +00:00
|
|
|
if err := attesterServer.beaconDB.SaveBlock(block); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
2019-04-02 08:49:45 +00:00
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, block, beaconState); err != nil {
|
2019-02-19 05:49:56 +00:00
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
2019-02-27 20:21:15 +00:00
|
|
|
req := &pb.AttestationDataRequest{
|
2019-02-11 16:15:25 +00:00
|
|
|
Shard: 0,
|
|
|
|
}
|
2019-02-27 20:21:15 +00:00
|
|
|
res, err := attesterServer.AttestationDataAtSlot(context.Background(), req)
|
2019-02-11 16:15:25 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not get attestation info at slot: %v", err)
|
|
|
|
}
|
2019-02-27 20:21:15 +00:00
|
|
|
expectedInfo := &pb.AttestationDataResponse{
|
2019-04-05 14:41:49 +00:00
|
|
|
HeadSlot: beaconState.Slot,
|
2019-02-11 16:15:25 +00:00
|
|
|
BeaconBlockRootHash32: blockRoot[:],
|
2019-02-15 23:19:36 +00:00
|
|
|
JustifiedEpoch: 2 + params.BeaconConfig().GenesisEpoch,
|
2019-02-11 16:15:25 +00:00
|
|
|
JustifiedBlockRootHash32: justifiedBlockRoot[:],
|
|
|
|
LatestCrosslink: &pbp2p.Crosslink{
|
2019-02-27 23:07:39 +00:00
|
|
|
CrosslinkDataRootHash32: []byte("A"),
|
2019-02-11 16:15:25 +00:00
|
|
|
},
|
|
|
|
}
|
2019-02-15 23:19:36 +00:00
|
|
|
|
2019-02-11 16:15:25 +00:00
|
|
|
if !proto.Equal(res, expectedInfo) {
|
|
|
|
t.Errorf("Expected attestation info to match, received %v, wanted %v", res, expectedInfo)
|
|
|
|
}
|
|
|
|
}
|
2019-04-03 22:50:35 +00:00
|
|
|
|
|
|
|
func TestAttestationDataAtSlot_handlesFarAwayJustifiedEpoch(t *testing.T) {
|
|
|
|
// Scenario:
|
|
|
|
//
|
|
|
|
// State slot = 10000
|
|
|
|
// Last justified slot = epoch start of 1500
|
|
|
|
// LatestBlockRootsLength = 8192
|
|
|
|
//
|
|
|
|
// More background: https://github.com/prysmaticlabs/prysm/issues/2153
|
|
|
|
db := internal.SetupDB(t)
|
|
|
|
defer internal.TeardownDB(t, db)
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
// Ensure LatestBlockRootsLength matches scenario
|
|
|
|
cfg := params.BeaconConfig()
|
|
|
|
cfg.LatestBlockRootsLength = 8192
|
|
|
|
params.OverrideBeaconConfig(cfg)
|
|
|
|
|
|
|
|
block := &pbp2p.BeaconBlock{
|
|
|
|
Slot: 10000 + params.BeaconConfig().GenesisSlot,
|
|
|
|
}
|
|
|
|
epochBoundaryBlock := &pbp2p.BeaconBlock{
|
|
|
|
Slot: helpers.StartSlot(helpers.SlotToEpoch(10000 + params.BeaconConfig().GenesisSlot)),
|
|
|
|
}
|
|
|
|
justifiedBlock := &pbp2p.BeaconBlock{
|
2019-04-05 19:15:08 +00:00
|
|
|
Slot: helpers.StartSlot(helpers.SlotToEpoch(1500+params.BeaconConfig().GenesisSlot)) - 2, // Imagine two skip block
|
2019-04-03 22:50:35 +00:00
|
|
|
}
|
|
|
|
blockRoot, err := hashutil.HashBeaconBlock(block)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash beacon block: %v", err)
|
|
|
|
}
|
|
|
|
justifiedBlockRoot, err := hashutil.HashBeaconBlock(justifiedBlock)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash justified block: %v", err)
|
|
|
|
}
|
|
|
|
epochBoundaryRoot, err := hashutil.HashBeaconBlock(epochBoundaryBlock)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not hash justified block: %v", err)
|
|
|
|
}
|
|
|
|
beaconState := &pbp2p.BeaconState{
|
|
|
|
Slot: 10000 + params.BeaconConfig().GenesisSlot,
|
|
|
|
JustifiedEpoch: helpers.SlotToEpoch(1500 + params.BeaconConfig().GenesisSlot),
|
|
|
|
LatestBlockRootHash32S: make([][]byte, params.BeaconConfig().LatestBlockRootsLength),
|
|
|
|
LatestCrosslinks: []*pbp2p.Crosslink{
|
|
|
|
{
|
|
|
|
CrosslinkDataRootHash32: []byte("A"),
|
|
|
|
},
|
|
|
|
},
|
2019-04-10 06:52:06 +00:00
|
|
|
JustifiedRoot: justifiedBlockRoot[:],
|
2019-04-03 22:50:35 +00:00
|
|
|
}
|
|
|
|
beaconState.LatestBlockRootHash32S[1] = blockRoot[:]
|
|
|
|
beaconState.LatestBlockRootHash32S[1*params.BeaconConfig().SlotsPerEpoch] = epochBoundaryRoot[:]
|
|
|
|
beaconState.LatestBlockRootHash32S[2*params.BeaconConfig().SlotsPerEpoch] = justifiedBlockRoot[:]
|
|
|
|
attesterServer := &AttesterServer{
|
|
|
|
beaconDB: db,
|
2019-05-06 20:30:29 +00:00
|
|
|
p2p: &mockBroadcaster{},
|
2019-05-09 00:27:29 +00:00
|
|
|
cache: cache.NewAttestationCache(),
|
2019-04-03 22:50:35 +00:00
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(epochBoundaryBlock); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, epochBoundaryBlock, beaconState); err != nil {
|
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(justifiedBlock); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, justifiedBlock, beaconState); err != nil {
|
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.SaveBlock(block); err != nil {
|
|
|
|
t.Fatalf("Could not save block in test db: %v", err)
|
|
|
|
}
|
|
|
|
if err := attesterServer.beaconDB.UpdateChainHead(ctx, block, beaconState); err != nil {
|
|
|
|
t.Fatalf("Could not update chain head in test db: %v", err)
|
|
|
|
}
|
|
|
|
req := &pb.AttestationDataRequest{
|
|
|
|
Shard: 0,
|
|
|
|
}
|
|
|
|
res, err := attesterServer.AttestationDataAtSlot(context.Background(), req)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Could not get attestation info at slot: %v", err)
|
|
|
|
}
|
|
|
|
expectedInfo := &pb.AttestationDataResponse{
|
2019-04-05 14:41:49 +00:00
|
|
|
HeadSlot: 10000 + params.BeaconConfig().GenesisSlot,
|
2019-04-03 22:50:35 +00:00
|
|
|
BeaconBlockRootHash32: blockRoot[:],
|
|
|
|
JustifiedEpoch: helpers.SlotToEpoch(1500 + params.BeaconConfig().GenesisSlot),
|
|
|
|
JustifiedBlockRootHash32: justifiedBlockRoot[:],
|
|
|
|
LatestCrosslink: &pbp2p.Crosslink{
|
|
|
|
CrosslinkDataRootHash32: []byte("A"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if !proto.Equal(res, expectedInfo) {
|
|
|
|
t.Errorf("Expected attestation info to match, received %v, wanted %v", res, expectedInfo)
|
|
|
|
}
|
|
|
|
}
|
2019-05-13 18:42:57 +00:00
|
|
|
|
|
|
|
func TestAttestationDataAtSlot_handlesInProgressRequest(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
|
|
|
server := &AttesterServer{
|
|
|
|
cache: cache.NewAttestationCache(),
|
|
|
|
}
|
|
|
|
|
|
|
|
req := &pb.AttestationDataRequest{
|
|
|
|
Shard: 1,
|
|
|
|
Slot: 2,
|
|
|
|
}
|
|
|
|
|
|
|
|
res := &pb.AttestationDataResponse{
|
|
|
|
HeadSlot: 55,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := server.cache.MarkInProgress(req); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
wg.Add(1)
|
|
|
|
defer wg.Done()
|
|
|
|
response, err := server.AttestationDataAtSlot(ctx, req)
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if !proto.Equal(res, response) {
|
|
|
|
t.Error("Expected equal responses from cache")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
wg.Add(1)
|
|
|
|
defer wg.Done()
|
|
|
|
|
|
|
|
if err := server.cache.Put(ctx, req, res); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if err := server.cache.MarkNotInProgress(req); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
}
|