package initialsync import ( "context" "fmt" "io/ioutil" "sync" "testing" "time" "github.com/ethereum/go-ethereum/p2p/enr" "github.com/libp2p/go-libp2p-core/network" "github.com/libp2p/go-libp2p-core/peer" types "github.com/prysmaticlabs/eth2-types" mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing" "github.com/prysmaticlabs/prysm/beacon-chain/core/helpers" "github.com/prysmaticlabs/prysm/beacon-chain/db" dbtest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/p2p/peers" p2pt "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing" p2pTypes "github.com/prysmaticlabs/prysm/beacon-chain/p2p/types" beaconsync "github.com/prysmaticlabs/prysm/beacon-chain/sync" "github.com/prysmaticlabs/prysm/cmd/beacon-chain/flags" p2ppb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" eth "github.com/prysmaticlabs/prysm/proto/eth/v1alpha1" "github.com/prysmaticlabs/prysm/proto/eth/v1alpha1/wrapper" "github.com/prysmaticlabs/prysm/shared/bytesutil" "github.com/prysmaticlabs/prysm/shared/featureconfig" "github.com/prysmaticlabs/prysm/shared/hashutil" "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/prysmaticlabs/prysm/shared/timeutils" "github.com/sirupsen/logrus" ) type testCache struct { sync.RWMutex rootCache map[types.Slot][32]byte parentSlotCache map[types.Slot]types.Slot } var cache = &testCache{} type peerData struct { blocks []types.Slot // slots that peer has blocks finalizedEpoch types.Epoch headSlot types.Slot failureSlots []types.Slot // slots at which the peer will return an error forkedPeer bool } func TestMain(m *testing.M) { logrus.SetLevel(logrus.DebugLevel) logrus.SetOutput(ioutil.Discard) resetCfg := featureconfig.InitWithReset(&featureconfig.Flags{ EnablePeerScorer: true, }) defer resetCfg() resetFlags := flags.Get() flags.Init(&flags.GlobalFlags{ BlockBatchLimit: 64, BlockBatchLimitBurstFactor: 10, }) defer func() { flags.Init(resetFlags) }() m.Run() } func initializeTestServices(t *testing.T, slots []types.Slot, peers []*peerData) (*mock.ChainService, *p2pt.TestP2P, db.Database) { cache.initializeRootCache(slots, t) beaconDB := dbtest.SetupDB(t) p := p2pt.NewTestP2P(t) connectPeers(t, p, peers, p.Peers()) cache.RLock() genesisRoot := cache.rootCache[0] cache.RUnlock() err := beaconDB.SaveBlock(context.Background(), wrapper.WrappedPhase0SignedBeaconBlock(testutil.NewBeaconBlock())) require.NoError(t, err) st, err := testutil.NewBeaconState() require.NoError(t, err) return &mock.ChainService{ State: st, Root: genesisRoot[:], DB: beaconDB, FinalizedCheckPoint: ð.Checkpoint{ Epoch: 0, }, }, p, beaconDB } // makeGenesisTime where now is the current slot. func makeGenesisTime(currentSlot types.Slot) time.Time { return timeutils.Now().Add(-1 * time.Second * time.Duration(currentSlot) * time.Duration(params.BeaconConfig().SecondsPerSlot)) } // sanity test on helper function func TestMakeGenesisTime(t *testing.T) { currentSlot := types.Slot(64) gt := makeGenesisTime(currentSlot) require.Equal(t, currentSlot, helpers.SlotsSince(gt)) } // helper function for sequences of block slots func makeSequence(start, end types.Slot) []types.Slot { if end < start { panic("cannot make sequence where end is before start") } seq := make([]types.Slot, 0, end-start+1) for i := start; i <= end; i++ { seq = append(seq, i) } return seq } func (c *testCache) initializeRootCache(reqSlots []types.Slot, t *testing.T) { c.Lock() defer c.Unlock() c.rootCache = make(map[types.Slot][32]byte) c.parentSlotCache = make(map[types.Slot]types.Slot) parentSlot := types.Slot(0) genesisBlock := testutil.NewBeaconBlock().Block genesisRoot, err := genesisBlock.HashTreeRoot() require.NoError(t, err) c.rootCache[0] = genesisRoot parentRoot := genesisRoot for _, slot := range reqSlots { currentBlock := testutil.NewBeaconBlock().Block currentBlock.Slot = slot currentBlock.ParentRoot = parentRoot[:] parentRoot, err = currentBlock.HashTreeRoot() require.NoError(t, err) c.rootCache[slot] = parentRoot c.parentSlotCache[slot] = parentSlot parentSlot = slot } } // sanity test on helper function func TestMakeSequence(t *testing.T) { got := makeSequence(3, 5) want := []types.Slot{3, 4, 5} require.DeepEqual(t, want, got) } // Connect peers with local host. This method sets up peer statuses and the appropriate handlers // for each test peer. func connectPeers(t *testing.T, host *p2pt.TestP2P, data []*peerData, peerStatus *peers.Status) { for _, d := range data { connectPeer(t, host, d, peerStatus) } } // connectPeer connects a peer to a local host. func connectPeer(t *testing.T, host *p2pt.TestP2P, datum *peerData, peerStatus *peers.Status) peer.ID { const topic = "/eth2/beacon_chain/req/beacon_blocks_by_range/1/ssz_snappy" p := p2pt.NewTestP2P(t) p.SetStreamHandler(topic, func(stream network.Stream) { defer func() { assert.NoError(t, stream.Close()) }() req := &p2ppb.BeaconBlocksByRangeRequest{} assert.NoError(t, p.Encoding().DecodeWithMaxLength(stream, req)) requestedBlocks := makeSequence(req.StartSlot, req.StartSlot.Add((req.Count-1)*req.Step)) // Expected failure range if len(sliceutil.IntersectionSlot(datum.failureSlots, requestedBlocks)) > 0 { _, err := stream.Write([]byte{0x01}) assert.NoError(t, err) msg := p2pTypes.ErrorMessage("bad") _, err = p.Encoding().EncodeWithMaxLength(stream, &msg) assert.NoError(t, err) return } // Determine the correct subset of blocks to return as dictated by the test scenario. slots := sliceutil.IntersectionSlot(datum.blocks, requestedBlocks) ret := make([]*eth.SignedBeaconBlock, 0) for _, slot := range slots { if (slot - req.StartSlot).Mod(req.Step) != 0 { continue } cache.RLock() parentRoot := cache.rootCache[cache.parentSlotCache[slot]] cache.RUnlock() blk := testutil.NewBeaconBlock() blk.Block.Slot = slot blk.Block.ParentRoot = parentRoot[:] // If forked peer, give a different parent root. if datum.forkedPeer { newRoot := hashutil.Hash(parentRoot[:]) blk.Block.ParentRoot = newRoot[:] } ret = append(ret, blk) currRoot, err := blk.Block.HashTreeRoot() require.NoError(t, err) logrus.Tracef("block with slot %d , signing root %#x and parent root %#x", slot, currRoot, parentRoot) } if uint64(len(ret)) > req.Count { ret = ret[:req.Count] } for i := 0; i < len(ret); i++ { assert.NoError(t, beaconsync.WriteChunk(stream, nil, p.Encoding(), ret[i])) } }) p.Connect(host) peerStatus.Add(new(enr.Record), p.PeerID(), nil, network.DirOutbound) peerStatus.SetConnectionState(p.PeerID(), peers.PeerConnected) peerStatus.SetChainState(p.PeerID(), &p2ppb.Status{ ForkDigest: params.BeaconConfig().GenesisForkVersion, FinalizedRoot: []byte(fmt.Sprintf("finalized_root %d", datum.finalizedEpoch)), FinalizedEpoch: datum.finalizedEpoch, HeadRoot: bytesutil.PadTo([]byte("head_root"), 32), HeadSlot: datum.headSlot, }) return p.PeerID() } // extendBlockSequence extends block chain sequentially (creating genesis block, if necessary). func extendBlockSequence(t *testing.T, inSeq []*eth.SignedBeaconBlock, size int) []*eth.SignedBeaconBlock { // Start from the original sequence. outSeq := make([]*eth.SignedBeaconBlock, len(inSeq)+size) copy(outSeq, inSeq) // See if genesis block needs to be created. startSlot := len(inSeq) if len(inSeq) == 0 { outSeq[0] = testutil.NewBeaconBlock() outSeq[0].Block.StateRoot = testutil.Random32Bytes(t) startSlot++ outSeq = append(outSeq, nil) } // Extend block chain sequentially. for slot := startSlot; slot < len(outSeq); slot++ { outSeq[slot] = testutil.NewBeaconBlock() outSeq[slot].Block.Slot = types.Slot(slot) parentRoot, err := outSeq[slot-1].Block.HashTreeRoot() require.NoError(t, err) outSeq[slot].Block.ParentRoot = parentRoot[:] // Make sure that blocks having the same slot number, produce different hashes. // That way different branches/forks will have different blocks for the same slots. outSeq[slot].Block.StateRoot = testutil.Random32Bytes(t) } return outSeq } // connectPeerHavingBlocks connect host with a peer having provided blocks. func connectPeerHavingBlocks( t *testing.T, host *p2pt.TestP2P, blocks []*eth.SignedBeaconBlock, finalizedSlot types.Slot, peerStatus *peers.Status, ) peer.ID { p := p2pt.NewTestP2P(t) p.SetStreamHandler("/eth2/beacon_chain/req/beacon_blocks_by_range/1/ssz_snappy", func(stream network.Stream) { defer func() { _err := stream.Close() _ = _err }() req := &p2ppb.BeaconBlocksByRangeRequest{} assert.NoError(t, p.Encoding().DecodeWithMaxLength(stream, req)) for i := req.StartSlot; i < req.StartSlot.Add(req.Count*req.Step); i += types.Slot(req.Step) { if uint64(i) >= uint64(len(blocks)) { break } require.NoError(t, beaconsync.WriteChunk(stream, nil, p.Encoding(), blocks[i])) } }) p.SetStreamHandler("/eth2/beacon_chain/req/beacon_blocks_by_root/1/ssz_snappy", func(stream network.Stream) { defer func() { _err := stream.Close() _ = _err }() req := new(p2pTypes.BeaconBlockByRootsReq) assert.NoError(t, p.Encoding().DecodeWithMaxLength(stream, req)) if len(*req) == 0 { return } for _, expectedRoot := range *req { for _, blk := range blocks { if root, err := blk.Block.HashTreeRoot(); err == nil && expectedRoot == root { log.Printf("Found blocks_by_root: %#x for slot: %v", root, blk.Block.Slot) _, err := stream.Write([]byte{0x00}) assert.NoError(t, err, "Failed to write to stream") _, err = p.Encoding().EncodeWithMaxLength(stream, blk) assert.NoError(t, err, "Could not send response back") } } } }) p.Connect(host) finalizedEpoch := helpers.SlotToEpoch(finalizedSlot) headRoot, err := blocks[len(blocks)-1].Block.HashTreeRoot() require.NoError(t, err) peerStatus.Add(new(enr.Record), p.PeerID(), nil, network.DirOutbound) peerStatus.SetConnectionState(p.PeerID(), peers.PeerConnected) peerStatus.SetChainState(p.PeerID(), &p2ppb.Status{ ForkDigest: params.BeaconConfig().GenesisForkVersion, FinalizedRoot: []byte(fmt.Sprintf("finalized_root %d", finalizedEpoch)), FinalizedEpoch: finalizedEpoch, HeadRoot: headRoot[:], HeadSlot: blocks[len(blocks)-1].Block.Slot, }) return p.PeerID() }