package depositcache import ( "bytes" "context" "math/big" "testing" ethpb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/prysm/shared/bytesutil" logTest "github.com/sirupsen/logrus/hooks/test" ) const nilDepositErr = "Ignoring nil deposit insertion" var _ = DepositFetcher(&DepositCache{}) func TestBeaconDB_InsertDeposit_LogsOnNilDepositInsertion(t *testing.T) { hook := logTest.NewGlobal() dc := DepositCache{} dc.InsertDeposit(context.Background(), nil, big.NewInt(1), 0, [32]byte{}) if len(dc.deposits) != 0 { t.Fatal("Number of deposits changed") } if hook.LastEntry().Message != nilDepositErr { t.Errorf("Did not log correct message, wanted \"Ignoring nil deposit insertion\", got \"%s\"", hook.LastEntry().Message) } } func TestBeaconDB_InsertDeposit_LogsOnNilBlockNumberInsertion(t *testing.T) { hook := logTest.NewGlobal() dc := DepositCache{} dc.InsertDeposit(context.Background(), ðpb.Deposit{}, nil, 0, [32]byte{}) if len(dc.deposits) != 0 { t.Fatal("Number of deposits changed") } if hook.LastEntry().Message != nilDepositErr { t.Errorf("Did not log correct message, wanted \"Ignoring nil deposit insertion\", got \"%s\"", hook.LastEntry().Message) } } func TestBeaconDB_InsertDeposit_MaintainsSortedOrderByIndex(t *testing.T) { dc := DepositCache{} insertions := []struct { blkNum *big.Int deposit *ethpb.Deposit index int }{ { blkNum: big.NewInt(0), deposit: ðpb.Deposit{}, index: 0, }, { blkNum: big.NewInt(0), deposit: ðpb.Deposit{}, index: 3, }, { blkNum: big.NewInt(0), deposit: ðpb.Deposit{}, index: 1, }, { blkNum: big.NewInt(0), deposit: ðpb.Deposit{}, index: 4, }, } for _, ins := range insertions { dc.InsertDeposit(context.Background(), ins.deposit, ins.blkNum, ins.index, [32]byte{}) } expectedIndices := []int{0, 1, 3, 4} for i, ei := range expectedIndices { if dc.deposits[i].Index != ei { t.Errorf("dc.deposits[%d].Index = %d, wanted %d", i, dc.deposits[i].Index, ei) } } } func TestBeaconDB_AllDeposits_ReturnsAllDeposits(t *testing.T) { dc := DepositCache{} deposits := []*DepositContainer{ { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, } dc.deposits = deposits d := dc.AllDeposits(context.Background(), nil) if len(d) != len(deposits) { t.Errorf("Return the wrong number of deposits (%d) wanted %d", len(d), len(deposits)) } } func TestBeaconDB_AllDeposits_FiltersDepositUpToAndIncludingBlockNumber(t *testing.T) { dc := DepositCache{} deposits := []*DepositContainer{ { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, } dc.deposits = deposits d := dc.AllDeposits(context.Background(), big.NewInt(11)) expected := 5 if len(d) != expected { t.Errorf("Return the wrong number of deposits (%d) wanted %d", len(d), expected) } } func TestBeaconDB_DepositsNumberAndRootAtHeight_ReturnsAppropriateCountAndRoot(t *testing.T) { dc := DepositCache{} dc.deposits = []*DepositContainer{ { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, depositRoot: bytesutil.ToBytes32([]byte("root")), }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{}, }, } n, root := dc.DepositsNumberAndRootAtHeight(context.Background(), big.NewInt(11)) if int(n) != 5 { t.Errorf("Returned unexpected deposits number %d wanted %d", n, 5) } if root != bytesutil.ToBytes32([]byte("root")) { t.Errorf("Returned unexpected root: %v", root) } } func TestBeaconDB_DepositsNumberAndRootAtHeight_ReturnsEmptyTrieIfBlockHeightLessThanOldestDeposit(t *testing.T) { dc := DepositCache{} dc.deposits = []*DepositContainer{ { Block: big.NewInt(10), Deposit: ðpb.Deposit{}, depositRoot: bytesutil.ToBytes32([]byte("root")), }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{}, depositRoot: bytesutil.ToBytes32([]byte("root")), }, } n, root := dc.DepositsNumberAndRootAtHeight(context.Background(), big.NewInt(2)) if int(n) != 0 { t.Errorf("Returned unexpected deposits number %d wanted %d", n, 0) } if root != [32]byte{} { t.Errorf("Returned unexpected root: %v", root) } } func TestBeaconDB_DepositByPubkey_ReturnsFirstMatchingDeposit(t *testing.T) { dc := DepositCache{} dc.deposits = []*DepositContainer{ { Block: big.NewInt(9), Deposit: ðpb.Deposit{ Data: ðpb.Deposit_Data{ PublicKey: []byte("pk0"), }, }, }, { Block: big.NewInt(10), Deposit: ðpb.Deposit{ Data: ðpb.Deposit_Data{ PublicKey: []byte("pk1"), }, }, }, { Block: big.NewInt(11), Deposit: ðpb.Deposit{ Data: ðpb.Deposit_Data{ PublicKey: []byte("pk1"), }, }, }, { Block: big.NewInt(12), Deposit: ðpb.Deposit{ Data: ðpb.Deposit_Data{ PublicKey: []byte("pk2"), }, }, }, } dep, blkNum := dc.DepositByPubkey(context.Background(), []byte("pk1")) if !bytes.Equal(dep.Data.PublicKey, []byte("pk1")) { t.Error("Returned wrong deposit") } if blkNum.Cmp(big.NewInt(10)) != 0 { t.Errorf("Returned wrong block number %v", blkNum) } }