mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-16 06:58:20 +00:00
d077483577
* v3 import renamings * tidy * fmt * rev * Update beacon-chain/core/epoch/precompute/reward_penalty_test.go * Update beacon-chain/core/helpers/validators_test.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/alias.go * Update beacon-chain/db/iface/BUILD.bazel * Update beacon-chain/db/kv/kv.go * Update beacon-chain/db/kv/state.go * Update beacon-chain/rpc/prysm/v1alpha1/validator/attester_test.go * Update beacon-chain/rpc/prysm/v1alpha1/validator/attester_test.go * Update beacon-chain/sync/initial-sync/service.go * fix deps * fix bad replacements * fix bad replacements * change back * gohashtree version * fix deps Co-authored-by: Nishant Das <nishdas93@gmail.com> Co-authored-by: Potuz <potuz@prysmaticlabs.com>
1239 lines
42 KiB
Go
1239 lines
42 KiB
Go
package beacon
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/golang/mock/gomock"
|
|
chainMock "github.com/prysmaticlabs/prysm/v3/beacon-chain/blockchain/testing"
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed"
|
|
blockfeed "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed/block"
|
|
statefeed "github.com/prysmaticlabs/prysm/v3/beacon-chain/core/feed/state"
|
|
dbTest "github.com/prysmaticlabs/prysm/v3/beacon-chain/db/testing"
|
|
v1 "github.com/prysmaticlabs/prysm/v3/beacon-chain/state/v1"
|
|
"github.com/prysmaticlabs/prysm/v3/cmd"
|
|
"github.com/prysmaticlabs/prysm/v3/config/features"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v3/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v3/consensus-types/interfaces"
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v3/testing/assert"
|
|
"github.com/prysmaticlabs/prysm/v3/testing/mock"
|
|
"github.com/prysmaticlabs/prysm/v3/testing/require"
|
|
"github.com/prysmaticlabs/prysm/v3/testing/util"
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
|
"google.golang.org/protobuf/proto"
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
|
)
|
|
|
|
func TestServer_ListBlocks_NoResults(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
wanted := ðpb.ListBlocksResponse{
|
|
BlockContainers: make([]*ethpb.BeaconBlockContainer, 0),
|
|
TotalSize: int32(0),
|
|
NextPageToken: strconv.Itoa(0),
|
|
}
|
|
res, err := bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{
|
|
Slot: 0,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
res, err = bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{
|
|
Slot: 0,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
res, err = bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Root{
|
|
Root: make([]byte, fieldparams.RootLength),
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBlocks_Genesis(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
|
|
// Should throw an error if no genesis block is found.
|
|
_, err := bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.ErrorContains(t, "Could not find genesis", err)
|
|
|
|
// Should return the proper genesis block if it exists.
|
|
parentRoot := [32]byte{'a'}
|
|
blk := util.NewBeaconBlock()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
root, err := blk.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
util.SaveBlock(t, ctx, db, blk)
|
|
require.NoError(t, db.SaveGenesisBlockRoot(ctx, root))
|
|
wanted := ðpb.ListBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{
|
|
{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{Phase0Block: blk},
|
|
BlockRoot: root[:],
|
|
Canonical: true,
|
|
},
|
|
},
|
|
NextPageToken: "0",
|
|
TotalSize: 1,
|
|
}
|
|
res, err := bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBlocks_Genesis_MultiBlocks(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
// Should return the proper genesis block if it exists.
|
|
parentRoot := [32]byte{1, 2, 3}
|
|
blk := util.NewBeaconBlock()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
root, err := blk.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
util.SaveBlock(t, ctx, db, blk)
|
|
require.NoError(t, db.SaveGenesisBlockRoot(ctx, root))
|
|
|
|
count := types.Slot(100)
|
|
blks := make([]interfaces.SignedBeaconBlock, count)
|
|
for i := types.Slot(0); i < count; i++ {
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot = i
|
|
require.NoError(t, err)
|
|
blks[i], err = blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
}
|
|
require.NoError(t, db.SaveBlocks(ctx, blks))
|
|
|
|
// Should throw an error if more than one blk returned.
|
|
_, err = bs.ListBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func TestServer_ListBlocks_Pagination(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MinimalSpecConfig())
|
|
|
|
db := dbTest.SetupDB(t)
|
|
chain := &chainMock.ChainService{
|
|
CanonicalRoots: map[[32]byte]bool{},
|
|
}
|
|
ctx := context.Background()
|
|
|
|
count := types.Slot(100)
|
|
blks := make([]interfaces.SignedBeaconBlock, count)
|
|
blkContainers := make([]*ethpb.BeaconBlockContainer, count)
|
|
for i := types.Slot(0); i < count; i++ {
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot = i
|
|
root, err := b.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
chain.CanonicalRoots[root] = true
|
|
blks[i], err = blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
blkContainers[i] = ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{Phase0Block: b},
|
|
BlockRoot: root[:],
|
|
Canonical: true,
|
|
}
|
|
}
|
|
require.NoError(t, db.SaveBlocks(ctx, blks))
|
|
|
|
orphanedBlk := util.NewBeaconBlock()
|
|
orphanedBlk.Block.Slot = 300
|
|
orphanedBlkRoot, err := orphanedBlk.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
util.SaveBlock(t, ctx, db, orphanedBlk)
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
CanonicalFetcher: chain,
|
|
}
|
|
|
|
root6, err := blks[6].Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
tests := []struct {
|
|
req *ethpb.ListBlocksRequest
|
|
res *ethpb.ListBlocksResponse
|
|
}{
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 5},
|
|
PageSize: 3},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{
|
|
{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{
|
|
Phase0Block: util.HydrateSignedBeaconBlock(ðpb.SignedBeaconBlock{
|
|
Block: ðpb.BeaconBlock{
|
|
Slot: 5,
|
|
},
|
|
}),
|
|
},
|
|
BlockRoot: blkContainers[5].BlockRoot,
|
|
Canonical: blkContainers[5].Canonical,
|
|
},
|
|
},
|
|
NextPageToken: "",
|
|
TotalSize: 1,
|
|
},
|
|
},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]},
|
|
PageSize: 3},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{
|
|
{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{
|
|
Phase0Block: util.HydrateSignedBeaconBlock(ðpb.SignedBeaconBlock{
|
|
Block: ðpb.BeaconBlock{
|
|
Slot: 6,
|
|
},
|
|
}),
|
|
},
|
|
BlockRoot: blkContainers[6].BlockRoot,
|
|
Canonical: blkContainers[6].Canonical,
|
|
},
|
|
},
|
|
TotalSize: 1,
|
|
NextPageToken: strconv.Itoa(0),
|
|
},
|
|
},
|
|
{req: ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]}},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{
|
|
{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{
|
|
Phase0Block: util.HydrateSignedBeaconBlock(ðpb.SignedBeaconBlock{
|
|
Block: ðpb.BeaconBlock{
|
|
Slot: 6,
|
|
},
|
|
}),
|
|
},
|
|
BlockRoot: blkContainers[6].BlockRoot,
|
|
Canonical: blkContainers[6].Canonical,
|
|
},
|
|
},
|
|
TotalSize: 1,
|
|
NextPageToken: strconv.Itoa(0),
|
|
},
|
|
},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 0},
|
|
PageSize: 100},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: blkContainers[0:params.BeaconConfig().SlotsPerEpoch],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(1),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 5},
|
|
PageSize: 3},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: blkContainers[43:46],
|
|
NextPageToken: "2",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(1),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 11},
|
|
PageSize: 7},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: blkContainers[95:96],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 12},
|
|
PageSize: 4},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: blkContainers[96:100],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch / 2)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 300},
|
|
PageSize: 3},
|
|
res: ðpb.ListBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{
|
|
{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{
|
|
Phase0Block: util.HydrateSignedBeaconBlock(ðpb.SignedBeaconBlock{
|
|
Block: ðpb.BeaconBlock{
|
|
Slot: 300,
|
|
},
|
|
}),
|
|
},
|
|
BlockRoot: orphanedBlkRoot[:],
|
|
Canonical: false,
|
|
},
|
|
},
|
|
NextPageToken: "",
|
|
TotalSize: 1}},
|
|
}
|
|
|
|
for i, test := range tests {
|
|
t.Run(fmt.Sprintf("test_%d", i), func(t *testing.T) {
|
|
res, err := bs.ListBlocks(ctx, test.req)
|
|
require.NoError(t, err)
|
|
require.DeepSSZEqual(t, res, test.res)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBlocks_Errors(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{BeaconDB: db}
|
|
exceedsMax := int32(cmd.Get().MaxRPCPageSize + 1)
|
|
|
|
wanted := fmt.Sprintf("Requested page size %d can not be greater than max size %d", exceedsMax, cmd.Get().MaxRPCPageSize)
|
|
req := ðpb.ListBlocksRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax}
|
|
_, err := bs.ListBlocks(ctx, req)
|
|
assert.ErrorContains(t, wanted, err)
|
|
|
|
wanted = "Must specify a filter criteria for fetching"
|
|
req = ðpb.ListBlocksRequest{}
|
|
_, err = bs.ListBlocks(ctx, req)
|
|
assert.ErrorContains(t, wanted, err)
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 0}}
|
|
res, err := bs.ListBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Slot{}}
|
|
res, err = bs.ListBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}}
|
|
res, err = bs.ListBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}}
|
|
res, err = bs.ListBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
}
|
|
|
|
// ensures that if any of the checkpoints are zero-valued, an error will be generated without genesis being present
|
|
func TestServer_GetChainHead_NoGenesis(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
|
|
s, err := util.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, s.SetSlot(1))
|
|
|
|
genBlock := util.NewBeaconBlock()
|
|
genBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'G'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, genBlock)
|
|
gRoot, err := genBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
cases := []struct {
|
|
name string
|
|
zeroSetter func(val *ethpb.Checkpoint) error
|
|
}{
|
|
{
|
|
name: "zero-value prev justified",
|
|
zeroSetter: s.SetPreviousJustifiedCheckpoint,
|
|
},
|
|
{
|
|
name: "zero-value current justified",
|
|
zeroSetter: s.SetCurrentJustifiedCheckpoint,
|
|
},
|
|
{
|
|
name: "zero-value finalized",
|
|
zeroSetter: s.SetFinalizedCheckpoint,
|
|
},
|
|
}
|
|
finalized := ðpb.Checkpoint{Epoch: 1, Root: gRoot[:]}
|
|
prevJustified := ðpb.Checkpoint{Epoch: 2, Root: gRoot[:]}
|
|
justified := ðpb.Checkpoint{Epoch: 3, Root: gRoot[:]}
|
|
|
|
for _, c := range cases {
|
|
t.Run(c.name, func(t *testing.T) {
|
|
require.NoError(t, s.SetPreviousJustifiedCheckpoint(prevJustified))
|
|
require.NoError(t, s.SetCurrentJustifiedCheckpoint(justified))
|
|
require.NoError(t, s.SetFinalizedCheckpoint(finalized))
|
|
require.NoError(t, c.zeroSetter(ðpb.Checkpoint{Epoch: 0, Root: params.BeaconConfig().ZeroHash[:]}))
|
|
})
|
|
wsb, err := blocks.NewSignedBeaconBlock(genBlock)
|
|
require.NoError(t, err)
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
HeadFetcher: &chainMock.ChainService{Block: wsb, State: s},
|
|
FinalizationFetcher: &chainMock.ChainService{
|
|
FinalizedCheckPoint: s.FinalizedCheckpoint(),
|
|
CurrentJustifiedCheckPoint: s.CurrentJustifiedCheckpoint(),
|
|
PreviousJustifiedCheckPoint: s.PreviousJustifiedCheckpoint()},
|
|
OptimisticModeFetcher: &chainMock.ChainService{},
|
|
}
|
|
_, err = bs.GetChainHead(context.Background(), nil)
|
|
require.ErrorContains(t, "Could not get genesis block", err)
|
|
}
|
|
}
|
|
|
|
func TestServer_GetChainHead_NoFinalizedBlock(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
|
|
s, err := util.NewBeaconState()
|
|
require.NoError(t, err)
|
|
require.NoError(t, s.SetSlot(1))
|
|
require.NoError(t, s.SetPreviousJustifiedCheckpoint(ðpb.Checkpoint{Epoch: 3, Root: bytesutil.PadTo([]byte{'A'}, fieldparams.RootLength)}))
|
|
require.NoError(t, s.SetCurrentJustifiedCheckpoint(ðpb.Checkpoint{Epoch: 2, Root: bytesutil.PadTo([]byte{'B'}, fieldparams.RootLength)}))
|
|
require.NoError(t, s.SetFinalizedCheckpoint(ðpb.Checkpoint{Epoch: 1, Root: bytesutil.PadTo([]byte{'C'}, fieldparams.RootLength)}))
|
|
|
|
genBlock := util.NewBeaconBlock()
|
|
genBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'G'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, genBlock)
|
|
gRoot, err := genBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveGenesisBlockRoot(context.Background(), gRoot))
|
|
|
|
wsb, err := blocks.NewSignedBeaconBlock(genBlock)
|
|
require.NoError(t, err)
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
HeadFetcher: &chainMock.ChainService{Block: wsb, State: s},
|
|
FinalizationFetcher: &chainMock.ChainService{
|
|
FinalizedCheckPoint: s.FinalizedCheckpoint(),
|
|
CurrentJustifiedCheckPoint: s.CurrentJustifiedCheckpoint(),
|
|
PreviousJustifiedCheckPoint: s.PreviousJustifiedCheckpoint()},
|
|
OptimisticModeFetcher: &chainMock.ChainService{},
|
|
}
|
|
|
|
_, err = bs.GetChainHead(context.Background(), nil)
|
|
require.ErrorContains(t, "Could not get finalized block", err)
|
|
}
|
|
|
|
func TestServer_GetChainHead_NoHeadBlock(t *testing.T) {
|
|
bs := &Server{
|
|
HeadFetcher: &chainMock.ChainService{Block: nil},
|
|
OptimisticModeFetcher: &chainMock.ChainService{},
|
|
}
|
|
_, err := bs.GetChainHead(context.Background(), nil)
|
|
assert.ErrorContains(t, "Head block of chain was nil", err)
|
|
}
|
|
|
|
func TestServer_GetChainHead(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MinimalSpecConfig())
|
|
|
|
db := dbTest.SetupDB(t)
|
|
genBlock := util.NewBeaconBlock()
|
|
genBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'G'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, genBlock)
|
|
gRoot, err := genBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveGenesisBlockRoot(context.Background(), gRoot))
|
|
|
|
finalizedBlock := util.NewBeaconBlock()
|
|
finalizedBlock.Block.Slot = 1
|
|
finalizedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'A'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, finalizedBlock)
|
|
fRoot, err := finalizedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
justifiedBlock := util.NewBeaconBlock()
|
|
justifiedBlock.Block.Slot = 2
|
|
justifiedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'B'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, justifiedBlock)
|
|
jRoot, err := justifiedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
prevJustifiedBlock := util.NewBeaconBlock()
|
|
prevJustifiedBlock.Block.Slot = 3
|
|
prevJustifiedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'C'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, prevJustifiedBlock)
|
|
pjRoot, err := prevJustifiedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
s, err := v1.InitializeFromProto(ðpb.BeaconState{
|
|
Slot: 1,
|
|
PreviousJustifiedCheckpoint: ðpb.Checkpoint{Epoch: 3, Root: pjRoot[:]},
|
|
CurrentJustifiedCheckpoint: ðpb.Checkpoint{Epoch: 2, Root: jRoot[:]},
|
|
FinalizedCheckpoint: ðpb.Checkpoint{Epoch: 1, Root: fRoot[:]},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot, err = slots.EpochStart(s.PreviousJustifiedCheckpoint().Epoch)
|
|
require.NoError(t, err)
|
|
b.Block.Slot++
|
|
wsb, err := blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
HeadFetcher: &chainMock.ChainService{Block: wsb, State: s},
|
|
OptimisticModeFetcher: &chainMock.ChainService{},
|
|
FinalizationFetcher: &chainMock.ChainService{
|
|
FinalizedCheckPoint: s.FinalizedCheckpoint(),
|
|
CurrentJustifiedCheckPoint: s.CurrentJustifiedCheckpoint(),
|
|
PreviousJustifiedCheckPoint: s.PreviousJustifiedCheckpoint()},
|
|
}
|
|
|
|
head, err := bs.GetChainHead(context.Background(), nil)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, types.Epoch(3), head.PreviousJustifiedEpoch, "Unexpected PreviousJustifiedEpoch")
|
|
assert.Equal(t, types.Epoch(2), head.JustifiedEpoch, "Unexpected JustifiedEpoch")
|
|
assert.Equal(t, types.Epoch(1), head.FinalizedEpoch, "Unexpected FinalizedEpoch")
|
|
assert.Equal(t, types.Slot(24), head.PreviousJustifiedSlot, "Unexpected PreviousJustifiedSlot")
|
|
assert.Equal(t, types.Slot(16), head.JustifiedSlot, "Unexpected JustifiedSlot")
|
|
assert.Equal(t, types.Slot(8), head.FinalizedSlot, "Unexpected FinalizedSlot")
|
|
assert.DeepEqual(t, pjRoot[:], head.PreviousJustifiedBlockRoot, "Unexpected PreviousJustifiedBlockRoot")
|
|
assert.DeepEqual(t, jRoot[:], head.JustifiedBlockRoot, "Unexpected JustifiedBlockRoot")
|
|
assert.DeepEqual(t, fRoot[:], head.FinalizedBlockRoot, "Unexpected FinalizedBlockRoot")
|
|
assert.Equal(t, false, head.OptimisticStatus)
|
|
}
|
|
|
|
func TestServer_StreamChainHead_ContextCanceled(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
chainService := &chainMock.ChainService{}
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
StateNotifier: chainService.StateNotifier(),
|
|
BeaconDB: db,
|
|
}
|
|
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamChainHeadServer(ctrl)
|
|
mockStream.EXPECT().Context().Return(ctx)
|
|
go func(tt *testing.T) {
|
|
assert.ErrorContains(tt, "Context canceled", server.StreamChainHead(&emptypb.Empty{}, mockStream))
|
|
<-exitRoutine
|
|
}(t)
|
|
cancel()
|
|
exitRoutine <- true
|
|
}
|
|
|
|
func TestServer_StreamChainHead_OnHeadUpdated(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MainnetConfig())
|
|
db := dbTest.SetupDB(t)
|
|
genBlock := util.NewBeaconBlock()
|
|
genBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'G'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, genBlock)
|
|
gRoot, err := genBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveGenesisBlockRoot(context.Background(), gRoot))
|
|
|
|
finalizedBlock := util.NewBeaconBlock()
|
|
finalizedBlock.Block.Slot = 32
|
|
finalizedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'A'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, finalizedBlock)
|
|
fRoot, err := finalizedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
justifiedBlock := util.NewBeaconBlock()
|
|
justifiedBlock.Block.Slot = 64
|
|
justifiedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'B'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, justifiedBlock)
|
|
jRoot, err := justifiedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
prevJustifiedBlock := util.NewBeaconBlock()
|
|
prevJustifiedBlock.Block.Slot = 96
|
|
prevJustifiedBlock.Block.ParentRoot = bytesutil.PadTo([]byte{'C'}, fieldparams.RootLength)
|
|
util.SaveBlock(t, context.Background(), db, prevJustifiedBlock)
|
|
pjRoot, err := prevJustifiedBlock.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
s, err := v1.InitializeFromProto(ðpb.BeaconState{
|
|
Slot: 1,
|
|
PreviousJustifiedCheckpoint: ðpb.Checkpoint{Epoch: 3, Root: pjRoot[:]},
|
|
CurrentJustifiedCheckpoint: ðpb.Checkpoint{Epoch: 2, Root: jRoot[:]},
|
|
FinalizedCheckpoint: ðpb.Checkpoint{Epoch: 1, Root: fRoot[:]},
|
|
})
|
|
require.NoError(t, err)
|
|
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot, err = slots.EpochStart(s.PreviousJustifiedCheckpoint().Epoch)
|
|
require.NoError(t, err)
|
|
|
|
hRoot, err := b.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
chainService := &chainMock.ChainService{}
|
|
ctx := context.Background()
|
|
wsb, err := blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
HeadFetcher: &chainMock.ChainService{Block: wsb, State: s},
|
|
BeaconDB: db,
|
|
StateNotifier: chainService.StateNotifier(),
|
|
FinalizationFetcher: &chainMock.ChainService{
|
|
FinalizedCheckPoint: s.FinalizedCheckpoint(),
|
|
CurrentJustifiedCheckPoint: s.CurrentJustifiedCheckpoint(),
|
|
PreviousJustifiedCheckPoint: s.PreviousJustifiedCheckpoint()},
|
|
OptimisticModeFetcher: &chainMock.ChainService{},
|
|
}
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamChainHeadServer(ctrl)
|
|
mockStream.EXPECT().Send(
|
|
ðpb.ChainHead{
|
|
HeadSlot: b.Block.Slot,
|
|
HeadEpoch: slots.ToEpoch(b.Block.Slot),
|
|
HeadBlockRoot: hRoot[:],
|
|
FinalizedSlot: 32,
|
|
FinalizedEpoch: 1,
|
|
FinalizedBlockRoot: fRoot[:],
|
|
JustifiedSlot: 64,
|
|
JustifiedEpoch: 2,
|
|
JustifiedBlockRoot: jRoot[:],
|
|
PreviousJustifiedSlot: 96,
|
|
PreviousJustifiedEpoch: 3,
|
|
PreviousJustifiedBlockRoot: pjRoot[:],
|
|
},
|
|
).Do(func(arg0 interface{}) {
|
|
exitRoutine <- true
|
|
})
|
|
mockStream.EXPECT().Context().Return(ctx).AnyTimes()
|
|
|
|
go func(tt *testing.T) {
|
|
assert.NoError(tt, server.StreamChainHead(&emptypb.Empty{}, mockStream), "Could not call RPC method")
|
|
}(t)
|
|
|
|
// Send in a loop to ensure it is delivered (busy wait for the service to subscribe to the state feed).
|
|
for sent := 0; sent == 0; {
|
|
sent = server.StateNotifier.StateFeed().Send(&feed.Event{
|
|
Type: statefeed.BlockProcessed,
|
|
Data: &statefeed.BlockProcessedData{},
|
|
})
|
|
}
|
|
<-exitRoutine
|
|
}
|
|
|
|
func TestServer_StreamBlocksVerified_ContextCanceled(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
chainService := &chainMock.ChainService{}
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
StateNotifier: chainService.StateNotifier(),
|
|
HeadFetcher: chainService,
|
|
BeaconDB: db,
|
|
}
|
|
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamBlocksServer(ctrl)
|
|
mockStream.EXPECT().Context().Return(ctx)
|
|
go func(tt *testing.T) {
|
|
assert.ErrorContains(tt, "Context canceled", server.StreamBlocks(ðpb.StreamBlocksRequest{
|
|
VerifiedOnly: true,
|
|
}, mockStream))
|
|
<-exitRoutine
|
|
}(t)
|
|
cancel()
|
|
exitRoutine <- true
|
|
}
|
|
|
|
func TestServer_StreamBlocks_ContextCanceled(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
chainService := &chainMock.ChainService{}
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
BlockNotifier: chainService.BlockNotifier(),
|
|
HeadFetcher: chainService,
|
|
BeaconDB: db,
|
|
}
|
|
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamBlocksServer(ctrl)
|
|
mockStream.EXPECT().Context().Return(ctx)
|
|
go func(tt *testing.T) {
|
|
assert.ErrorContains(tt, "Context canceled", server.StreamBlocks(ðpb.StreamBlocksRequest{}, mockStream))
|
|
<-exitRoutine
|
|
}(t)
|
|
cancel()
|
|
exitRoutine <- true
|
|
}
|
|
|
|
func TestServer_StreamBlocks_OnHeadUpdated(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MainnetConfig())
|
|
|
|
ctx := context.Background()
|
|
beaconState, privs := util.DeterministicGenesisState(t, 32)
|
|
b, err := util.GenerateFullBlock(beaconState, privs, util.DefaultBlockGenConfig(), 1)
|
|
require.NoError(t, err)
|
|
chainService := &chainMock.ChainService{State: beaconState}
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
BlockNotifier: chainService.BlockNotifier(),
|
|
HeadFetcher: chainService,
|
|
}
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamBlocksServer(ctrl)
|
|
mockStream.EXPECT().Send(b).Do(func(arg0 interface{}) {
|
|
exitRoutine <- true
|
|
})
|
|
mockStream.EXPECT().Context().Return(ctx).AnyTimes()
|
|
|
|
go func(tt *testing.T) {
|
|
assert.NoError(tt, server.StreamBlocks(ðpb.StreamBlocksRequest{}, mockStream), "Could not call RPC method")
|
|
}(t)
|
|
|
|
// Send in a loop to ensure it is delivered (busy wait for the service to subscribe to the state feed).
|
|
for sent := 0; sent == 0; {
|
|
wsb, err := blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
sent = server.BlockNotifier.BlockFeed().Send(&feed.Event{
|
|
Type: blockfeed.ReceivedBlock,
|
|
Data: &blockfeed.ReceivedBlockData{SignedBlock: wsb},
|
|
})
|
|
}
|
|
<-exitRoutine
|
|
}
|
|
|
|
func TestServer_StreamBlocksVerified_OnHeadUpdated(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MainnetConfig())
|
|
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
beaconState, privs := util.DeterministicGenesisState(t, 32)
|
|
b, err := util.GenerateFullBlock(beaconState, privs, util.DefaultBlockGenConfig(), 1)
|
|
require.NoError(t, err)
|
|
r, err := b.Block.HashTreeRoot()
|
|
require.NoError(t, err)
|
|
util.SaveBlock(t, ctx, db, b)
|
|
chainService := &chainMock.ChainService{State: beaconState}
|
|
server := &Server{
|
|
Ctx: ctx,
|
|
StateNotifier: chainService.StateNotifier(),
|
|
HeadFetcher: chainService,
|
|
BeaconDB: db,
|
|
}
|
|
exitRoutine := make(chan bool)
|
|
ctrl := gomock.NewController(t)
|
|
defer ctrl.Finish()
|
|
mockStream := mock.NewMockBeaconChain_StreamBlocksServer(ctrl)
|
|
mockStream.EXPECT().Send(b).Do(func(arg0 interface{}) {
|
|
exitRoutine <- true
|
|
})
|
|
mockStream.EXPECT().Context().Return(ctx).AnyTimes()
|
|
|
|
go func(tt *testing.T) {
|
|
assert.NoError(tt, server.StreamBlocks(ðpb.StreamBlocksRequest{
|
|
VerifiedOnly: true,
|
|
}, mockStream), "Could not call RPC method")
|
|
}(t)
|
|
|
|
// Send in a loop to ensure it is delivered (busy wait for the service to subscribe to the state feed).
|
|
for sent := 0; sent == 0; {
|
|
wsb, err := blocks.NewSignedBeaconBlock(b)
|
|
require.NoError(t, err)
|
|
sent = server.StateNotifier.StateFeed().Send(&feed.Event{
|
|
Type: statefeed.BlockProcessed,
|
|
Data: &statefeed.BlockProcessedData{Slot: b.Block.Slot, BlockRoot: r, SignedBlock: wsb},
|
|
})
|
|
}
|
|
<-exitRoutine
|
|
}
|
|
|
|
func TestServer_ListBeaconBlocks_NoResults(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
wanted := ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: make([]*ethpb.BeaconBlockContainer, 0),
|
|
TotalSize: int32(0),
|
|
NextPageToken: strconv.Itoa(0),
|
|
}
|
|
res, err := bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{
|
|
Slot: 0,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
res, err = bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{
|
|
Slot: 0,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
res, err = bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Root{
|
|
Root: make([]byte, 32),
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBeaconBlocks_Genesis(t *testing.T) {
|
|
t.Run("phase 0 block", func(t *testing.T) {
|
|
parentRoot := [32]byte{'a'}
|
|
blk := util.NewBeaconBlock()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
blkContainer := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{Phase0Block: blk}}
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBlocksGenesis(t, wrappedB, blkContainer)
|
|
})
|
|
t.Run("altair block", func(t *testing.T) {
|
|
parentRoot := [32]byte{'a'}
|
|
blk := util.NewBeaconBlockAltair()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
wrapped, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
blkContainer := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_AltairBlock{AltairBlock: blk}}
|
|
runListBlocksGenesis(t, wrapped, blkContainer)
|
|
})
|
|
t.Run("bellatrix block", func(t *testing.T) {
|
|
parentRoot := [32]byte{'a'}
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
wrapped, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
blinded, err := wrapped.ToBlinded()
|
|
assert.NoError(t, err)
|
|
blindedProto, err := blinded.PbBlindedBellatrixBlock()
|
|
assert.NoError(t, err)
|
|
blkContainer := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_BlindedBellatrixBlock{BlindedBellatrixBlock: blindedProto}}
|
|
runListBlocksGenesis(t, wrapped, blkContainer)
|
|
})
|
|
}
|
|
|
|
func runListBlocksGenesis(t *testing.T, blk interfaces.SignedBeaconBlock, blkContainer *ethpb.BeaconBlockContainer) {
|
|
resetFn := features.InitWithReset(&features.Flags{
|
|
EnableOnlyBlindedBeaconBlocks: true,
|
|
})
|
|
defer resetFn()
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
|
|
// Should throw an error if no genesis block is found.
|
|
_, err := bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.ErrorContains(t, "Could not find genesis", err)
|
|
|
|
root, err := blk.Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveBlock(ctx, blk))
|
|
require.NoError(t, db.SaveGenesisBlockRoot(ctx, root))
|
|
blkContainer.BlockRoot = root[:]
|
|
blkContainer.Canonical = true
|
|
|
|
wanted := ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{blkContainer},
|
|
NextPageToken: "0",
|
|
TotalSize: 1,
|
|
}
|
|
res, err := bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
if !proto.Equal(wanted, res) {
|
|
t.Errorf("Wanted %v, received %v", wanted, res)
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBeaconBlocks_Genesis_MultiBlocks(t *testing.T) {
|
|
t.Run("phase 0 block", func(t *testing.T) {
|
|
parentRoot := [32]byte{1, 2, 3}
|
|
blk := util.NewBeaconBlock()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
genB, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksGenesisMultiBlocks(t, genB, blockCreator)
|
|
})
|
|
t.Run("altair block", func(t *testing.T) {
|
|
parentRoot := [32]byte{1, 2, 3}
|
|
blk := util.NewBeaconBlockAltair()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlockAltair()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
gBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksGenesisMultiBlocks(t, gBlock, blockCreator)
|
|
})
|
|
t.Run("bellatrix block", func(t *testing.T) {
|
|
parentRoot := [32]byte{1, 2, 3}
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
blk.Block.ParentRoot = parentRoot[:]
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlockBellatrix()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
gBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksGenesisMultiBlocks(t, gBlock, blockCreator)
|
|
})
|
|
}
|
|
|
|
func runListBeaconBlocksGenesisMultiBlocks(t *testing.T, genBlock interfaces.SignedBeaconBlock,
|
|
blockCreator func(i types.Slot) interfaces.SignedBeaconBlock) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
// Should return the proper genesis block if it exists.
|
|
root, err := genBlock.Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveBlock(ctx, genBlock))
|
|
require.NoError(t, db.SaveGenesisBlockRoot(ctx, root))
|
|
|
|
count := types.Slot(100)
|
|
blks := make([]interfaces.SignedBeaconBlock, count)
|
|
for i := types.Slot(0); i < count; i++ {
|
|
blks[i] = blockCreator(i)
|
|
}
|
|
require.NoError(t, db.SaveBlocks(ctx, blks))
|
|
|
|
// Should throw an error if more than one blk returned.
|
|
_, err = bs.ListBeaconBlocks(ctx, ðpb.ListBlocksRequest{
|
|
QueryFilter: ðpb.ListBlocksRequest_Genesis{
|
|
Genesis: true,
|
|
},
|
|
})
|
|
require.NoError(t, err)
|
|
}
|
|
|
|
func TestServer_ListBeaconBlocks_Pagination(t *testing.T) {
|
|
params.SetupTestConfigCleanup(t)
|
|
params.OverrideBeaconConfig(params.MinimalSpecConfig())
|
|
t.Run("phase 0 block", func(t *testing.T) {
|
|
blk := util.NewBeaconBlock()
|
|
blk.Block.Slot = 300
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
containerCreator := func(i types.Slot, root []byte, canonical bool) *ethpb.BeaconBlockContainer {
|
|
b := util.NewBeaconBlock()
|
|
b.Block.Slot = i
|
|
ctr := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_Phase0Block{
|
|
Phase0Block: util.HydrateSignedBeaconBlock(b)},
|
|
BlockRoot: root,
|
|
Canonical: canonical}
|
|
return ctr
|
|
}
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksPagination(t, wrappedB, blockCreator, containerCreator)
|
|
})
|
|
t.Run("altair block", func(t *testing.T) {
|
|
blk := util.NewBeaconBlockAltair()
|
|
blk.Block.Slot = 300
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlockAltair()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
containerCreator := func(i types.Slot, root []byte, canonical bool) *ethpb.BeaconBlockContainer {
|
|
b := util.NewBeaconBlockAltair()
|
|
b.Block.Slot = i
|
|
ctr := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_AltairBlock{
|
|
AltairBlock: util.HydrateSignedBeaconBlockAltair(b)},
|
|
BlockRoot: root,
|
|
Canonical: canonical}
|
|
return ctr
|
|
}
|
|
orphanedB, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksPagination(t, orphanedB, blockCreator, containerCreator)
|
|
})
|
|
t.Run("bellatrix block", func(t *testing.T) {
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
blk.Block.Slot = 300
|
|
blockCreator := func(i types.Slot) interfaces.SignedBeaconBlock {
|
|
b := util.NewBeaconBlockBellatrix()
|
|
b.Block.Slot = i
|
|
wrappedB, err := blocks.NewSignedBeaconBlock(b)
|
|
assert.NoError(t, err)
|
|
return wrappedB
|
|
}
|
|
containerCreator := func(i types.Slot, root []byte, canonical bool) *ethpb.BeaconBlockContainer {
|
|
b := util.NewBeaconBlockBellatrix()
|
|
b.Block.Slot = i
|
|
ctr := ðpb.BeaconBlockContainer{
|
|
Block: ðpb.BeaconBlockContainer_BellatrixBlock{
|
|
BellatrixBlock: util.HydrateSignedBeaconBlockBellatrix(b)},
|
|
BlockRoot: root,
|
|
Canonical: canonical}
|
|
return ctr
|
|
}
|
|
orphanedB, err := blocks.NewSignedBeaconBlock(blk)
|
|
assert.NoError(t, err)
|
|
runListBeaconBlocksPagination(t, orphanedB, blockCreator, containerCreator)
|
|
})
|
|
}
|
|
|
|
func runListBeaconBlocksPagination(t *testing.T, orphanedBlk interfaces.SignedBeaconBlock,
|
|
blockCreator func(i types.Slot) interfaces.SignedBeaconBlock, containerCreator func(i types.Slot, root []byte, canonical bool) *ethpb.BeaconBlockContainer) {
|
|
|
|
db := dbTest.SetupDB(t)
|
|
chain := &chainMock.ChainService{
|
|
CanonicalRoots: map[[32]byte]bool{},
|
|
}
|
|
ctx := context.Background()
|
|
|
|
count := types.Slot(100)
|
|
blks := make([]interfaces.SignedBeaconBlock, count)
|
|
blkContainers := make([]*ethpb.BeaconBlockContainer, count)
|
|
for i := types.Slot(0); i < count; i++ {
|
|
b := blockCreator(i)
|
|
root, err := b.Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
chain.CanonicalRoots[root] = true
|
|
blks[i] = b
|
|
ctr, err := convertToBlockContainer(blks[i], root, true)
|
|
require.NoError(t, err)
|
|
blkContainers[i] = ctr
|
|
}
|
|
require.NoError(t, db.SaveBlocks(ctx, blks))
|
|
|
|
orphanedBlkRoot, err := orphanedBlk.Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
require.NoError(t, db.SaveBlock(ctx, orphanedBlk))
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
CanonicalFetcher: chain,
|
|
}
|
|
|
|
root6, err := blks[6].Block().HashTreeRoot()
|
|
require.NoError(t, err)
|
|
|
|
tests := []struct {
|
|
req *ethpb.ListBlocksRequest
|
|
res *ethpb.ListBeaconBlocksResponse
|
|
}{
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 5},
|
|
PageSize: 3},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{containerCreator(5, blkContainers[5].BlockRoot, blkContainers[5].Canonical)},
|
|
NextPageToken: "",
|
|
TotalSize: 1,
|
|
},
|
|
},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]},
|
|
PageSize: 3},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{containerCreator(6, blkContainers[6].BlockRoot, blkContainers[6].Canonical)},
|
|
TotalSize: 1,
|
|
NextPageToken: strconv.Itoa(0)}},
|
|
{req: ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: root6[:]}},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{containerCreator(6, blkContainers[6].BlockRoot, blkContainers[6].Canonical)},
|
|
TotalSize: 1, NextPageToken: strconv.Itoa(0)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 0},
|
|
PageSize: 100},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: blkContainers[0:params.BeaconConfig().SlotsPerEpoch],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(1),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 5},
|
|
PageSize: 3},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: blkContainers[43:46],
|
|
NextPageToken: "2",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(1),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 11},
|
|
PageSize: 7},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: blkContainers[95:96],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Epoch{Epoch: 12},
|
|
PageSize: 4},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: blkContainers[96:100],
|
|
NextPageToken: "",
|
|
TotalSize: int32(params.BeaconConfig().SlotsPerEpoch / 2)}},
|
|
{req: ðpb.ListBlocksRequest{
|
|
PageToken: strconv.Itoa(0),
|
|
QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 300},
|
|
PageSize: 3},
|
|
res: ðpb.ListBeaconBlocksResponse{
|
|
BlockContainers: []*ethpb.BeaconBlockContainer{containerCreator(300, orphanedBlkRoot[:], false)},
|
|
NextPageToken: "",
|
|
TotalSize: 1}},
|
|
}
|
|
|
|
for i, test := range tests {
|
|
t.Run(fmt.Sprintf("test_%d", i), func(t *testing.T) {
|
|
res, err := bs.ListBeaconBlocks(ctx, test.req)
|
|
require.NoError(t, err)
|
|
require.DeepSSZEqual(t, res, test.res)
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestServer_ListBeaconBlocks_Errors(t *testing.T) {
|
|
db := dbTest.SetupDB(t)
|
|
ctx := context.Background()
|
|
|
|
bs := &Server{
|
|
BeaconDB: db,
|
|
}
|
|
exceedsMax := int32(cmd.Get().MaxRPCPageSize + 1)
|
|
|
|
wanted := fmt.Sprintf("Requested page size %d can not be greater than max size %d", exceedsMax, cmd.Get().MaxRPCPageSize)
|
|
req := ðpb.ListBlocksRequest{PageToken: strconv.Itoa(0), PageSize: exceedsMax}
|
|
_, err := bs.ListBlocks(ctx, req)
|
|
assert.ErrorContains(t, wanted, err)
|
|
|
|
wanted = "Must specify a filter criteria for fetching"
|
|
req = ðpb.ListBlocksRequest{}
|
|
_, err = bs.ListBeaconBlocks(ctx, req)
|
|
assert.ErrorContains(t, wanted, err)
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Slot{Slot: 0}}
|
|
res, err := bs.ListBeaconBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Slot{}}
|
|
res, err = bs.ListBeaconBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}}
|
|
res, err = bs.ListBeaconBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
|
|
req = ðpb.ListBlocksRequest{QueryFilter: ðpb.ListBlocksRequest_Root{Root: []byte{'A'}}}
|
|
res, err = bs.ListBeaconBlocks(ctx, req)
|
|
require.NoError(t, err)
|
|
assert.Equal(t, 0, len(res.BlockContainers), "Wanted empty list")
|
|
assert.Equal(t, int32(0), res.TotalSize, "Wanted total size 0")
|
|
}
|