2021-04-23 17:06:13 +00:00
|
|
|
package slasherkv
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
|
2022-08-16 12:20:13 +00:00
|
|
|
slashertypes "github.com/prysmaticlabs/prysm/v3/beacon-chain/slasher/types"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/testing/require"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
2021-04-23 17:06:13 +00:00
|
|
|
logTest "github.com/sirupsen/logrus/hooks/test"
|
|
|
|
bolt "go.etcd.io/bbolt"
|
|
|
|
)
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
func TestStore_PruneProposalsAtEpoch(t *testing.T) {
|
2021-04-23 17:06:13 +00:00
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
// If the lowest stored epoch in the database is >= the end epoch of the pruning process,
|
|
|
|
// there is nothing to prune, so we also expect exiting early.
|
|
|
|
t.Run("lowest_stored_epoch_greater_than_pruning_limit_epoch", func(t *testing.T) {
|
|
|
|
hook := logTest.NewGlobal()
|
|
|
|
beaconDB := setupDB(t)
|
|
|
|
|
|
|
|
// With a current epoch of 20 and a history length of 10, we should be pruning
|
|
|
|
// everything before epoch (20 - 10) = 10.
|
|
|
|
currentEpoch := types.Epoch(20)
|
|
|
|
historyLength := types.Epoch(10)
|
|
|
|
|
|
|
|
pruningLimitEpoch := currentEpoch - historyLength
|
2021-10-01 20:17:57 +00:00
|
|
|
lowestStoredSlot, err := slots.EpochEnd(pruningLimitEpoch)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
err = beaconDB.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
bkt := tx.Bucket(proposalRecordsBucket)
|
2021-08-05 21:27:05 +00:00
|
|
|
key, err := keyForValidatorProposal(lowestStoredSlot+1, 0 /* proposer index */)
|
2021-04-23 17:06:13 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return bkt.Put(key, []byte("hi"))
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
_, err = beaconDB.PruneProposalsAtEpoch(ctx, pruningLimitEpoch)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
expectedLog := fmt.Sprintf(
|
2021-08-05 21:27:05 +00:00
|
|
|
"Lowest slot %d is > pruning slot %d, nothing to prune", lowestStoredSlot+1, lowestStoredSlot,
|
2021-04-23 17:06:13 +00:00
|
|
|
)
|
|
|
|
require.LogsContain(t, hook, expectedLog)
|
|
|
|
})
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
t.Run("prune_and_verify_deletions", func(t *testing.T) {
|
2021-04-23 17:06:13 +00:00
|
|
|
beaconDB := setupDB(t)
|
|
|
|
|
2021-12-10 04:18:47 +00:00
|
|
|
params.SetupTestConfigCleanup(t)
|
2021-04-23 17:06:13 +00:00
|
|
|
config := params.BeaconConfig()
|
2021-12-10 04:18:47 +00:00
|
|
|
config.SlotsPerEpoch = 2
|
|
|
|
params.OverrideBeaconConfig(config)
|
2021-04-23 17:06:13 +00:00
|
|
|
|
|
|
|
historyLength := types.Epoch(10)
|
|
|
|
currentEpoch := types.Epoch(20)
|
|
|
|
pruningLimitEpoch := currentEpoch - historyLength
|
|
|
|
|
|
|
|
// We create proposals from genesis to the current epoch, with 2 proposals
|
|
|
|
// at each slot to ensure the entire pruning logic works correctly.
|
|
|
|
slotsPerEpoch := params.BeaconConfig().SlotsPerEpoch
|
|
|
|
proposals := make([]*slashertypes.SignedBlockHeaderWrapper, 0, uint64(currentEpoch)*uint64(slotsPerEpoch)*2)
|
|
|
|
for i := types.Epoch(0); i < currentEpoch; i++ {
|
2021-10-01 20:17:57 +00:00
|
|
|
startSlot, err := slots.EpochStart(i)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
2021-10-01 20:17:57 +00:00
|
|
|
endSlot, err := slots.EpochStart(i + 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
for j := startSlot; j < endSlot; j++ {
|
|
|
|
prop1 := createProposalWrapper(t, j, 0 /* proposer index */, []byte{0})
|
2021-08-05 21:27:05 +00:00
|
|
|
prop2 := createProposalWrapper(t, j, 1 /* proposer index */, []byte{1})
|
2021-04-23 17:06:13 +00:00
|
|
|
proposals = append(proposals, prop1, prop2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
require.NoError(t, beaconDB.SaveBlockProposals(ctx, proposals))
|
|
|
|
|
|
|
|
// We expect pruning completes without an issue and properly logs progress.
|
2021-08-05 21:27:05 +00:00
|
|
|
_, err := beaconDB.PruneProposalsAtEpoch(ctx, pruningLimitEpoch)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Everything before epoch 10 should be deleted.
|
|
|
|
for i := types.Epoch(0); i < pruningLimitEpoch; i++ {
|
|
|
|
err = beaconDB.db.View(func(tx *bolt.Tx) error {
|
|
|
|
bkt := tx.Bucket(proposalRecordsBucket)
|
2021-10-01 20:17:57 +00:00
|
|
|
startSlot, err := slots.EpochStart(i)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
2021-10-01 20:17:57 +00:00
|
|
|
endSlot, err := slots.EpochStart(i + 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
for j := startSlot; j < endSlot; j++ {
|
2021-08-05 21:27:05 +00:00
|
|
|
prop1Key, err := keyForValidatorProposal(j, 0)
|
2021-04-23 17:06:13 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-08-05 21:27:05 +00:00
|
|
|
prop2Key, err := keyForValidatorProposal(j, 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if bkt.Get(prop1Key) != nil {
|
|
|
|
return fmt.Errorf("proposal still exists for epoch %d, validator 0", j)
|
|
|
|
}
|
|
|
|
if bkt.Get(prop2Key) != nil {
|
|
|
|
return fmt.Errorf("proposal still exists for slot %d, validator 1", j)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStore_PruneAttestations_OK(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
// If the lowest stored epoch in the database is >= the end epoch of the pruning process,
|
|
|
|
// there is nothing to prune, so we also expect exiting early.
|
|
|
|
t.Run("lowest_stored_epoch_greater_than_pruning_limit_epoch", func(t *testing.T) {
|
|
|
|
hook := logTest.NewGlobal()
|
|
|
|
beaconDB := setupDB(t)
|
|
|
|
|
|
|
|
// With a current epoch of 20 and a history length of 10, we should be pruning
|
|
|
|
// everything before epoch (20 - 10) = 10.
|
|
|
|
currentEpoch := types.Epoch(20)
|
|
|
|
historyLength := types.Epoch(10)
|
|
|
|
|
|
|
|
pruningLimitEpoch := currentEpoch - historyLength
|
|
|
|
lowestStoredEpoch := pruningLimitEpoch
|
|
|
|
|
|
|
|
err := beaconDB.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
bkt := tx.Bucket(attestationDataRootsBucket)
|
|
|
|
encIdx := encodeValidatorIndex(types.ValidatorIndex(0))
|
2021-08-05 21:27:05 +00:00
|
|
|
encodedTargetEpoch := encodeTargetEpoch(lowestStoredEpoch + 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
key := append(encodedTargetEpoch, encIdx...)
|
|
|
|
return bkt.Put(key, []byte("hi"))
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
_, err = beaconDB.PruneAttestationsAtEpoch(ctx, pruningLimitEpoch)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
expectedLog := fmt.Sprintf(
|
2021-08-05 21:27:05 +00:00
|
|
|
"Lowest epoch %d is > pruning epoch %d, nothing to prune", lowestStoredEpoch+1, lowestStoredEpoch,
|
2021-04-23 17:06:13 +00:00
|
|
|
)
|
|
|
|
require.LogsContain(t, hook, expectedLog)
|
|
|
|
})
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
t.Run("prune_and_verify_deletions", func(t *testing.T) {
|
2021-04-23 17:06:13 +00:00
|
|
|
beaconDB := setupDB(t)
|
|
|
|
|
2021-12-10 04:18:47 +00:00
|
|
|
params.SetupTestConfigCleanup(t)
|
2021-04-23 17:06:13 +00:00
|
|
|
config := params.BeaconConfig()
|
2021-12-10 04:18:47 +00:00
|
|
|
config.SlotsPerEpoch = 2
|
|
|
|
params.OverrideBeaconConfig(config)
|
2021-04-23 17:06:13 +00:00
|
|
|
|
|
|
|
historyLength := types.Epoch(10)
|
|
|
|
currentEpoch := types.Epoch(20)
|
|
|
|
pruningLimitEpoch := currentEpoch - historyLength
|
|
|
|
|
|
|
|
// We create attestations from genesis to the current epoch, with 2 attestations
|
|
|
|
// at each slot to ensure the entire pruning logic works correctly.
|
|
|
|
slotsPerEpoch := params.BeaconConfig().SlotsPerEpoch
|
|
|
|
attestations := make([]*slashertypes.IndexedAttestationWrapper, 0, uint64(currentEpoch)*uint64(slotsPerEpoch)*2)
|
|
|
|
for i := types.Epoch(0); i < currentEpoch; i++ {
|
2021-10-01 20:17:57 +00:00
|
|
|
startSlot, err := slots.EpochStart(i)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
2021-10-01 20:17:57 +00:00
|
|
|
endSlot, err := slots.EpochStart(i + 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
for j := startSlot; j < endSlot; j++ {
|
|
|
|
attester1 := uint64(j + 10)
|
|
|
|
attester2 := uint64(j + 11)
|
|
|
|
target := i
|
|
|
|
var source types.Epoch
|
|
|
|
if i > 0 {
|
|
|
|
source = target - 1
|
|
|
|
}
|
|
|
|
att1 := createAttestationWrapper(source, target, []uint64{attester1}, []byte{0})
|
|
|
|
att2 := createAttestationWrapper(source, target, []uint64{attester2}, []byte{1})
|
|
|
|
attestations = append(attestations, att1, att2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
require.NoError(t, beaconDB.SaveAttestationRecordsForValidators(ctx, attestations))
|
|
|
|
|
2021-08-05 21:27:05 +00:00
|
|
|
// We expect pruning completes without an issue.
|
|
|
|
_, err := beaconDB.PruneAttestationsAtEpoch(ctx, pruningLimitEpoch)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Everything before epoch 10 should be deleted.
|
|
|
|
for i := types.Epoch(0); i < pruningLimitEpoch; i++ {
|
|
|
|
err = beaconDB.db.View(func(tx *bolt.Tx) error {
|
|
|
|
bkt := tx.Bucket(attestationDataRootsBucket)
|
2021-10-01 20:17:57 +00:00
|
|
|
startSlot, err := slots.EpochStart(i)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
2021-10-01 20:17:57 +00:00
|
|
|
endSlot, err := slots.EpochStart(i + 1)
|
2021-04-23 17:06:13 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
for j := startSlot; j < endSlot; j++ {
|
|
|
|
attester1 := types.ValidatorIndex(j + 10)
|
|
|
|
attester2 := types.ValidatorIndex(j + 11)
|
|
|
|
key1 := append(encodeTargetEpoch(i), encodeValidatorIndex(attester1)...)
|
|
|
|
key2 := append(encodeTargetEpoch(i), encodeValidatorIndex(attester2)...)
|
|
|
|
if bkt.Get(key1) != nil {
|
|
|
|
return fmt.Errorf("still exists for epoch %d, validator %d", i, attester1)
|
|
|
|
}
|
|
|
|
if bkt.Get(key2) != nil {
|
|
|
|
return fmt.Errorf("still exists for slot %d, validator %d", i, attester2)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|