package blockchain import ( "bytes" "context" "reflect" "testing" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" testDB "github.com/prysmaticlabs/prysm/beacon-chain/db/testing" "github.com/prysmaticlabs/prysm/beacon-chain/state/stateutil" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/testutil" logTest "github.com/sirupsen/logrus/hooks/test" ) func TestSaveHead_Same(t *testing.T) { db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) r := [32]byte{'A'} service.head = &head{slot: 0, root: r} if err := service.saveHead(context.Background(), r); err != nil { t.Fatal(err) } if service.headSlot() != 0 { t.Error("Head did not stay the same") } if service.headRoot() != r { t.Error("Head did not stay the same") } } func TestSaveHead_Different(t *testing.T) { ctx := context.Background() db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) oldRoot := [32]byte{'A'} service.head = &head{slot: 0, root: oldRoot} newHeadBlock := ðpb.BeaconBlock{Slot: 1} newHeadSignedBlock := ðpb.SignedBeaconBlock{Block: newHeadBlock} if err := service.beaconDB.SaveBlock(context.Background(), newHeadSignedBlock); err != nil { t.Fatal(err) } newRoot, err := stateutil.BlockRoot(newHeadBlock) if err != nil { t.Fatal(err) } headState := testutil.NewBeaconState() if err := headState.SetSlot(1); err != nil { t.Fatal(err) } if err := service.beaconDB.SaveStateSummary(context.Background(), &pb.StateSummary{Slot: 1, Root: newRoot[:]}); err != nil { t.Fatal(err) } if err := service.beaconDB.SaveState(context.Background(), headState, newRoot); err != nil { t.Fatal(err) } if err := service.saveHead(context.Background(), newRoot); err != nil { t.Fatal(err) } if service.HeadSlot() != 1 { t.Error("Head did not change") } cachedRoot, err := service.HeadRoot(context.Background()) if err != nil { t.Fatal(err) } if !bytes.Equal(cachedRoot, newRoot[:]) { t.Error("Head did not change") } if !reflect.DeepEqual(service.headBlock(), newHeadSignedBlock) { t.Error("Head did not change") } if !reflect.DeepEqual(service.headState(ctx).CloneInnerState(), headState.CloneInnerState()) { t.Error("Head did not change") } } func TestSaveHead_Different_Reorg(t *testing.T) { ctx := context.Background() hook := logTest.NewGlobal() db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) oldRoot := [32]byte{'A'} service.head = &head{slot: 0, root: oldRoot} reorgChainParent := [32]byte{'B'} newHeadBlock := ðpb.BeaconBlock{ Slot: 1, ParentRoot: reorgChainParent[:], } newHeadSignedBlock := ðpb.SignedBeaconBlock{Block: newHeadBlock} if err := service.beaconDB.SaveBlock(context.Background(), newHeadSignedBlock); err != nil { t.Fatal(err) } newRoot, err := stateutil.BlockRoot(newHeadBlock) if err != nil { t.Fatal(err) } headState := testutil.NewBeaconState() if err := headState.SetSlot(1); err != nil { t.Fatal(err) } if err := service.beaconDB.SaveStateSummary(context.Background(), &pb.StateSummary{Slot: 1, Root: newRoot[:]}); err != nil { t.Fatal(err) } if err := service.beaconDB.SaveState(context.Background(), headState, newRoot); err != nil { t.Fatal(err) } if err := service.saveHead(context.Background(), newRoot); err != nil { t.Fatal(err) } if service.HeadSlot() != 1 { t.Error("Head did not change") } cachedRoot, err := service.HeadRoot(context.Background()) if err != nil { t.Fatal(err) } if !bytes.Equal(cachedRoot, newRoot[:]) { t.Error("Head did not change") } if !reflect.DeepEqual(service.headBlock(), newHeadSignedBlock) { t.Error("Head did not change") } if !reflect.DeepEqual(service.headState(ctx).CloneInnerState(), headState.CloneInnerState()) { t.Error("Head did not change") } testutil.AssertLogsContain(t, hook, "Chain reorg occurred") } func TestUpdateRecentCanonicalBlocks_CanUpdateWithoutParent(t *testing.T) { db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) r := [32]byte{'a'} if err := service.updateRecentCanonicalBlocks(context.Background(), r); err != nil { t.Fatal(err) } canonical, err := service.IsCanonical(context.Background(), r) if err != nil { t.Fatal(err) } if !canonical { t.Error("Block should be canonical") } } func TestUpdateRecentCanonicalBlocks_CanUpdateWithParent(t *testing.T) { db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) oldHead := [32]byte{'a'} if err := service.forkChoiceStore.ProcessBlock(context.Background(), 1, oldHead, [32]byte{'g'}, [32]byte{}, 0, 0); err != nil { t.Fatal(err) } currentHead := [32]byte{'b'} if err := service.forkChoiceStore.ProcessBlock(context.Background(), 3, currentHead, oldHead, [32]byte{}, 0, 0); err != nil { t.Fatal(err) } forkedRoot := [32]byte{'c'} if err := service.forkChoiceStore.ProcessBlock(context.Background(), 2, forkedRoot, oldHead, [32]byte{}, 0, 0); err != nil { t.Fatal(err) } if err := service.updateRecentCanonicalBlocks(context.Background(), currentHead); err != nil { t.Fatal(err) } canonical, err := service.IsCanonical(context.Background(), currentHead) if err != nil { t.Fatal(err) } if !canonical { t.Error("Block should be canonical") } canonical, err = service.IsCanonical(context.Background(), oldHead) if err != nil { t.Fatal(err) } if !canonical { t.Error("Block should be canonical") } canonical, err = service.IsCanonical(context.Background(), forkedRoot) if err != nil { t.Fatal(err) } if canonical { t.Error("Block should not be canonical") } } func TestCacheJustifiedStateBalances_CanCache(t *testing.T) { db, sc := testDB.SetupDB(t) service := setupBeaconChain(t, db, sc) state, _ := testutil.DeterministicGenesisState(t, 100) r := [32]byte{'a'} if err := service.beaconDB.SaveStateSummary(context.Background(), &pb.StateSummary{Root: r[:]}); err != nil { t.Fatal(err) } if err := service.beaconDB.SaveState(context.Background(), state, r); err != nil { t.Fatal(err) } if err := service.cacheJustifiedStateBalances(context.Background(), r); err != nil { t.Fatal(err) } if !reflect.DeepEqual(state.Balances(), service.getJustifiedBalances()) { t.Fatal("Incorrect justified balances") } }