2019-09-20 17:08:32 +00:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/libp2p/go-libp2p-core/network"
|
2019-09-20 17:54:32 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/protocol"
|
2019-11-27 05:08:18 +00:00
|
|
|
ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1"
|
2019-09-20 17:08:32 +00:00
|
|
|
"github.com/prysmaticlabs/go-ssz"
|
|
|
|
mock "github.com/prysmaticlabs/prysm/beacon-chain/blockchain/testing"
|
|
|
|
dbtest "github.com/prysmaticlabs/prysm/beacon-chain/db/testing"
|
|
|
|
p2ptest "github.com/prysmaticlabs/prysm/beacon-chain/p2p/testing"
|
2019-09-20 17:54:32 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/sync/peerstatus"
|
2019-09-20 17:08:32 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
logrus.SetLevel(logrus.DebugLevel)
|
|
|
|
}
|
|
|
|
|
|
|
|
// /- b1 - b2
|
|
|
|
// b0
|
|
|
|
// \- b3
|
|
|
|
// Test b1 was missing then received and we can process b0 -> b1 -> b2
|
|
|
|
func TestRegularSyncBeaconBlockSubscriber_ProcessPendingBlocks1(t *testing.T) {
|
|
|
|
db := dbtest.SetupDB(t)
|
|
|
|
defer dbtest.TeardownDB(t, db)
|
|
|
|
|
|
|
|
r := &RegularSync{
|
2019-12-03 15:56:04 +00:00
|
|
|
db: db,
|
|
|
|
chain: &mock.ChainService{
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
},
|
|
|
|
},
|
2019-09-20 17:08:32 +00:00
|
|
|
slotToPendingBlocks: make(map[uint64]*ethpb.BeaconBlock),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
|
|
|
}
|
|
|
|
|
|
|
|
b0 := ðpb.BeaconBlock{}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b0); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
b0Root, _ := ssz.SigningRoot(b0)
|
|
|
|
b3 := ðpb.BeaconBlock{Slot: 3, ParentRoot: b0Root[:]}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b3); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
// Incomplete block link
|
|
|
|
b1 := ðpb.BeaconBlock{Slot: 1, ParentRoot: b0Root[:]}
|
|
|
|
b1Root, _ := ssz.SigningRoot(b1)
|
|
|
|
b2 := ðpb.BeaconBlock{Slot: 2, ParentRoot: b1Root[:]}
|
|
|
|
b2Root, _ := ssz.SigningRoot(b1)
|
|
|
|
|
|
|
|
// Add b2 to the cache
|
|
|
|
r.slotToPendingBlocks[b2.Slot] = b2
|
|
|
|
r.seenPendingBlocks[b2Root] = true
|
|
|
|
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 1 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
if len(r.seenPendingBlocks) != 1 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add b1 to the cache
|
|
|
|
r.slotToPendingBlocks[b1.Slot] = b1
|
|
|
|
r.seenPendingBlocks[b1Root] = true
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b1); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
if len(r.seenPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// /- b1 - b2 - b5
|
|
|
|
// b0
|
|
|
|
// \- b3 - b4
|
|
|
|
// Test b2 and b3 were missed, after receiving them we can process 2 chains.
|
|
|
|
func TestRegularSyncBeaconBlockSubscriber_ProcessPendingBlocks2(t *testing.T) {
|
|
|
|
db := dbtest.SetupDB(t)
|
|
|
|
defer dbtest.TeardownDB(t, db)
|
|
|
|
p1 := p2ptest.NewTestP2P(t)
|
|
|
|
p2 := p2ptest.NewTestP2P(t)
|
|
|
|
p1.Connect(p2)
|
|
|
|
if len(p1.Host.Network().Peers()) != 1 {
|
|
|
|
t.Error("Expected peers to be connected")
|
|
|
|
}
|
|
|
|
pcl := protocol.ID("/eth2/beacon_chain/req/hello/1/ssz")
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(1)
|
|
|
|
p2.Host.SetStreamHandler(pcl, func(stream network.Stream) {
|
|
|
|
defer wg.Done()
|
|
|
|
code, errMsg, err := ReadStatusCode(stream, p1.Encoding())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if code == 0 {
|
|
|
|
t.Error("Expected a non-zero code")
|
|
|
|
}
|
|
|
|
if errMsg != errWrongForkVersion.Error() {
|
|
|
|
t.Logf("Received error string len %d, wanted error string len %d", len(errMsg), len(errWrongForkVersion.Error()))
|
|
|
|
t.Errorf("Received unexpected message response in the stream: %s. Wanted %s.", errMsg, errWrongForkVersion.Error())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
r := &RegularSync{
|
2019-12-03 15:56:04 +00:00
|
|
|
p2p: p1,
|
|
|
|
db: db,
|
|
|
|
chain: &mock.ChainService{
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 0,
|
|
|
|
},
|
|
|
|
}, slotToPendingBlocks: make(map[uint64]*ethpb.BeaconBlock),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
2019-09-20 17:08:32 +00:00
|
|
|
}
|
2019-09-20 17:54:32 +00:00
|
|
|
peerstatus.Set(p2.PeerID(), &pb.Status{})
|
2019-09-20 17:08:32 +00:00
|
|
|
|
|
|
|
b0 := ðpb.BeaconBlock{}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b0); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
b0Root, _ := ssz.SigningRoot(b0)
|
|
|
|
b1 := ðpb.BeaconBlock{Slot: 1, ParentRoot: b0Root[:]}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b1); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
b1Root, _ := ssz.SigningRoot(b1)
|
|
|
|
|
|
|
|
// Incomplete block links
|
|
|
|
b2 := ðpb.BeaconBlock{Slot: 2, ParentRoot: b1Root[:]}
|
|
|
|
b2Root, _ := ssz.SigningRoot(b2)
|
|
|
|
b5 := ðpb.BeaconBlock{Slot: 5, ParentRoot: b2Root[:]}
|
|
|
|
b5Root, _ := ssz.SigningRoot(b5)
|
|
|
|
b3 := ðpb.BeaconBlock{Slot: 3, ParentRoot: b0Root[:]}
|
|
|
|
b3Root, _ := ssz.SigningRoot(b3)
|
|
|
|
b4 := ðpb.BeaconBlock{Slot: 4, ParentRoot: b3Root[:]}
|
|
|
|
b4Root, _ := ssz.SigningRoot(b4)
|
|
|
|
|
|
|
|
r.slotToPendingBlocks[b4.Slot] = b4
|
|
|
|
r.seenPendingBlocks[b4Root] = true
|
|
|
|
r.slotToPendingBlocks[b5.Slot] = b5
|
|
|
|
r.seenPendingBlocks[b5Root] = true
|
|
|
|
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 2 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
if len(r.seenPendingBlocks) != 2 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add b3 to the cache
|
|
|
|
r.slotToPendingBlocks[b3.Slot] = b3
|
|
|
|
r.seenPendingBlocks[b3Root] = true
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b3); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 1 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
if len(r.seenPendingBlocks) != 1 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add b2 to the cache
|
|
|
|
r.slotToPendingBlocks[b2.Slot] = b2
|
|
|
|
r.seenPendingBlocks[b2Root] = true
|
|
|
|
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b2); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
t.Log(r.seenPendingBlocks)
|
|
|
|
if len(r.seenPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
}
|
2019-12-03 15:56:04 +00:00
|
|
|
|
|
|
|
func TestRegularSyncBeaconBlockSubscriber_PruneOldPendingBlocks(t *testing.T) {
|
|
|
|
db := dbtest.SetupDB(t)
|
|
|
|
defer dbtest.TeardownDB(t, db)
|
|
|
|
p1 := p2ptest.NewTestP2P(t)
|
|
|
|
p2 := p2ptest.NewTestP2P(t)
|
|
|
|
p1.Connect(p2)
|
|
|
|
if len(p1.Host.Network().Peers()) != 1 {
|
|
|
|
t.Error("Expected peers to be connected")
|
|
|
|
}
|
|
|
|
|
|
|
|
r := &RegularSync{
|
|
|
|
p2p: p1,
|
|
|
|
db: db,
|
|
|
|
chain: &mock.ChainService{
|
|
|
|
FinalizedCheckPoint: ðpb.Checkpoint{
|
|
|
|
Epoch: 1,
|
|
|
|
},
|
|
|
|
}, slotToPendingBlocks: make(map[uint64]*ethpb.BeaconBlock),
|
|
|
|
seenPendingBlocks: make(map[[32]byte]bool),
|
|
|
|
}
|
|
|
|
peerstatus.Set(p2.PeerID(), &pb.Status{})
|
|
|
|
|
|
|
|
b0 := ðpb.BeaconBlock{}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b0); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
b0Root, _ := ssz.SigningRoot(b0)
|
|
|
|
b1 := ðpb.BeaconBlock{Slot: 1, ParentRoot: b0Root[:]}
|
|
|
|
if err := r.db.SaveBlock(context.Background(), b1); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
b1Root, _ := ssz.SigningRoot(b1)
|
|
|
|
|
|
|
|
// Incomplete block links
|
|
|
|
b2 := ðpb.BeaconBlock{Slot: 2, ParentRoot: b1Root[:]}
|
|
|
|
b2Root, _ := ssz.SigningRoot(b2)
|
|
|
|
b5 := ðpb.BeaconBlock{Slot: 5, ParentRoot: b2Root[:]}
|
|
|
|
b5Root, _ := ssz.SigningRoot(b5)
|
|
|
|
b3 := ðpb.BeaconBlock{Slot: 3, ParentRoot: b0Root[:]}
|
|
|
|
b3Root, _ := ssz.SigningRoot(b3)
|
|
|
|
b4 := ðpb.BeaconBlock{Slot: 4, ParentRoot: b3Root[:]}
|
|
|
|
b4Root, _ := ssz.SigningRoot(b4)
|
|
|
|
|
|
|
|
r.slotToPendingBlocks[b2.Slot] = b2
|
|
|
|
r.seenPendingBlocks[b2Root] = true
|
|
|
|
r.slotToPendingBlocks[b3.Slot] = b3
|
|
|
|
r.seenPendingBlocks[b3Root] = true
|
|
|
|
r.slotToPendingBlocks[b4.Slot] = b4
|
|
|
|
r.seenPendingBlocks[b4Root] = true
|
|
|
|
r.slotToPendingBlocks[b5.Slot] = b5
|
|
|
|
r.seenPendingBlocks[b5Root] = true
|
|
|
|
|
|
|
|
if err := r.processPendingBlocks(context.Background()); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if len(r.slotToPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for slot to pending blocks cache: got %d", len(r.slotToPendingBlocks))
|
|
|
|
}
|
|
|
|
if len(r.seenPendingBlocks) != 0 {
|
|
|
|
t.Errorf("Incorrect size for seen pending block: got %d", len(r.seenPendingBlocks))
|
|
|
|
}
|
|
|
|
}
|