mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-05 10:32:19 +00:00
a2433455f9
Now keep few beacon block headers in mdbx
276 lines
8.2 KiB
Go
276 lines
8.2 KiB
Go
package beacon_indicies
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
"github.com/ledgerwatch/erigon/cl/persistence/base_encoding"
|
|
_ "modernc.org/sqlite"
|
|
)
|
|
|
|
func WriteHighestFinalized(tx kv.RwTx, slot uint64) error {
|
|
return tx.Put(kv.HighestFinalized, kv.HighestFinalizedKey, base_encoding.Encode64ToBytes4(slot))
|
|
}
|
|
|
|
func ReadHighestFinalized(tx kv.Tx) (uint64, error) {
|
|
val, err := tx.GetOne(kv.HighestFinalized, kv.HighestFinalizedKey)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if len(val) == 0 {
|
|
return 0, nil
|
|
}
|
|
return base_encoding.Decode64FromBytes4(val), nil
|
|
}
|
|
|
|
// WriteBlockRootSlot writes the slot associated with a block root.
|
|
func WriteHeaderSlot(tx kv.RwTx, blockRoot libcommon.Hash, slot uint64) error {
|
|
return tx.Put(kv.BlockRootToSlot, blockRoot[:], base_encoding.Encode64ToBytes4(slot))
|
|
}
|
|
|
|
func ReadBlockSlotByBlockRoot(tx kv.Tx, blockRoot libcommon.Hash) (*uint64, error) {
|
|
slotBytes, err := tx.GetOne(kv.BlockRootToSlot, blockRoot[:])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(slotBytes) == 0 {
|
|
return nil, nil
|
|
}
|
|
slot := new(uint64)
|
|
*slot = base_encoding.Decode64FromBytes4(slotBytes)
|
|
return slot, nil
|
|
}
|
|
|
|
// WriteBlockRootSlot writes the slot associated with a block root.
|
|
func WriteStateRoot(tx kv.RwTx, blockRoot libcommon.Hash, stateRoot libcommon.Hash) error {
|
|
if err := tx.Put(kv.BlockRootToStateRoot, blockRoot[:], stateRoot[:]); err != nil {
|
|
return err
|
|
}
|
|
return tx.Put(kv.StateRootToBlockRoot, stateRoot[:], blockRoot[:])
|
|
}
|
|
|
|
func ReadStateRootByBlockRoot(ctx context.Context, tx kv.Tx, blockRoot libcommon.Hash) (libcommon.Hash, error) {
|
|
var stateRoot libcommon.Hash
|
|
|
|
sRoot, err := tx.GetOne(kv.BlockRootToStateRoot, blockRoot[:])
|
|
if err != nil {
|
|
return libcommon.Hash{}, err
|
|
}
|
|
|
|
copy(stateRoot[:], sRoot)
|
|
|
|
return stateRoot, nil
|
|
}
|
|
|
|
func ReadBlockRootByStateRoot(tx kv.Tx, stateRoot libcommon.Hash) (libcommon.Hash, error) {
|
|
var blockRoot libcommon.Hash
|
|
|
|
bRoot, err := tx.GetOne(kv.StateRootToBlockRoot, stateRoot[:])
|
|
if err != nil {
|
|
return libcommon.Hash{}, err
|
|
}
|
|
|
|
copy(blockRoot[:], bRoot)
|
|
|
|
return blockRoot, nil
|
|
}
|
|
|
|
func ReadCanonicalBlockRoot(tx kv.Tx, slot uint64) (libcommon.Hash, error) {
|
|
var blockRoot libcommon.Hash
|
|
|
|
bRoot, err := tx.GetOne(kv.CanonicalBlockRoots, base_encoding.Encode64ToBytes4(slot))
|
|
if err != nil {
|
|
return libcommon.Hash{}, err
|
|
}
|
|
|
|
copy(blockRoot[:], bRoot)
|
|
return blockRoot, nil
|
|
}
|
|
|
|
func WriteLastBeaconSnapshot(tx kv.RwTx, slot uint64) error {
|
|
return tx.Put(kv.LastBeaconSnapshot, []byte(kv.LastBeaconSnapshotKey), base_encoding.Encode64ToBytes4(slot))
|
|
}
|
|
|
|
func ReadLastBeaconSnapshot(tx kv.Tx) (uint64, error) {
|
|
val, err := tx.GetOne(kv.LastBeaconSnapshot, []byte(kv.LastBeaconSnapshotKey))
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if len(val) == 0 {
|
|
return 0, nil
|
|
}
|
|
return base_encoding.Decode64FromBytes4(val), nil
|
|
}
|
|
|
|
func MarkRootCanonical(ctx context.Context, tx kv.RwTx, slot uint64, blockRoot libcommon.Hash) error {
|
|
return tx.Put(kv.CanonicalBlockRoots, base_encoding.Encode64ToBytes4(slot), blockRoot[:])
|
|
}
|
|
|
|
func WriteExecutionBlockNumber(tx kv.RwTx, blockRoot libcommon.Hash, blockNumber uint64) error {
|
|
return tx.Put(kv.BlockRootToBlockNumber, blockRoot[:], base_encoding.Encode64ToBytes4(blockNumber))
|
|
}
|
|
|
|
func WriteExecutionBlockHash(tx kv.RwTx, blockRoot, blockHash libcommon.Hash) error {
|
|
return tx.Put(kv.BlockRootToBlockHash, blockRoot[:], blockHash[:])
|
|
}
|
|
|
|
func ReadExecutionBlockNumber(tx kv.Tx, blockRoot libcommon.Hash) (*uint64, error) {
|
|
val, err := tx.GetOne(kv.BlockRootToBlockNumber, blockRoot[:])
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(val) == 0 {
|
|
return nil, nil
|
|
}
|
|
ret := new(uint64)
|
|
*ret = base_encoding.Decode64FromBytes4(val)
|
|
return ret, nil
|
|
}
|
|
|
|
func ReadExecutionBlockHash(tx kv.Tx, blockRoot libcommon.Hash) (libcommon.Hash, error) {
|
|
val, err := tx.GetOne(kv.BlockRootToBlockHash, blockRoot[:])
|
|
if err != nil {
|
|
return libcommon.Hash{}, err
|
|
}
|
|
if len(val) == 0 {
|
|
return libcommon.Hash{}, nil
|
|
}
|
|
return libcommon.BytesToHash(val), nil
|
|
}
|
|
|
|
func WriteBeaconBlockHeader(ctx context.Context, tx kv.RwTx, signedHeader *cltypes.SignedBeaconBlockHeader) error {
|
|
headersBytes, err := signedHeader.EncodeSSZ(nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
blockRoot, err := signedHeader.Header.HashSSZ()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return tx.Put(kv.BeaconBlockHeaders, blockRoot[:], headersBytes)
|
|
}
|
|
|
|
func WriteBeaconBlockHeaderAndIndicies(ctx context.Context, tx kv.RwTx, signedHeader *cltypes.SignedBeaconBlockHeader, forceCanonical bool) error {
|
|
blockRoot, err := signedHeader.Header.HashSSZ()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := WriteBeaconBlockHeader(ctx, tx, signedHeader); err != nil {
|
|
return err
|
|
}
|
|
if err := WriteHeaderSlot(tx, blockRoot, signedHeader.Header.Slot); err != nil {
|
|
return err
|
|
}
|
|
if err := WriteStateRoot(tx, blockRoot, signedHeader.Header.Root); err != nil {
|
|
return err
|
|
}
|
|
if err := WriteParentBlockRoot(ctx, tx, blockRoot, signedHeader.Header.ParentRoot); err != nil {
|
|
return err
|
|
}
|
|
if forceCanonical {
|
|
if err := MarkRootCanonical(ctx, tx, signedHeader.Header.Slot, blockRoot); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func ReadParentBlockRoot(ctx context.Context, tx kv.Tx, blockRoot libcommon.Hash) (libcommon.Hash, error) {
|
|
var parentRoot libcommon.Hash
|
|
|
|
pRoot, err := tx.GetOne(kv.BlockRootToParentRoot, blockRoot[:])
|
|
if err != nil {
|
|
return libcommon.Hash{}, err
|
|
}
|
|
|
|
copy(parentRoot[:], pRoot)
|
|
|
|
return parentRoot, nil
|
|
}
|
|
|
|
func WriteParentBlockRoot(ctx context.Context, tx kv.RwTx, blockRoot, parentRoot libcommon.Hash) error {
|
|
return tx.Put(kv.BlockRootToParentRoot, blockRoot[:], parentRoot[:])
|
|
}
|
|
|
|
func TruncateCanonicalChain(ctx context.Context, tx kv.RwTx, slot uint64) error {
|
|
return tx.ForEach(kv.CanonicalBlockRoots, base_encoding.Encode64ToBytes4(slot), func(k, _ []byte) error {
|
|
return tx.Delete(kv.CanonicalBlockRoots, k)
|
|
})
|
|
}
|
|
|
|
func PruneSignedHeaders(tx kv.RwTx, from uint64) error {
|
|
cursor, err := tx.RwCursor(kv.BeaconBlockHeaders)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for k, _, err := cursor.Seek(base_encoding.Encode64ToBytes4(from)); err == nil && k != nil; k, _, err = cursor.Prev() {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := cursor.DeleteCurrent(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func RangeBlockRoots(ctx context.Context, tx kv.Tx, fromSlot, toSlot uint64, fn func(slot uint64, beaconBlockRoot libcommon.Hash) bool) error {
|
|
cursor, err := tx.Cursor(kv.CanonicalBlockRoots)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for k, v, err := cursor.Seek(base_encoding.Encode64ToBytes4(fromSlot)); err == nil && k != nil && base_encoding.Decode64FromBytes4(k) <= toSlot; k, v, err = cursor.Next() {
|
|
if !fn(base_encoding.Decode64FromBytes4(k), libcommon.BytesToHash(v)) {
|
|
break
|
|
}
|
|
}
|
|
return err
|
|
}
|
|
|
|
func PruneBlockRoots(ctx context.Context, tx kv.RwTx, fromSlot, toSlot uint64) error {
|
|
cursor, err := tx.RwCursor(kv.CanonicalBlockRoots)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for k, _, err := cursor.Seek(base_encoding.Encode64ToBytes4(fromSlot)); err == nil && k != nil && base_encoding.Decode64FromBytes4(k) <= toSlot; k, _, err = cursor.Next() {
|
|
if err := cursor.DeleteCurrent(); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return err
|
|
}
|
|
|
|
func ReadBeaconBlockRootsInSlotRange(ctx context.Context, tx kv.Tx, fromSlot, count uint64) ([]libcommon.Hash, []uint64, error) {
|
|
blockRoots := make([]libcommon.Hash, 0, count)
|
|
slots := make([]uint64, 0, count)
|
|
err := RangeBlockRoots(ctx, tx, fromSlot, fromSlot+count, func(slot uint64, beaconBlockRoot libcommon.Hash) bool {
|
|
blockRoots = append(blockRoots, beaconBlockRoot)
|
|
slots = append(slots, slot)
|
|
return true
|
|
})
|
|
return blockRoots, slots, err
|
|
}
|
|
|
|
func ReadSignedHeaderByBlockRoot(ctx context.Context, tx kv.Tx, blockRoot libcommon.Hash) (*cltypes.SignedBeaconBlockHeader, bool, error) {
|
|
h := &cltypes.SignedBeaconBlockHeader{Header: &cltypes.BeaconBlockHeader{}}
|
|
headerBytes, err := tx.GetOne(kv.BeaconBlockHeaders, blockRoot[:])
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
if len(headerBytes) == 0 {
|
|
return nil, false, nil
|
|
}
|
|
if err := h.DecodeSSZ(headerBytes, 0); err != nil {
|
|
return nil, false, fmt.Errorf("failed to decode BeaconHeader: %v", err)
|
|
}
|
|
canonical, err := ReadCanonicalBlockRoot(tx, h.Header.Slot)
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
return h, canonical == blockRoot, nil
|
|
}
|