package initialsync import ( "context" "fmt" "sort" "sync" "testing" "time" "github.com/kevinms/leakybucket-go" core "github.com/libp2p/go-libp2p-core" "github.com/libp2p/go-libp2p-core/network" eth "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" dbtest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/flags" p2pm "github.com/prysmaticlabs/prysm/beacon-chain/p2p" p2pt "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing" beaconsync "github.com/prysmaticlabs/prysm/beacon-chain/sync" p2ppb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/params" "github.com/prysmaticlabs/prysm/shared/sliceutil" "github.com/prysmaticlabs/prysm/shared/testutil" "github.com/prysmaticlabs/prysm/shared/testutil/assert" "github.com/prysmaticlabs/prysm/shared/testutil/require" "github.com/sirupsen/logrus" logTest "github.com/sirupsen/logrus/hooks/test" ) func TestBlocksFetcher_InitStartStop(t *testing.T) { mc, p2p, _ := initializeTestServices(t, []uint64{}, []*peerData{}) ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher( ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, }, ) t.Run("check for leaked goroutines", func(t *testing.T) { err := fetcher.start() require.NoError(t, err) fetcher.stop() // should block up until all resources are reclaimed select { case <-fetcher.requestResponses(): default: t.Error("fetchResponses channel is leaked") } }) t.Run("re-starting of stopped fetcher", func(t *testing.T) { assert.ErrorContains(t, errFetcherCtxIsDone.Error(), fetcher.start()) }) t.Run("multiple stopping attempts", func(t *testing.T) { fetcher := newBlocksFetcher( context.Background(), &blocksFetcherConfig{ chain: mc, p2p: p2p, }) require.NoError(t, fetcher.start()) fetcher.stop() fetcher.stop() }) t.Run("cancellation", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) fetcher := newBlocksFetcher( ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, }) require.NoError(t, fetcher.start()) cancel() fetcher.stop() }) t.Run("peer filter capacity weight", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher( ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, peerFilterCapacityWeight: 2, }) require.NoError(t, fetcher.start()) assert.Equal(t, peerFilterCapacityWeight, fetcher.capacityWeight) }) } func TestBlocksFetcher_RoundRobin(t *testing.T) { blockBatchLimit := uint64(flags.Get().BlockBatchLimit) requestsGenerator := func(start, end uint64, batchSize uint64) []*fetchRequestParams { var requests []*fetchRequestParams for i := start; i <= end; i += batchSize { requests = append(requests, &fetchRequestParams{ start: i, count: batchSize, }) } return requests } tests := []struct { name string expectedBlockSlots []uint64 peers []*peerData requests []*fetchRequestParams }{ { name: "Single peer with all blocks", expectedBlockSlots: makeSequence(1, 3*blockBatchLimit), peers: []*peerData{ { blocks: makeSequence(1, 3*blockBatchLimit), finalizedEpoch: helpers.SlotToEpoch(3 * blockBatchLimit), headSlot: 3 * blockBatchLimit, }, }, requests: requestsGenerator(1, 3*blockBatchLimit, blockBatchLimit), }, { name: "Single peer with all blocks (many small requests)", expectedBlockSlots: makeSequence(1, 3*blockBatchLimit), peers: []*peerData{ { blocks: makeSequence(1, 3*blockBatchLimit), finalizedEpoch: helpers.SlotToEpoch(3 * blockBatchLimit), headSlot: 3 * blockBatchLimit, }, }, requests: requestsGenerator(1, 3*blockBatchLimit, blockBatchLimit/4), }, { name: "Multiple peers with all blocks", expectedBlockSlots: makeSequence(1, 3*blockBatchLimit), peers: []*peerData{ { blocks: makeSequence(1, 3*blockBatchLimit), finalizedEpoch: helpers.SlotToEpoch(3 * blockBatchLimit), headSlot: 3 * blockBatchLimit, }, { blocks: makeSequence(1, 3*blockBatchLimit), finalizedEpoch: helpers.SlotToEpoch(3 * blockBatchLimit), headSlot: 3 * blockBatchLimit, }, { blocks: makeSequence(1, 3*blockBatchLimit), finalizedEpoch: helpers.SlotToEpoch(3 * blockBatchLimit), headSlot: 3 * blockBatchLimit, }, }, requests: requestsGenerator(1, 3*blockBatchLimit, blockBatchLimit), }, { name: "Multiple peers with skipped slots", expectedBlockSlots: append(makeSequence(1, 64), makeSequence(500, 640)...), // up to 18th epoch peers: []*peerData{ { blocks: append(makeSequence(1, 64), makeSequence(500, 640)...), finalizedEpoch: 18, headSlot: 640, }, { blocks: append(makeSequence(1, 64), makeSequence(500, 640)...), finalizedEpoch: 18, headSlot: 640, }, { blocks: append(makeSequence(1, 64), makeSequence(500, 640)...), finalizedEpoch: 18, headSlot: 640, }, { blocks: append(makeSequence(1, 64), makeSequence(500, 640)...), finalizedEpoch: 18, headSlot: 640, }, { blocks: append(makeSequence(1, 64), makeSequence(500, 640)...), finalizedEpoch: 18, headSlot: 640, }, }, requests: []*fetchRequestParams{ { start: 1, count: blockBatchLimit, }, { start: blockBatchLimit + 1, count: blockBatchLimit, }, { start: 2*blockBatchLimit + 1, count: blockBatchLimit, }, { start: 500, count: 53, }, { start: 553, count: 200, }, }, }, { name: "Multiple peers with failures", expectedBlockSlots: makeSequence(1, 2*blockBatchLimit), peers: []*peerData{ { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, failureSlots: makeSequence(1, 32), // first epoch }, { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, }, requests: []*fetchRequestParams{ { start: 1, count: blockBatchLimit, }, { start: blockBatchLimit + 1, count: blockBatchLimit, }, }, }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { cache.initializeRootCache(tt.expectedBlockSlots, t) beaconDB := dbtest.SetupDB(t) p := p2pt.NewTestP2P(t) connectPeers(t, p, tt.peers, p.Peers()) cache.RLock() genesisRoot := cache.rootCache[0] cache.RUnlock() err := beaconDB.SaveBlock(context.Background(), testutil.NewBeaconBlock()) require.NoError(t, err) st := testutil.NewBeaconState() mc := &mock.ChainService{ State: st, Root: genesisRoot[:], DB: beaconDB, FinalizedCheckPoint: ð.Checkpoint{ Epoch: 0, }, } ctx, cancel := context.WithCancel(context.Background()) fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{ chain: mc, p2p: p, }) require.NoError(t, fetcher.start()) var wg sync.WaitGroup wg.Add(len(tt.requests)) // how many block requests we are going to make go func() { wg.Wait() log.Debug("Stopping fetcher") fetcher.stop() }() processFetchedBlocks := func() ([]*eth.SignedBeaconBlock, error) { defer cancel() var unionRespBlocks []*eth.SignedBeaconBlock for { select { case resp, ok := <-fetcher.requestResponses(): if !ok { // channel closed, aggregate return unionRespBlocks, nil } if resp.err != nil { log.WithError(resp.err).Debug("Block fetcher returned error") } else { unionRespBlocks = append(unionRespBlocks, resp.blocks...) if len(resp.blocks) == 0 { log.WithFields(logrus.Fields{ "start": resp.start, "count": resp.count, }).Debug("Received empty slot") } } wg.Done() case <-ctx.Done(): log.Debug("Context closed, exiting goroutine") return unionRespBlocks, nil } } } maxExpectedBlocks := uint64(0) for _, requestParams := range tt.requests { err = fetcher.scheduleRequest(context.Background(), requestParams.start, requestParams.count) assert.NoError(t, err) maxExpectedBlocks += requestParams.count } blocks, err := processFetchedBlocks() assert.NoError(t, err) sort.Slice(blocks, func(i, j int) bool { return blocks[i].Block.Slot < blocks[j].Block.Slot }) slots := make([]uint64, len(blocks)) for i, block := range blocks { slots[i] = block.Block.Slot } log.WithFields(logrus.Fields{ "blocksLen": len(blocks), "slots": slots, }).Debug("Finished block fetching") if len(blocks) > int(maxExpectedBlocks) { t.Errorf("Too many blocks returned. Wanted %d got %d", maxExpectedBlocks, len(blocks)) } assert.Equal(t, len(tt.expectedBlockSlots), len(blocks), "Processes wrong number of blocks") var receivedBlockSlots []uint64 for _, blk := range blocks { receivedBlockSlots = append(receivedBlockSlots, blk.Block.Slot) } missing := sliceutil.NotUint64( sliceutil.IntersectionUint64(tt.expectedBlockSlots, receivedBlockSlots), tt.expectedBlockSlots) if len(missing) > 0 { t.Errorf("Missing blocks at slots %v", missing) } }) } } func TestBlocksFetcher_scheduleRequest(t *testing.T) { blockBatchLimit := uint64(flags.Get().BlockBatchLimit) t.Run("context cancellation", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{}) cancel() assert.ErrorContains(t, "context canceled", fetcher.scheduleRequest(ctx, 1, blockBatchLimit)) }) t.Run("unblock on context cancellation", func(t *testing.T) { fetcher := newBlocksFetcher(context.Background(), &blocksFetcherConfig{}) for i := 0; i < maxPendingRequests; i++ { assert.NoError(t, fetcher.scheduleRequest(context.Background(), 1, blockBatchLimit)) } // Will block on next request (and wait until requests are either processed or context is closed). go func() { fetcher.cancel() }() assert.ErrorContains(t, errFetcherCtxIsDone.Error(), fetcher.scheduleRequest(context.Background(), 1, blockBatchLimit)) }) } func TestBlocksFetcher_handleRequest(t *testing.T) { blockBatchLimit := uint64(flags.Get().BlockBatchLimit) chainConfig := struct { expectedBlockSlots []uint64 peers []*peerData }{ expectedBlockSlots: makeSequence(1, blockBatchLimit), peers: []*peerData{ { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, }, } mc, p2p, _ := initializeTestServices(t, chainConfig.expectedBlockSlots, chainConfig.peers) t.Run("context cancellation", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, }) cancel() response := fetcher.handleRequest(ctx, 1, blockBatchLimit) assert.ErrorContains(t, "context canceled", response.err) }) t.Run("receive blocks", func(t *testing.T) { ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, }) requestCtx, reqCancel := context.WithTimeout(context.Background(), 2*time.Second) defer reqCancel() go func() { response := fetcher.handleRequest(requestCtx, 1 /* start */, blockBatchLimit /* count */) select { case <-ctx.Done(): case fetcher.fetchResponses <- response: } }() var blocks []*eth.SignedBeaconBlock select { case <-ctx.Done(): t.Error(ctx.Err()) case resp := <-fetcher.requestResponses(): if resp.err != nil { t.Error(resp.err) } else { blocks = resp.blocks } } if uint64(len(blocks)) != blockBatchLimit { t.Errorf("incorrect number of blocks returned, expected: %v, got: %v", blockBatchLimit, len(blocks)) } var receivedBlockSlots []uint64 for _, blk := range blocks { receivedBlockSlots = append(receivedBlockSlots, blk.Block.Slot) } missing := sliceutil.NotUint64( sliceutil.IntersectionUint64(chainConfig.expectedBlockSlots, receivedBlockSlots), chainConfig.expectedBlockSlots) if len(missing) > 0 { t.Errorf("Missing blocks at slots %v", missing) } }) } func TestBlocksFetcher_requestBeaconBlocksByRange(t *testing.T) { blockBatchLimit := uint64(flags.Get().BlockBatchLimit) chainConfig := struct { expectedBlockSlots []uint64 peers []*peerData }{ expectedBlockSlots: makeSequence(1, 320), peers: []*peerData{ { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, { blocks: makeSequence(1, 320), finalizedEpoch: 8, headSlot: 320, }, }, } mc, p2p, _ := initializeTestServices(t, chainConfig.expectedBlockSlots, chainConfig.peers) ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher( ctx, &blocksFetcherConfig{ chain: mc, p2p: p2p, }) _, peerIDs := p2p.Peers().BestFinalized(params.BeaconConfig().MaxPeersToSync, helpers.SlotToEpoch(mc.HeadSlot())) req := &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 1, Step: 1, Count: blockBatchLimit, } blocks, err := fetcher.requestBlocks(ctx, req, peerIDs[0]) assert.NoError(t, err) assert.Equal(t, blockBatchLimit, uint64(len(blocks)), "Incorrect number of blocks returned") // Test context cancellation. ctx, cancel = context.WithCancel(context.Background()) cancel() _, err = fetcher.requestBlocks(ctx, req, peerIDs[0]) assert.ErrorContains(t, "context canceled", err) } func TestBlocksFetcher_RequestBlocksRateLimitingLocks(t *testing.T) { p1 := p2pt.NewTestP2P(t) p2 := p2pt.NewTestP2P(t) p3 := p2pt.NewTestP2P(t) p1.Connect(p2) p1.Connect(p3) require.Equal(t, 2, len(p1.BHost.Network().Peers()), "Expected peers to be connected") req := &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, } topic := p2pm.RPCBlocksByRangeTopic protocol := core.ProtocolID(topic + p2.Encoding().ProtocolSuffix()) streamHandlerFn := func(stream network.Stream) { assert.NoError(t, stream.Close()) } p2.BHost.SetStreamHandler(protocol, streamHandlerFn) p3.BHost.SetStreamHandler(protocol, streamHandlerFn) burstFactor := uint64(flags.Get().BlockBatchLimitBurstFactor) ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{p2p: p1}) fetcher.rateLimiter = leakybucket.NewCollector(float64(req.Count), int64(req.Count*burstFactor), false) hook := logTest.NewGlobal() wg := new(sync.WaitGroup) wg.Add(1) go func() { // Exhaust available rate for p2, so that rate limiting is triggered. for i := uint64(0); i <= burstFactor; i++ { if i == burstFactor { // The next request will trigger rate limiting for p2. Now, allow concurrent // p3 data request (p3 shouldn't be rate limited). time.AfterFunc(1*time.Second, func() { wg.Done() }) } _, err := fetcher.requestBlocks(ctx, req, p2.PeerID()) if err != nil { assert.ErrorContains(t, errFetcherCtxIsDone.Error(), err) } } }() // Wait until p2 exhausts its rate and is spinning on rate limiting timer. wg.Wait() // The next request should NOT trigger rate limiting as rate is exhausted for p2, not p3. ch := make(chan struct{}, 1) go func() { _, err := fetcher.requestBlocks(ctx, req, p3.PeerID()) assert.NoError(t, err) ch <- struct{}{} }() timer := time.NewTimer(2 * time.Second) select { case <-timer.C: t.Error("p3 takes too long to respond: lock contention") case <-ch: // p3 responded w/o waiting for rate limiter's lock (on which p2 spins). } // Make sure that p2 has been rate limited. require.LogsContain(t, hook, fmt.Sprintf("msg=\"Slowing down for rate limit\" peer=%s", p2.PeerID())) } func TestBlocksFetcher_requestBlocksFromPeerReturningInvalidBlocks(t *testing.T) { p1 := p2pt.NewTestP2P(t) tests := []struct { name string req *p2ppb.BeaconBlocksByRangeRequest handlerGenFn func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) wantedErr string validate func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) }{ { name: "no error", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 4, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { for i := req.StartSlot; i < req.StartSlot+req.Count*req.Step; i += req.Step { blk := testutil.NewBeaconBlock() blk.Block.Slot = i assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) } assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, req.Count, uint64(len(blocks))) }, }, { name: "too many blocks", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { for i := req.StartSlot; i < req.StartSlot+req.Count*req.Step+1; i += req.Step { blk := testutil.NewBeaconBlock() blk.Block.Slot = i assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) } assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), }, { name: "not in a consecutive order", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { blk := testutil.NewBeaconBlock() blk.Block.Slot = 163 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) blk = testutil.NewBeaconBlock() blk.Block.Slot = 162 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), }, { name: "same slot number", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { blk := testutil.NewBeaconBlock() blk.Block.Slot = 160 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) blk = testutil.NewBeaconBlock() blk.Block.Slot = 160 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), }, { name: "slot is too low", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { defer func() { assert.NoError(t, stream.Close()) }() for i := req.StartSlot; i < req.StartSlot+req.Count*req.Step; i += req.Step { blk := testutil.NewBeaconBlock() // Patch mid block, with invalid slot number. if i == (req.StartSlot + req.Count*req.Step/2) { blk.Block.Slot = req.StartSlot - 1 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) break } else { blk.Block.Slot = i assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) } } } }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, }, { name: "slot is too high", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 1, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { defer func() { assert.NoError(t, stream.Close()) }() for i := req.StartSlot; i < req.StartSlot+req.Count*req.Step; i += req.Step { blk := testutil.NewBeaconBlock() // Patch mid block, with invalid slot number. if i == (req.StartSlot + req.Count*req.Step/2) { blk.Block.Slot = req.StartSlot + req.Count*req.Step assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) break } else { blk.Block.Slot = i assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) } } } }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, }, { name: "valid step increment", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 5, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { blk := testutil.NewBeaconBlock() blk.Block.Slot = 100 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) blk = testutil.NewBeaconBlock() blk.Block.Slot = 105 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 2, len(blocks)) }, }, { name: "invalid step increment", req: &p2ppb.BeaconBlocksByRangeRequest{ StartSlot: 100, Step: 5, Count: 64, }, handlerGenFn: func(req *p2ppb.BeaconBlocksByRangeRequest) func(stream network.Stream) { return func(stream network.Stream) { blk := testutil.NewBeaconBlock() blk.Block.Slot = 100 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) blk = testutil.NewBeaconBlock() blk.Block.Slot = 103 assert.NoError(t, beaconsync.WriteChunk(stream, p1.Encoding(), blk)) assert.NoError(t, stream.Close()) } }, validate: func(req *p2ppb.BeaconBlocksByRangeRequest, blocks []*eth.SignedBeaconBlock) { assert.Equal(t, 0, len(blocks)) }, wantedErr: beaconsync.ErrInvalidFetchedData.Error(), }, } topic := p2pm.RPCBlocksByRangeTopic protocol := core.ProtocolID(topic + p1.Encoding().ProtocolSuffix()) ctx, cancel := context.WithCancel(context.Background()) defer cancel() fetcher := newBlocksFetcher(ctx, &blocksFetcherConfig{p2p: p1}) fetcher.rateLimiter = leakybucket.NewCollector(0.000001, 640, false) for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { p2 := p2pt.NewTestP2P(t) p1.Connect(p2) p2.BHost.SetStreamHandler(protocol, tt.handlerGenFn(tt.req)) blocks, err := fetcher.requestBlocks(ctx, tt.req, p2.PeerID()) if tt.wantedErr != "" { assert.ErrorContains(t, tt.wantedErr, err) } else { assert.NoError(t, err) tt.validate(tt.req, blocks) } }) } }