2023-10-24 19:32:29 +00:00
|
|
|
package freezeblocks
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2023-11-15 14:07:16 +00:00
|
|
|
"context"
|
2023-11-13 13:10:57 +00:00
|
|
|
"fmt"
|
2023-10-30 12:48:14 +00:00
|
|
|
"sync"
|
2023-10-24 19:32:29 +00:00
|
|
|
|
2023-11-13 13:10:57 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
2023-11-15 14:07:16 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2023-10-24 19:32:29 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
2023-11-15 14:07:16 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/persistence"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/persistence/beacon_indicies"
|
2023-10-24 19:32:29 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/persistence/format/snapshot_format"
|
2023-10-30 12:48:14 +00:00
|
|
|
"github.com/pierrec/lz4"
|
2023-10-24 19:32:29 +00:00
|
|
|
)
|
|
|
|
|
2023-10-30 12:48:14 +00:00
|
|
|
var buffersPool = sync.Pool{
|
|
|
|
New: func() interface{} { return &bytes.Buffer{} },
|
|
|
|
}
|
|
|
|
|
|
|
|
var lz4ReaderPool = sync.Pool{
|
|
|
|
New: func() interface{} {
|
|
|
|
return lz4.NewReader(nil)
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2023-10-24 19:32:29 +00:00
|
|
|
type BeaconSnapshotReader interface {
|
|
|
|
// ReadBlock reads the block at the given slot.
|
|
|
|
// If the block is not present, it returns nil.
|
2023-11-15 14:07:16 +00:00
|
|
|
ReadBlockBySlot(ctx context.Context, tx kv.Tx, slot uint64) (*cltypes.SignedBeaconBlock, error)
|
|
|
|
ReadBlockByRoot(ctx context.Context, tx kv.Tx, blockRoot libcommon.Hash) (*cltypes.SignedBeaconBlock, error)
|
2023-11-22 00:45:15 +00:00
|
|
|
ReadHeaderByRoot(ctx context.Context, tx kv.Tx, blockRoot libcommon.Hash) (*cltypes.SignedBeaconBlockHeader, error)
|
2023-10-24 19:32:29 +00:00
|
|
|
|
|
|
|
FrozenSlots() uint64
|
|
|
|
}
|
|
|
|
|
|
|
|
type beaconSnapshotReader struct {
|
|
|
|
sn *CaplinSnapshots
|
|
|
|
|
|
|
|
eth1Getter snapshot_format.ExecutionBlockReaderByNumber
|
2023-11-15 14:07:16 +00:00
|
|
|
beaconDB persistence.BlockSource
|
2023-10-24 19:32:29 +00:00
|
|
|
cfg *clparams.BeaconChainConfig
|
|
|
|
}
|
|
|
|
|
2023-11-15 14:07:16 +00:00
|
|
|
func NewBeaconSnapshotReader(snapshots *CaplinSnapshots, eth1Getter snapshot_format.ExecutionBlockReaderByNumber, beaconDB persistence.BlockSource, cfg *clparams.BeaconChainConfig) BeaconSnapshotReader {
|
|
|
|
return &beaconSnapshotReader{sn: snapshots, eth1Getter: eth1Getter, cfg: cfg, beaconDB: beaconDB}
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *beaconSnapshotReader) FrozenSlots() uint64 {
|
|
|
|
return r.sn.BlocksAvailable()
|
|
|
|
}
|
|
|
|
|
2023-11-15 14:07:16 +00:00
|
|
|
func (r *beaconSnapshotReader) ReadBlockBySlot(ctx context.Context, tx kv.Tx, slot uint64) (*cltypes.SignedBeaconBlock, error) {
|
2023-11-13 13:10:57 +00:00
|
|
|
view := r.sn.View()
|
|
|
|
defer view.Close()
|
|
|
|
|
|
|
|
var buf []byte
|
2023-11-15 14:07:16 +00:00
|
|
|
if slot > r.sn.BlocksAvailable() {
|
|
|
|
data, err := r.beaconDB.GetBlock(ctx, tx, slot)
|
2023-12-06 09:48:36 +00:00
|
|
|
if data == nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-11-15 14:07:16 +00:00
|
|
|
return data.Data, err
|
|
|
|
}
|
|
|
|
if r.eth1Getter == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2023-11-13 13:10:57 +00:00
|
|
|
|
|
|
|
seg, ok := view.BeaconBlocksSegment(slot)
|
|
|
|
if !ok {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if seg.idxSlot == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
if slot < seg.idxSlot.BaseDataID() {
|
|
|
|
return nil, fmt.Errorf("slot %d is before the base data id %d", slot, seg.idxSlot.BaseDataID())
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
2023-11-13 13:10:57 +00:00
|
|
|
blockOffset := seg.idxSlot.OrdinalLookup(slot - seg.idxSlot.BaseDataID())
|
|
|
|
|
|
|
|
gg := seg.seg.MakeGetter()
|
|
|
|
gg.Reset(blockOffset)
|
|
|
|
if !gg.HasNext() {
|
2023-10-24 19:32:29 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
2023-10-30 12:48:14 +00:00
|
|
|
|
2023-11-13 13:10:57 +00:00
|
|
|
buf = buf[:0]
|
|
|
|
buf, _ = gg.Next(buf)
|
|
|
|
if len(buf) == 0 {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
// Decompress this thing
|
2023-10-30 12:48:14 +00:00
|
|
|
buffer := buffersPool.Get().(*bytes.Buffer)
|
|
|
|
defer buffersPool.Put(buffer)
|
2023-11-13 13:10:57 +00:00
|
|
|
|
2023-10-30 12:48:14 +00:00
|
|
|
buffer.Reset()
|
|
|
|
buffer.Write(buf)
|
|
|
|
lzReader := lz4ReaderPool.Get().(*lz4.Reader)
|
|
|
|
defer lz4ReaderPool.Put(lzReader)
|
|
|
|
lzReader.Reset(buffer)
|
|
|
|
|
2023-11-13 13:10:57 +00:00
|
|
|
// Use pooled buffers and readers to avoid allocations.
|
2023-10-30 12:48:14 +00:00
|
|
|
return snapshot_format.ReadBlockFromSnapshot(lzReader, r.eth1Getter, r.cfg)
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
|
|
|
|
2023-11-15 14:07:16 +00:00
|
|
|
func (r *beaconSnapshotReader) ReadBlockByRoot(ctx context.Context, tx kv.Tx, root libcommon.Hash) (*cltypes.SignedBeaconBlock, error) {
|
2023-10-24 19:32:29 +00:00
|
|
|
view := r.sn.View()
|
|
|
|
defer view.Close()
|
|
|
|
|
2023-11-22 00:45:15 +00:00
|
|
|
slot, err := beacon_indicies.ReadBlockSlotByBlockRoot(tx, root)
|
2023-11-15 14:07:16 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
if slot == nil {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
|
2023-10-24 19:32:29 +00:00
|
|
|
var buf []byte
|
2023-11-22 00:45:15 +00:00
|
|
|
if *slot > r.sn.BlocksAvailable() {
|
|
|
|
data, err := r.beaconDB.GetBlock(ctx, tx, *slot)
|
2023-11-15 14:07:16 +00:00
|
|
|
return data.Data, err
|
|
|
|
}
|
|
|
|
if r.eth1Getter == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
// Find canonical block
|
|
|
|
canonicalBlockRoot, err := beacon_indicies.ReadCanonicalBlockRoot(tx, *slot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// root non-canonical? BAD
|
|
|
|
if canonicalBlockRoot != root {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2023-10-24 19:32:29 +00:00
|
|
|
|
2023-11-22 00:45:15 +00:00
|
|
|
seg, ok := view.BeaconBlocksSegment(*slot)
|
2023-10-24 19:32:29 +00:00
|
|
|
if !ok {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, nil
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if seg.idxSlot == nil {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
if *slot < seg.idxSlot.BaseDataID() {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, fmt.Errorf("slot %d is before the base data id %d", slot, seg.idxSlot.BaseDataID())
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
blockOffset := seg.idxSlot.OrdinalLookup(*slot - seg.idxSlot.BaseDataID())
|
2023-10-24 19:32:29 +00:00
|
|
|
|
|
|
|
gg := seg.seg.MakeGetter()
|
|
|
|
gg.Reset(blockOffset)
|
|
|
|
if !gg.HasNext() {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, nil
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
2023-11-13 13:10:57 +00:00
|
|
|
|
2023-11-15 14:07:16 +00:00
|
|
|
buf = buf[:0]
|
2023-10-24 19:32:29 +00:00
|
|
|
buf, _ = gg.Next(buf)
|
2023-11-13 13:10:57 +00:00
|
|
|
if len(buf) == 0 {
|
2023-11-15 14:07:16 +00:00
|
|
|
return nil, nil
|
2023-11-13 13:10:57 +00:00
|
|
|
}
|
|
|
|
// Decompress this thing
|
|
|
|
buffer := buffersPool.Get().(*bytes.Buffer)
|
|
|
|
defer buffersPool.Put(buffer)
|
|
|
|
|
|
|
|
buffer.Reset()
|
|
|
|
buffer.Write(buf)
|
|
|
|
lzReader := lz4ReaderPool.Get().(*lz4.Reader)
|
|
|
|
defer lz4ReaderPool.Put(lzReader)
|
|
|
|
lzReader.Reset(buffer)
|
|
|
|
|
|
|
|
// Use pooled buffers and readers to avoid allocations.
|
2023-11-15 14:07:16 +00:00
|
|
|
return snapshot_format.ReadBlockFromSnapshot(lzReader, r.eth1Getter, r.cfg)
|
2023-10-24 19:32:29 +00:00
|
|
|
}
|
2023-11-22 00:45:15 +00:00
|
|
|
|
|
|
|
func (r *beaconSnapshotReader) ReadHeaderByRoot(ctx context.Context, tx kv.Tx, root libcommon.Hash) (*cltypes.SignedBeaconBlockHeader, error) {
|
|
|
|
view := r.sn.View()
|
|
|
|
defer view.Close()
|
|
|
|
|
|
|
|
slot, err := beacon_indicies.ReadBlockSlotByBlockRoot(tx, root)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if slot == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if *slot > r.sn.BlocksAvailable() {
|
|
|
|
h, _, err := beacon_indicies.ReadSignedHeaderByBlockRoot(ctx, tx, root)
|
|
|
|
return h, err
|
|
|
|
}
|
|
|
|
// Find canonical block
|
|
|
|
canonicalBlockRoot, err := beacon_indicies.ReadCanonicalBlockRoot(tx, *slot)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// root non-canonical? BAD
|
|
|
|
if canonicalBlockRoot != root {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
h, _, _, err := r.sn.ReadHeader(*slot)
|
|
|
|
// Use pooled buffers and readers to avoid allocations.
|
|
|
|
return h, err
|
|
|
|
}
|