2019-08-12 19:33:07 +00:00
|
|
|
package kv
|
|
|
|
|
|
|
|
import (
|
2020-12-15 22:07:01 +00:00
|
|
|
"context"
|
2023-03-01 00:07:23 +00:00
|
|
|
"fmt"
|
2019-08-12 19:33:07 +00:00
|
|
|
"testing"
|
|
|
|
|
2023-03-17 18:52:56 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v4/config/features"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/testing/require"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/testing/util"
|
2022-07-13 17:18:30 +00:00
|
|
|
bolt "go.etcd.io/bbolt"
|
2019-08-12 19:33:07 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// setupDB instantiates and returns a Store instance.
|
|
|
|
func setupDB(t testing.TB) *Store {
|
2022-08-17 12:22:41 +00:00
|
|
|
db, err := NewKVStore(context.Background(), t.TempDir())
|
2020-08-08 18:39:01 +00:00
|
|
|
require.NoError(t, err, "Failed to instantiate DB")
|
2020-05-04 01:14:34 +00:00
|
|
|
t.Cleanup(func() {
|
2020-08-08 18:39:01 +00:00
|
|
|
require.NoError(t, db.Close(), "Failed to close database")
|
2020-05-04 01:14:34 +00:00
|
|
|
})
|
2019-08-12 19:33:07 +00:00
|
|
|
return db
|
|
|
|
}
|
2022-07-13 17:18:30 +00:00
|
|
|
|
2023-03-01 00:07:23 +00:00
|
|
|
func Test_setupBlockStorageType(t *testing.T) {
|
|
|
|
ctx := context.Background()
|
|
|
|
t.Run("fresh database with feature enabled to store full blocks should store full blocks", func(t *testing.T) {
|
|
|
|
resetFn := features.InitWithReset(&features.Flags{
|
|
|
|
SaveFullExecutionPayloads: true,
|
|
|
|
})
|
|
|
|
defer resetFn()
|
|
|
|
store := setupDB(t)
|
|
|
|
|
|
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayload.BlockNumber = 1
|
|
|
|
wrappedBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err := wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
require.NoError(t, store.SaveStateSummary(ctx, ðpb.StateSummary{Root: root[:]}))
|
|
|
|
require.NoError(t, store.SaveHeadBlockRoot(ctx, root))
|
|
|
|
retrievedBlk, err := store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, false, retrievedBlk.IsBlinded())
|
|
|
|
require.DeepEqual(t, wrappedBlock, retrievedBlk)
|
|
|
|
})
|
|
|
|
t.Run("fresh database with default settings should store blinded", func(t *testing.T) {
|
|
|
|
resetFn := features.InitWithReset(&features.Flags{
|
|
|
|
SaveFullExecutionPayloads: false,
|
|
|
|
})
|
|
|
|
defer resetFn()
|
|
|
|
store := setupDB(t)
|
|
|
|
|
|
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayload.BlockNumber = 1
|
|
|
|
wrappedBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err := wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
require.NoError(t, store.SaveStateSummary(ctx, ðpb.StateSummary{Root: root[:]}))
|
|
|
|
require.NoError(t, store.SaveHeadBlockRoot(ctx, root))
|
|
|
|
retrievedBlk, err := store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, true, retrievedBlk.IsBlinded())
|
|
|
|
|
|
|
|
wantedBlk, err := wrappedBlock.ToBlinded()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, wantedBlk, retrievedBlk)
|
|
|
|
})
|
|
|
|
t.Run("existing database with blinded blocks but no key in metadata bucket should continue storing blinded blocks", func(t *testing.T) {
|
|
|
|
store := setupDB(t)
|
|
|
|
require.NoError(t, store.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return tx.Bucket(chainMetadataBucket).Put(saveBlindedBeaconBlocksKey, []byte{1})
|
|
|
|
}))
|
|
|
|
|
|
|
|
blk := util.NewBlindedBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayloadHeader.BlockNumber = 1
|
|
|
|
wrappedBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err := wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
require.NoError(t, store.SaveStateSummary(ctx, ðpb.StateSummary{Root: root[:]}))
|
|
|
|
require.NoError(t, store.SaveHeadBlockRoot(ctx, root))
|
|
|
|
retrievedBlk, err := store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, true, retrievedBlk.IsBlinded())
|
|
|
|
require.DeepEqual(t, wrappedBlock, retrievedBlk)
|
|
|
|
|
|
|
|
// We then delete the key from the bucket.
|
|
|
|
require.NoError(t, store.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return tx.Bucket(chainMetadataBucket).Delete(saveBlindedBeaconBlocksKey)
|
|
|
|
}))
|
|
|
|
|
|
|
|
// Not a fresh database, has blinded blocks already and should continue being that way.
|
|
|
|
err = store.setupBlockStorageType(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var shouldSaveBlinded bool
|
|
|
|
require.NoError(t, store.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
bkt := tx.Bucket(chainMetadataBucket)
|
|
|
|
shouldSaveBlinded = len(bkt.Get(saveBlindedBeaconBlocksKey)) > 0
|
|
|
|
return nil
|
|
|
|
}))
|
|
|
|
|
|
|
|
// Should have set the chain metadata bucket to save blinded
|
|
|
|
require.Equal(t, true, shouldSaveBlinded)
|
|
|
|
|
|
|
|
blkFull := util.NewBeaconBlockBellatrix()
|
|
|
|
blkFull.Block.Body.ExecutionPayload.BlockNumber = 2
|
|
|
|
wrappedBlock, err = blocks.NewSignedBeaconBlock(blkFull)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err = wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
retrievedBlk, err = store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t, true, retrievedBlk.IsBlinded())
|
|
|
|
wrappedBlinded, err := wrappedBlock.ToBlinded()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, wrappedBlinded, retrievedBlk)
|
|
|
|
})
|
|
|
|
t.Run("existing database with full blocks type should continue storing full blocks", func(t *testing.T) {
|
|
|
|
store := setupDB(t)
|
|
|
|
require.NoError(t, store.db.Update(func(tx *bolt.Tx) error {
|
|
|
|
return tx.Bucket(chainMetadataBucket).Delete(saveBlindedBeaconBlocksKey)
|
|
|
|
}))
|
|
|
|
|
|
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayload.BlockNumber = 1
|
|
|
|
wrappedBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err := wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
require.NoError(t, store.SaveStateSummary(ctx, ðpb.StateSummary{Root: root[:]}))
|
|
|
|
require.NoError(t, store.SaveHeadBlockRoot(ctx, root))
|
|
|
|
retrievedBlk, err := store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, false, retrievedBlk.IsBlinded())
|
|
|
|
require.DeepEqual(t, wrappedBlock, retrievedBlk)
|
|
|
|
|
|
|
|
// Not a fresh database, has full blocks already and should continue being that way.
|
|
|
|
err = store.setupBlockStorageType(ctx)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
blk = util.NewBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayload.BlockNumber = 2
|
|
|
|
wrappedBlock, err = blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err = wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
retrievedBlk, err = store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, false, retrievedBlk.IsBlinded())
|
|
|
|
require.DeepEqual(t, wrappedBlock, retrievedBlk)
|
|
|
|
})
|
|
|
|
t.Run("existing database with blinded blocks type should error if user enables full blocks feature flag", func(t *testing.T) {
|
|
|
|
store := setupDB(t)
|
|
|
|
|
|
|
|
blk := util.NewBeaconBlockBellatrix()
|
|
|
|
blk.Block.Body.ExecutionPayload.BlockNumber = 1
|
|
|
|
wrappedBlock, err := blocks.NewSignedBeaconBlock(blk)
|
|
|
|
require.NoError(t, err)
|
|
|
|
root, err := wrappedBlock.Block().HashTreeRoot()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.NoError(t, store.SaveBlock(ctx, wrappedBlock))
|
|
|
|
require.NoError(t, store.SaveStateSummary(ctx, ðpb.StateSummary{Root: root[:]}))
|
|
|
|
require.NoError(t, store.SaveHeadBlockRoot(ctx, root))
|
|
|
|
retrievedBlk, err := store.Block(ctx, root)
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, true, retrievedBlk.IsBlinded())
|
|
|
|
wantedBlk, err := wrappedBlock.ToBlinded()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.DeepEqual(t, wantedBlk, retrievedBlk)
|
|
|
|
|
|
|
|
// Trying to enable full blocks with a database that is already storing blinded blocks should error.
|
|
|
|
resetFn := features.InitWithReset(&features.Flags{
|
|
|
|
SaveFullExecutionPayloads: true,
|
|
|
|
})
|
|
|
|
defer resetFn()
|
|
|
|
err = store.setupBlockStorageType(ctx)
|
|
|
|
errMsg := "cannot use the %s flag with this existing database, as it has already been initialized"
|
|
|
|
require.ErrorContains(t, fmt.Sprintf(errMsg, features.SaveFullExecutionPayloads.Name), err)
|
2022-07-13 17:18:30 +00:00
|
|
|
})
|
|
|
|
}
|