erigon-pulse/cl/persistence/block_saver_test.go
2023-08-29 02:37:30 +02:00

176 lines
4.7 KiB
Go

package persistence
import (
"context"
"database/sql"
"testing"
_ "embed"
libcommon "github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon/cl/clparams"
"github.com/ledgerwatch/erigon/cl/cltypes"
"github.com/ledgerwatch/erigon/cl/cltypes/solid"
"github.com/ledgerwatch/erigon/cl/persistence/sql_migrations"
"github.com/ledgerwatch/erigon/cl/phase1/execution_client"
"github.com/ledgerwatch/erigon/cl/utils"
"github.com/ledgerwatch/erigon/core/types"
"github.com/spf13/afero"
"github.com/stretchr/testify/require"
)
type mockEngine struct {
blocks map[uint64]*types.Block
}
func newMockEngine() execution_client.ExecutionEngine {
return &mockEngine{
blocks: make(map[uint64]*types.Block),
}
}
func (m *mockEngine) ForkChoiceUpdate(finalized libcommon.Hash, head libcommon.Hash) error {
panic("unimplemented")
}
func (m *mockEngine) NewPayload(payload *cltypes.Eth1Block, beaconParentRoot *libcommon.Hash) (bool, error) {
panic("unimplemented")
}
func (m *mockEngine) SupportInsertion() bool {
return true
}
func (m *mockEngine) InsertBlocks([]*types.Block) error {
panic("unimplemented")
}
func (m *mockEngine) IsCanonicalHash(libcommon.Hash) (bool, error) {
panic("unimplemented")
}
func (m *mockEngine) Ready() (bool, error) {
return true, nil
}
func (m *mockEngine) InsertBlock(b *types.Block) error {
m.blocks[b.NumberU64()] = b
return nil
}
func (m *mockEngine) GetBodiesByRange(start, count uint64) ([]*types.RawBody, error) {
bds := []*types.RawBody{}
for i := start; i < start+count; i++ {
blk, ok := m.blocks[i]
if !ok {
break
}
bds = append(bds, blk.RawBody())
}
return bds, nil
}
func (m *mockEngine) GetBodiesByHashes(hashes []libcommon.Hash) ([]*types.RawBody, error) {
panic("unimplemented")
}
//go:embed test_data/test_block.ssz_snappy
var testBlock []byte
func getTestBlock() *cltypes.SignedBeaconBlock {
enc, err := utils.DecompressSnappy(testBlock)
if err != nil {
panic(err)
}
bcBlock := cltypes.NewSignedBeaconBlock(&clparams.MainnetBeaconConfig)
if err := bcBlock.DecodeSSZ(enc, int(clparams.CapellaVersion)); err != nil {
panic(err)
}
bcBlock.Block.Slot = (clparams.MainnetBeaconConfig.CapellaForkEpoch + 1) * 32
bcBlock.Block.Body.ExecutionPayload.Transactions = solid.NewTransactionsSSZFromTransactions(nil)
bcBlock.Block.Body.ExecutionPayload.BlockNumber = 100
bcBlock.Block.Body.ExecutionPayload.BlockHash = libcommon.HexToHash("0x78e6ce0d5a80c7416138af475d20c0a0a22124ae67b6dc5a0d0d0fe6f95e365d")
return bcBlock
}
func setupStore(t *testing.T, full bool) (BeaconChainDatabase, *sql.DB, execution_client.ExecutionEngine) {
// Open an in-memory SQLite database for testing
db, err := sql.Open("sqlite3", ":memory:")
if err != nil {
t.Fatalf("Failed to open database: %v", err)
}
// Start a transaction for testing
tx, err := db.Begin()
if err != nil {
t.Fatalf("Failed to start transaction: %v", err)
}
defer tx.Rollback()
// Call ApplyMigrations with the test transaction
err = sql_migrations.ApplyMigrations(context.Background(), tx)
if err != nil {
t.Fatalf("ApplyMigrations failed: %v", err)
}
tx.Commit()
// Create an in-memory filesystem
fs := afero.NewMemMapFs()
engine := newMockEngine()
return NewBeaconChainDatabaseFilesystem(fs, engine, full, &clparams.MainnetBeaconConfig, db), db, engine
}
func TestBlockSaverStoreLoadPurgeFull(t *testing.T) {
store, db, _ := setupStore(t, true)
defer db.Close()
ctx := context.Background()
block := getTestBlock()
require.NoError(t, store.WriteBlock(ctx, block, true))
blks, err := store.GetRange(context.Background(), block.Block.Slot, 1)
require.NoError(t, err)
require.Equal(t, len(blks), 1)
expectedRoot, err := block.HashSSZ()
require.NoError(t, err)
haveRoot, err := blks[0].Data.HashSSZ()
require.NoError(t, err)
require.Equal(t, expectedRoot, haveRoot)
require.NoError(t, store.PurgeRange(ctx, 0, 99999999999)) // THE PUURGE
newBlks, err := store.GetRange(context.Background(), block.Block.Slot, 1)
require.NoError(t, err)
require.Equal(t, len(newBlks), 0)
}
func TestBlockSaverStoreAndLoadPartial(t *testing.T) {
store, db, engine := setupStore(t, false)
defer db.Close()
ctx := context.Background()
block := getTestBlock()
require.NoError(t, store.WriteBlock(ctx, block, true))
eth1Block := block.Block.Body.ExecutionPayload
header, err := eth1Block.RlpHeader()
require.NoError(t, err)
engine.InsertBlock(types.NewBlock(header, nil, nil, nil, eth1Block.Body().Withdrawals))
blks, err := store.GetRange(context.Background(), block.Block.Slot, 1)
require.NoError(t, err)
require.Equal(t, len(blks), 1)
expectedRoot, err := block.HashSSZ()
require.NoError(t, err)
haveRoot, err := blks[0].Data.HashSSZ()
require.NoError(t, err)
require.Equal(t, expectedRoot, haveRoot)
}