2023-07-28 22:22:38 +00:00
|
|
|
package eth1_chain_reader
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-08-05 00:26:53 +00:00
|
|
|
"fmt"
|
2023-07-28 22:22:38 +00:00
|
|
|
"math/big"
|
2023-08-05 00:26:53 +00:00
|
|
|
"time"
|
2023-07-28 22:22:38 +00:00
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon-lib/chain"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/gointerfaces/execution"
|
2023-08-26 00:25:48 +00:00
|
|
|
types2 "github.com/ledgerwatch/erigon-lib/gointerfaces/types"
|
2023-07-28 22:22:38 +00:00
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/execution/eth1/eth1_utils"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
2023-07-30 21:35:55 +00:00
|
|
|
"google.golang.org/protobuf/types/known/emptypb"
|
2023-07-28 22:22:38 +00:00
|
|
|
)
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
type ChainReaderWriterEth1 struct {
|
2023-07-28 22:22:38 +00:00
|
|
|
ctx context.Context
|
|
|
|
cfg *chain.Config
|
|
|
|
executionModule execution.ExecutionClient
|
2023-08-05 00:26:53 +00:00
|
|
|
|
|
|
|
fcuTimoutMillis uint64
|
2023-07-28 22:22:38 +00:00
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func NewChainReaderEth1(ctx context.Context, cfg *chain.Config, executionModule execution.ExecutionClient, fcuTimoutMillis uint64) ChainReaderWriterEth1 {
|
|
|
|
return ChainReaderWriterEth1{
|
2023-07-28 22:22:38 +00:00
|
|
|
ctx: ctx,
|
|
|
|
cfg: cfg,
|
|
|
|
executionModule: executionModule,
|
2023-08-05 00:26:53 +00:00
|
|
|
fcuTimoutMillis: fcuTimoutMillis,
|
2023-07-28 22:22:38 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) Config() *chain.Config {
|
2023-07-28 22:22:38 +00:00
|
|
|
return c.cfg
|
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) CurrentHeader() *types.Header {
|
2023-07-30 21:35:55 +00:00
|
|
|
resp, err := c.executionModule.CurrentHeader(c.ctx, &emptypb.Empty{})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeader failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ret, err := eth1_utils.HeaderRpcToHeader(resp.Header)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeader decoding", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return ret
|
2023-07-28 22:22:38 +00:00
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetHeader(hash libcommon.Hash, number uint64) *types.Header {
|
2023-07-28 22:22:38 +00:00
|
|
|
resp, err := c.executionModule.GetHeader(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: &number,
|
|
|
|
BlockHash: gointerfaces.ConvertHashToH256(hash),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeader failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ret, err := eth1_utils.HeaderRpcToHeader(resp.Header)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeader decoding", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetBlockByHash(hash libcommon.Hash) *types.Block {
|
|
|
|
header := c.GetHeaderByHash(hash)
|
|
|
|
if header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
number := header.Number.Uint64()
|
|
|
|
resp, err := c.executionModule.GetBody(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: &number,
|
|
|
|
BlockHash: gointerfaces.ConvertHashToH256(hash),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Body == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2024-02-07 16:39:14 +00:00
|
|
|
body, err := eth1_utils.ConvertRawBlockBodyFromRpc(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-05 00:26:53 +00:00
|
|
|
txs, err := types.DecodeTransactions(body.Transactions)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return types.NewBlock(header, txs, nil, nil, body.Withdrawals)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) GetBlockByNumber(number uint64) *types.Block {
|
|
|
|
header := c.GetHeaderByNumber(number)
|
|
|
|
if header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, err := c.executionModule.GetBody(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: &number,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByNumber failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Body == nil {
|
|
|
|
return nil
|
|
|
|
}
|
2024-02-07 16:39:14 +00:00
|
|
|
body, err := eth1_utils.ConvertRawBlockBodyFromRpc(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByNumber failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-05 00:26:53 +00:00
|
|
|
txs, err := types.DecodeTransactions(body.Transactions)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetBlockByNumber failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return types.NewBlock(header, txs, nil, nil, body.Withdrawals)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) GetHeaderByHash(hash libcommon.Hash) *types.Header {
|
2023-07-28 22:22:38 +00:00
|
|
|
resp, err := c.executionModule.GetHeader(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: nil,
|
|
|
|
BlockHash: gointerfaces.ConvertHashToH256(hash),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeaderByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ret, err := eth1_utils.HeaderRpcToHeader(resp.Header)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeaderByHash decoding", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetHeaderByNumber(number uint64) *types.Header {
|
2023-07-28 22:22:38 +00:00
|
|
|
resp, err := c.executionModule.GetHeader(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: &number,
|
|
|
|
BlockHash: nil,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeaderByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Header == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
ret, err := eth1_utils.HeaderRpcToHeader(resp.Header)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeaderByHash decoding", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetTd(hash libcommon.Hash, number uint64) *big.Int {
|
2023-07-28 22:22:38 +00:00
|
|
|
resp, err := c.executionModule.GetTD(c.ctx, &execution.GetSegmentRequest{
|
|
|
|
BlockNumber: &number,
|
|
|
|
BlockHash: gointerfaces.ConvertHashToH256(hash),
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeaderByHash failed", "err", err)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if resp == nil || resp.Td == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return eth1_utils.ConvertBigIntFromRpc(resp.Td)
|
|
|
|
}
|
|
|
|
|
2023-11-17 12:04:02 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetBodiesByHashes(hashes []libcommon.Hash) ([]*types.RawBody, error) {
|
2023-08-26 00:25:48 +00:00
|
|
|
grpcHashes := make([]*types2.H256, len(hashes))
|
|
|
|
for i := range grpcHashes {
|
|
|
|
grpcHashes[i] = gointerfaces.ConvertHashToH256(hashes[i])
|
|
|
|
}
|
|
|
|
resp, err := c.executionModule.GetBodiesByHashes(c.ctx, &execution.GetBodiesByHashesRequest{
|
|
|
|
Hashes: grpcHashes,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-11-17 12:04:02 +00:00
|
|
|
return nil, err
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
|
|
|
ret := make([]*types.RawBody, len(resp.Bodies))
|
|
|
|
for i := range ret {
|
2024-02-07 16:39:14 +00:00
|
|
|
ret[i], err = eth1_utils.ConvertRawBlockBodyFromRpc(resp.Bodies[i])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
2023-11-17 12:04:02 +00:00
|
|
|
return ret, nil
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
|
|
|
|
2023-11-17 12:04:02 +00:00
|
|
|
func (c ChainReaderWriterEth1) GetBodiesByRange(start, count uint64) ([]*types.RawBody, error) {
|
2023-08-26 00:25:48 +00:00
|
|
|
resp, err := c.executionModule.GetBodiesByRange(c.ctx, &execution.GetBodiesByRangeRequest{
|
|
|
|
Start: start,
|
|
|
|
Count: count,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-11-17 12:04:02 +00:00
|
|
|
return nil, err
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
|
|
|
ret := make([]*types.RawBody, len(resp.Bodies))
|
|
|
|
for i := range ret {
|
2024-02-07 16:39:14 +00:00
|
|
|
ret[i], err = eth1_utils.ConvertRawBlockBodyFromRpc(resp.Bodies[i])
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
2023-11-17 12:04:02 +00:00
|
|
|
return ret, nil
|
2023-08-26 00:25:48 +00:00
|
|
|
}
|
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) Ready() (bool, error) {
|
|
|
|
resp, err := c.executionModule.Ready(c.ctx, &emptypb.Empty{})
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return resp.Ready, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) HeaderNumber(hash libcommon.Hash) (*uint64, error) {
|
2023-07-28 22:22:38 +00:00
|
|
|
resp, err := c.executionModule.GetHeaderHashNumber(c.ctx, gointerfaces.ConvertHashToH256(hash))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if resp == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return resp.BlockNumber, nil
|
|
|
|
}
|
2023-08-01 00:08:15 +00:00
|
|
|
|
2023-08-05 00:26:53 +00:00
|
|
|
func (c ChainReaderWriterEth1) IsCanonicalHash(hash libcommon.Hash) (bool, error) {
|
2023-08-01 00:08:15 +00:00
|
|
|
resp, err := c.executionModule.IsCanonicalHash(c.ctx, gointerfaces.ConvertHashToH256(hash))
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if resp == nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
return resp.Canonical, nil
|
|
|
|
}
|
2023-08-05 00:26:53 +00:00
|
|
|
|
2023-10-16 13:35:26 +00:00
|
|
|
func (c ChainReaderWriterEth1) FrozenBlocks() uint64 {
|
|
|
|
ret, err := c.executionModule.FrozenBlocks(c.ctx, &emptypb.Empty{})
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
return ret.FrozenBlocks
|
2023-08-05 00:26:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const retryTimeout = 10 * time.Millisecond
|
|
|
|
|
2023-08-11 21:07:36 +00:00
|
|
|
func (c ChainReaderWriterEth1) InsertBlocksAndWait(blocks []*types.Block) error {
|
|
|
|
request := &execution.InsertBlocksRequest{
|
|
|
|
Blocks: eth1_utils.ConvertBlocksToRPC(blocks),
|
2023-08-05 00:26:53 +00:00
|
|
|
}
|
2023-08-11 21:07:36 +00:00
|
|
|
response, err := c.executionModule.InsertBlocks(c.ctx, request)
|
2023-08-05 00:26:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
retryInterval := time.NewTicker(retryTimeout)
|
|
|
|
defer retryInterval.Stop()
|
|
|
|
for response.Result == execution.ExecutionStatus_Busy {
|
|
|
|
select {
|
|
|
|
case <-retryInterval.C:
|
2023-08-11 21:07:36 +00:00
|
|
|
response, err = c.executionModule.InsertBlocks(c.ctx, request)
|
2023-08-05 00:26:53 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case <-c.ctx.Done():
|
|
|
|
return context.Canceled
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if response.Result != execution.ExecutionStatus_Success {
|
|
|
|
return fmt.Errorf("insertHeadersAndWait: invalid code recieved from execution module: %s", response.Result.String())
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-08-11 21:07:36 +00:00
|
|
|
func (c ChainReaderWriterEth1) InsertBlockAndWait(block *types.Block) error {
|
|
|
|
return c.InsertBlocksAndWait([]*types.Block{block})
|
2023-08-05 00:26:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) ValidateChain(hash libcommon.Hash, number uint64) (execution.ExecutionStatus, libcommon.Hash, error) {
|
|
|
|
resp, err := c.executionModule.ValidateChain(c.ctx, &execution.ValidationRequest{
|
|
|
|
Hash: gointerfaces.ConvertHashToH256(hash),
|
|
|
|
Number: number,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return 0, libcommon.Hash{}, err
|
|
|
|
}
|
|
|
|
return resp.ValidationStatus, gointerfaces.ConvertH256ToHash(resp.LatestValidHash), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) UpdateForkChoice(headHash, safeHash, finalizeHash libcommon.Hash) (execution.ExecutionStatus, libcommon.Hash, error) {
|
|
|
|
resp, err := c.executionModule.UpdateForkChoice(c.ctx, &execution.ForkChoice{
|
|
|
|
HeadBlockHash: gointerfaces.ConvertHashToH256(headHash),
|
|
|
|
SafeBlockHash: gointerfaces.ConvertHashToH256(safeHash),
|
|
|
|
FinalizedBlockHash: gointerfaces.ConvertHashToH256(finalizeHash),
|
|
|
|
Timeout: c.fcuTimoutMillis,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return 0, libcommon.Hash{}, err
|
|
|
|
}
|
|
|
|
return resp.Status, gointerfaces.ConvertH256ToHash(resp.LatestValidHash), err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c ChainReaderWriterEth1) GetForkchoice() (headHash, finalizedHash, safeHash libcommon.Hash, err error) {
|
|
|
|
var resp *execution.ForkChoice
|
|
|
|
resp, err = c.executionModule.GetForkChoice(c.ctx, &emptypb.Empty{})
|
|
|
|
if err != nil {
|
|
|
|
log.Error("GetHeader failed", "err", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return gointerfaces.ConvertH256ToHash(resp.HeadBlockHash), gointerfaces.ConvertH256ToHash(resp.FinalizedBlockHash),
|
|
|
|
gointerfaces.ConvertH256ToHash(resp.SafeBlockHash), nil
|
|
|
|
}
|