mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-16 06:58:20 +00:00
b5c4dc2a75
* init-sync updates * slasher/db/kv tests * beacon-chain/rpc/beacon tests * update kv_test * beacon-chain/rpc-validator tests updated * slasher/db/kv - remove teardown method * beacon-chain/sync tests updated * beacon-chain/db/kv tests updated * beacon-chain/blockchain tests updated * beacon-chain/state/stategen tests updated * beacon-chain/powchain updates * updates rest of slasher tests * validator/db tests * rest of the tests * minor comments update * gazelle * Merge refs/heads/master into teardowndb-to-cleanup
271 lines
8.7 KiB
Go
271 lines
8.7 KiB
Go
package db
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"reflect"
|
|
"strings"
|
|
"testing"
|
|
|
|
"github.com/prysmaticlabs/go-bitfield"
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
)
|
|
|
|
func TestProposalHistoryForEpoch_InitializesNewPubKeys(t *testing.T) {
|
|
pubkeys := [][48]byte{{30}, {25}, {20}}
|
|
db := SetupDB(t, pubkeys)
|
|
|
|
for _, pub := range pubkeys {
|
|
slotBits, err := db.ProposalHistoryForEpoch(context.Background(), pub[:], 0)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
cleanBits := bitfield.NewBitlist(params.BeaconConfig().SlotsPerEpoch)
|
|
if !bytes.Equal(slotBits.Bytes(), cleanBits.Bytes()) {
|
|
t.Fatalf("Expected proposal history slot bits to be empty, received %v", slotBits.Bytes())
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestProposalHistoryForEpoch_NilDB(t *testing.T) {
|
|
valPubkey := [48]byte{1, 2, 3}
|
|
db := SetupDB(t, [][48]byte{})
|
|
|
|
_, err := db.ProposalHistoryForEpoch(context.Background(), valPubkey[:], 0)
|
|
if err == nil {
|
|
t.Fatal("unexpected non-error")
|
|
}
|
|
|
|
if !strings.Contains(err.Error(), "validator history empty for public key") {
|
|
t.Fatalf("Unexpected error for nil DB, received: %v", err)
|
|
}
|
|
}
|
|
|
|
func TestSaveProposalHistoryForEpoch_OK(t *testing.T) {
|
|
pubkey := [48]byte{3}
|
|
db := SetupDB(t, [][48]byte{pubkey})
|
|
|
|
epoch := uint64(2)
|
|
slot := uint64(2)
|
|
slotBits := bitfield.Bitlist{0x04, 0x00, 0x00, 0x00, 0x04}
|
|
|
|
if err := db.SaveProposalHistoryForEpoch(context.Background(), pubkey[:], epoch, slotBits); err != nil {
|
|
t.Fatalf("Saving proposal history failed: %v", err)
|
|
}
|
|
savedBits, err := db.ProposalHistoryForEpoch(context.Background(), pubkey[:], epoch)
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
|
|
if savedBits == nil || !bytes.Equal(slotBits, savedBits) {
|
|
t.Fatalf("Expected DB to keep object the same, received: %v", savedBits)
|
|
}
|
|
if !savedBits.BitAt(slot) {
|
|
t.Fatalf("Expected slot %d to be marked as proposed", slot)
|
|
}
|
|
if savedBits.BitAt(slot + 1) {
|
|
t.Fatalf("Expected slot %d to not be marked as proposed", slot+1)
|
|
}
|
|
if savedBits.BitAt(slot - 1) {
|
|
t.Fatalf("Expected slot %d to not be marked as proposed", slot-1)
|
|
}
|
|
}
|
|
|
|
func TestSaveProposalHistoryForEpoch_Overwrites(t *testing.T) {
|
|
pubkey := [48]byte{0}
|
|
tests := []struct {
|
|
slot uint64
|
|
slotBits bitfield.Bitlist
|
|
}{
|
|
{
|
|
slot: uint64(1),
|
|
slotBits: bitfield.Bitlist{0x02, 0x00, 0x00, 0x00, 0x02},
|
|
},
|
|
{
|
|
slot: uint64(2),
|
|
slotBits: bitfield.Bitlist{0x04, 0x00, 0x00, 0x00, 0x04},
|
|
},
|
|
{
|
|
slot: uint64(3),
|
|
slotBits: bitfield.Bitlist{0x08, 0x00, 0x00, 0x00, 0x08},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
db := SetupDB(t, [][48]byte{pubkey})
|
|
if err := db.SaveProposalHistoryForEpoch(context.Background(), pubkey[:], 0, tt.slotBits); err != nil {
|
|
t.Fatalf("Saving proposal history failed: %v", err)
|
|
}
|
|
savedBits, err := db.ProposalHistoryForEpoch(context.Background(), pubkey[:], 0)
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
|
|
if savedBits == nil || !reflect.DeepEqual(savedBits, tt.slotBits) {
|
|
t.Fatalf("Expected DB to keep object the same, received: %v, expected %v", savedBits, tt.slotBits)
|
|
}
|
|
if !savedBits.BitAt(tt.slot) {
|
|
t.Fatalf("Expected slot %d to be marked as proposed", tt.slot)
|
|
}
|
|
if savedBits.BitAt(tt.slot + 1) {
|
|
t.Fatalf("Expected slot %d to not be marked as proposed", tt.slot+1)
|
|
}
|
|
if savedBits.BitAt(tt.slot - 1) {
|
|
t.Fatalf("Expected slot %d to not be marked as proposed", tt.slot-1)
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestProposalHistoryForEpoch_MultipleEpochs(t *testing.T) {
|
|
pubKey := [48]byte{0}
|
|
tests := []struct {
|
|
slots []uint64
|
|
expectedBits []bitfield.Bitlist
|
|
}{
|
|
{
|
|
slots: []uint64{1, 2, 8, 31},
|
|
expectedBits: []bitfield.Bitlist{{0b00000110, 0b00000001, 0b00000000, 0b10000000, 0b00000001}},
|
|
},
|
|
{
|
|
slots: []uint64{1, 33, 8},
|
|
expectedBits: []bitfield.Bitlist{
|
|
{0b00000010, 0b00000001, 0b00000000, 0b00000000, 0b00000001},
|
|
{0b00000010, 0b00000000, 0b00000000, 0b00000000, 0b00000001},
|
|
},
|
|
},
|
|
{
|
|
slots: []uint64{2, 34, 36},
|
|
expectedBits: []bitfield.Bitlist{
|
|
{0b00000100, 0b00000000, 0b00000000, 0b00000000, 0b00000001},
|
|
{0b00010100, 0b00000000, 0b00000000, 0b00000000, 0b00000001},
|
|
},
|
|
},
|
|
{
|
|
slots: []uint64{32, 33, 34},
|
|
expectedBits: []bitfield.Bitlist{
|
|
{0, 0, 0, 0, 1},
|
|
{0b00000111, 0b00000000, 0b00000000, 0b00000000, 0b00000001},
|
|
},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
db := SetupDB(t, [][48]byte{pubKey})
|
|
for _, slot := range tt.slots {
|
|
slotBits, err := db.ProposalHistoryForEpoch(context.Background(), pubKey[:], helpers.SlotToEpoch(slot))
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
slotBits.SetBitAt(slot%params.BeaconConfig().SlotsPerEpoch, true)
|
|
if err := db.SaveProposalHistoryForEpoch(context.Background(), pubKey[:], helpers.SlotToEpoch(slot), slotBits); err != nil {
|
|
t.Fatalf("Saving proposal history failed: %v", err)
|
|
}
|
|
}
|
|
|
|
for i, slotBits := range tt.expectedBits {
|
|
savedBits, err := db.ProposalHistoryForEpoch(context.Background(), pubKey[:], uint64(i))
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
if !bytes.Equal(slotBits, savedBits) {
|
|
t.Fatalf("unexpected difference in bytes for slots %v, expected %v vs received %v", tt.slots, slotBits, savedBits)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestPruneProposalHistory_OK(t *testing.T) {
|
|
slotsPerEpoch := params.BeaconConfig().SlotsPerEpoch
|
|
wsPeriod := params.BeaconConfig().WeakSubjectivityPeriod
|
|
pubKey := [48]byte{0}
|
|
tests := []struct {
|
|
slots []uint64
|
|
storedEpochs []uint64
|
|
removedEpochs []uint64
|
|
}{
|
|
{
|
|
// Go 2 epochs past pruning point.
|
|
slots: []uint64{slotsPerEpoch / 2, slotsPerEpoch*5 + 6, (wsPeriod+3)*slotsPerEpoch + 8},
|
|
storedEpochs: []uint64{5, 54003},
|
|
removedEpochs: []uint64{0},
|
|
},
|
|
{
|
|
// Go 10 epochs past pruning point.
|
|
slots: []uint64{slotsPerEpoch + 4, slotsPerEpoch * 2, slotsPerEpoch * 3, slotsPerEpoch * 4, slotsPerEpoch * 5, (wsPeriod+10)*slotsPerEpoch + 8},
|
|
storedEpochs: []uint64{54010},
|
|
removedEpochs: []uint64{1, 2, 3, 4},
|
|
},
|
|
{
|
|
// Prune none.
|
|
slots: []uint64{slotsPerEpoch + 4, slotsPerEpoch*2 + 3, slotsPerEpoch*3 + 4, slotsPerEpoch*4 + 3, slotsPerEpoch*5 + 3},
|
|
storedEpochs: []uint64{1, 2, 3, 4, 5},
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
db := SetupDB(t, [][48]byte{pubKey})
|
|
for _, slot := range tt.slots {
|
|
slotBits, err := db.ProposalHistoryForEpoch(context.Background(), pubKey[:], helpers.SlotToEpoch(slot))
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
slotBits.SetBitAt(slot%params.BeaconConfig().SlotsPerEpoch, true)
|
|
if err := db.SaveProposalHistoryForEpoch(context.Background(), pubKey[:], helpers.SlotToEpoch(slot), slotBits); err != nil {
|
|
t.Fatalf("Saving proposal history failed: %v", err)
|
|
}
|
|
}
|
|
|
|
for _, epoch := range tt.removedEpochs {
|
|
savedBits, err := db.ProposalHistoryForEpoch(context.Background(), pubKey[:], epoch)
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
if !bytes.Equal(bitfield.NewBitlist(slotsPerEpoch), savedBits) {
|
|
t.Fatalf("unexpected difference in bytes for epoch %d, expected %#x vs received %v", epoch, bitfield.NewBitlist(slotsPerEpoch), savedBits)
|
|
}
|
|
}
|
|
for _, epoch := range tt.storedEpochs {
|
|
savedBits, err := db.ProposalHistoryForEpoch(context.Background(), pubKey[:], epoch)
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
if bytes.Equal(bitfield.NewBitlist(slotsPerEpoch), savedBits) {
|
|
t.Fatalf("unexpected difference in bytes for epoch %d, expected %v vs received %v", epoch, bitfield.NewBitlist(slotsPerEpoch), savedBits)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestDeleteProposalHistory_OK(t *testing.T) {
|
|
pubkey := [48]byte{2}
|
|
db := SetupDB(t, [][48]byte{pubkey})
|
|
|
|
slotBits := bitfield.Bitlist{0x01, 0x00, 0x00, 0x00, 0x02}
|
|
|
|
if err := db.SaveProposalHistoryForEpoch(context.Background(), pubkey[:], 0, slotBits); err != nil {
|
|
t.Fatalf("Save proposal history failed: %v", err)
|
|
}
|
|
// Making sure everything is saved.
|
|
savedHistory, err := db.ProposalHistoryForEpoch(context.Background(), pubkey[:], 0)
|
|
if err != nil {
|
|
t.Fatalf("Failed to get proposal history: %v", err)
|
|
}
|
|
if savedHistory == nil || !bytes.Equal(savedHistory, slotBits) {
|
|
t.Fatalf("Expected DB to keep object the same, received: %v, expected %v", savedHistory, slotBits)
|
|
}
|
|
if err := db.DeleteProposalHistory(context.Background(), pubkey[:]); err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
// Check after deleting from DB.
|
|
_, err = db.ProposalHistoryForEpoch(context.Background(), pubkey[:], 0)
|
|
if err == nil {
|
|
t.Fatalf("Unexpected success in deleting history: %v", err)
|
|
}
|
|
if !strings.Contains(err.Error(), "validator history empty for public key ") {
|
|
t.Fatalf("Unexpected error, received %v", err)
|
|
}
|
|
}
|