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) }