erigon-pulse/cl/sentinel/handlers/blocks_by_root_test.go
Bayram Guvanjov a2f375c0b1
Add blocksByRange & blocksByRoot P2P rpc handlers (#8885)
This PR is ready to review. 

PR introduces `blocksByRange` and `blocksByRoot` P2P RPC methods
- `blocksByRange` - allows peers to request a range of blocks
- `blocksByRoot` - enables block requests using their root hashes(list
format)

Reference:
https://github.com/ethereum/consensus-specs/blob/dev/specs/phase0/p2p-interface.md#beaconblocksbyrange
2023-12-23 14:32:00 +01:00

145 lines
4.0 KiB
Go

package handlers
import (
"bytes"
"context"
"encoding/binary"
"fmt"
"io"
"testing"
"github.com/golang/snappy"
libcommon "github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon/cl/clparams"
"github.com/ledgerwatch/erigon/cl/cltypes"
"github.com/ledgerwatch/erigon/cl/cltypes/solid"
"github.com/ledgerwatch/erigon/cl/fork"
"github.com/ledgerwatch/erigon/cl/persistence"
"github.com/ledgerwatch/erigon/cl/persistence/beacon_indicies"
"github.com/ledgerwatch/erigon/cl/sentinel/communication"
"github.com/ledgerwatch/erigon/cl/sentinel/communication/ssz_snappy"
"github.com/ledgerwatch/erigon/cl/sentinel/peers"
"github.com/ledgerwatch/erigon/cl/utils"
"github.com/libp2p/go-libp2p"
"github.com/libp2p/go-libp2p/core/peer"
"github.com/libp2p/go-libp2p/core/protocol"
"github.com/stretchr/testify/require"
)
func TestBlocksByRangeHandler(t *testing.T) {
ctx := context.Background()
listenAddrHost := "/ip4/127.0.0.1/tcp/6000"
host, err := libp2p.New(libp2p.ListenAddrStrings(listenAddrHost))
require.NoError(t, err)
listenAddrHost1 := "/ip4/127.0.0.1/tcp/6001"
host1, err := libp2p.New(libp2p.ListenAddrStrings(listenAddrHost1))
require.NoError(t, err)
err = host.Connect(ctx, peer.AddrInfo{
ID: host1.ID(),
Addrs: host1.Addrs(),
})
require.NoError(t, err)
peersPool := peers.NewPool()
beaconDB, indiciesDB := setupStore(t)
store := persistence.NewBeaconChainDatabaseFilesystem(beaconDB, nil, &clparams.MainnetBeaconConfig)
tx, _ := indiciesDB.BeginRw(ctx)
startSlot := uint64(100)
count := uint64(10)
expBlocks := populateDatabaseWithBlocks(t, store, tx, startSlot, count)
var blockRoots []libcommon.Hash
blockRoots, _, _ = beacon_indicies.ReadBeaconBlockRootsInSlotRange(ctx, tx, startSlot, startSlot+count)
tx.Commit()
genesisCfg, _, beaconCfg := clparams.GetConfigsByNetwork(1)
c := NewConsensusHandlers(
ctx,
beaconDB,
indiciesDB,
host,
peersPool,
beaconCfg,
genesisCfg,
&cltypes.Metadata{},
)
c.Start()
var req solid.HashListSSZ = solid.NewHashList(len(expBlocks))
for _, block := range blockRoots {
req.Append(block)
}
var reqBuf bytes.Buffer
if err := ssz_snappy.EncodeAndWrite(&reqBuf, req); err != nil {
return
}
reqData := libcommon.CopyBytes(reqBuf.Bytes())
stream, err := host1.NewStream(ctx, host.ID(), protocol.ID(communication.BeaconBlocksByRootProtocolV1))
require.NoError(t, err)
_, err = stream.Write(reqData)
require.NoError(t, err)
firstByte := make([]byte, 1)
_, err = stream.Read(firstByte)
require.NoError(t, err)
require.Equal(t, firstByte[0], byte(1))
for i := 0; i < len(blockRoots); i++ {
forkDigest := make([]byte, 4)
_, err := stream.Read(forkDigest)
if err != nil && err != io.EOF {
require.NoError(t, err)
}
encodedLn, _, err := ssz_snappy.ReadUvarint(stream)
require.NoError(t, err)
raw := make([]byte, encodedLn)
sr := snappy.NewReader(stream)
bytesRead := 0
for bytesRead < int(encodedLn) {
n, err := sr.Read(raw[bytesRead:])
if err != nil {
require.NoError(t, err)
}
bytesRead += n
}
// Fork digests
respForkDigest := binary.BigEndian.Uint32(forkDigest)
if respForkDigest == 0 {
require.NoError(t, fmt.Errorf("null fork digest"))
}
version, err := fork.ForkDigestVersion(utils.Uint32ToBytes4(respForkDigest), beaconCfg, genesisCfg.GenesisValidatorRoot)
if err != nil {
require.NoError(t, err)
}
block := cltypes.NewSignedBeaconBlock(&clparams.MainnetBeaconConfig)
if err = block.DecodeSSZ(raw, int(version)); err != nil {
require.NoError(t, err)
return
}
require.Equal(t, expBlocks[i].Block.Slot, block.Block.Slot)
require.Equal(t, expBlocks[i].Block.StateRoot, block.Block.StateRoot)
require.Equal(t, expBlocks[i].Block.ParentRoot, block.Block.ParentRoot)
require.Equal(t, expBlocks[i].Block.ProposerIndex, block.Block.ProposerIndex)
require.Equal(t, expBlocks[i].Block.Body.ExecutionPayload.BlockNumber, block.Block.Body.ExecutionPayload.BlockNumber)
}
_, err = stream.Read(make([]byte, 1))
if err != io.EOF {
t.Fatal("Stream is not empty")
}
defer indiciesDB.Close()
defer tx.Rollback()
}