mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-20 09:21:11 +00:00
303 lines
7.6 KiB
Go
303 lines
7.6 KiB
Go
package jsonrpc
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutility"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon-lib/kv/order"
|
|
"github.com/ledgerwatch/erigon-lib/kv/rawdbv3"
|
|
"github.com/ledgerwatch/erigon-lib/kv/temporal/historyv2"
|
|
"github.com/ledgerwatch/erigon/turbo/services"
|
|
|
|
"github.com/ledgerwatch/erigon/common/hexutil"
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/core/types/accounts"
|
|
"github.com/ledgerwatch/erigon/rpc"
|
|
"github.com/ledgerwatch/erigon/turbo/adapter/ethapi"
|
|
"github.com/ledgerwatch/erigon/turbo/rpchelper"
|
|
)
|
|
|
|
// GetHeaderByNumber implements erigon_getHeaderByNumber. Returns a block's header given a block number ignoring the block's transaction and uncle list (may be faster).
|
|
func (api *ErigonImpl) GetHeaderByNumber(ctx context.Context, blockNumber rpc.BlockNumber) (*types.Header, error) {
|
|
// Pending block is only known by the miner
|
|
if blockNumber == rpc.PendingBlockNumber {
|
|
block := api.pendingBlock()
|
|
if block == nil {
|
|
return nil, nil
|
|
}
|
|
return block.Header(), nil
|
|
}
|
|
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
blockNum, _, _, err := rpchelper.GetBlockNumber(rpc.BlockNumberOrHashWithNumber(blockNumber), tx, api.filters)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
header, err := api._blockReader.HeaderByNumber(ctx, tx, blockNum)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if header == nil {
|
|
return nil, fmt.Errorf("block header not found: %d", blockNum)
|
|
}
|
|
|
|
return header, nil
|
|
}
|
|
|
|
// GetHeaderByHash implements erigon_getHeaderByHash. Returns a block's header given a block's hash.
|
|
func (api *ErigonImpl) GetHeaderByHash(ctx context.Context, hash common.Hash) (*types.Header, error) {
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
header, err := api._blockReader.HeaderByHash(ctx, tx, hash)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if header == nil {
|
|
return nil, fmt.Errorf("block header not found: %s", hash.String())
|
|
}
|
|
|
|
return header, nil
|
|
}
|
|
|
|
func (api *ErigonImpl) GetBlockByTimestamp(ctx context.Context, timeStamp rpc.Timestamp, fullTx bool) (map[string]interface{}, error) {
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
uintTimestamp := timeStamp.TurnIntoUint64()
|
|
|
|
currentHeader := rawdb.ReadCurrentHeader(tx)
|
|
currentHeaderTime := currentHeader.Time
|
|
highestNumber := currentHeader.Number.Uint64()
|
|
|
|
firstHeader, err := api._blockReader.HeaderByNumber(ctx, tx, 0)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if firstHeader == nil {
|
|
return nil, errors.New("no genesis header found")
|
|
}
|
|
|
|
firstHeaderTime := firstHeader.Time
|
|
|
|
if currentHeaderTime <= uintTimestamp {
|
|
blockResponse, err := buildBlockResponse(api._blockReader, tx, highestNumber, fullTx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return blockResponse, nil
|
|
}
|
|
|
|
if firstHeaderTime >= uintTimestamp {
|
|
blockResponse, err := buildBlockResponse(api._blockReader, tx, 0, fullTx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return blockResponse, nil
|
|
}
|
|
|
|
blockNum := sort.Search(int(currentHeader.Number.Uint64()), func(blockNum int) bool {
|
|
currentHeader, err := api._blockReader.HeaderByNumber(ctx, tx, uint64(blockNum))
|
|
if err != nil {
|
|
return false
|
|
}
|
|
|
|
if currentHeader == nil {
|
|
return false
|
|
}
|
|
|
|
return currentHeader.Time >= uintTimestamp
|
|
})
|
|
|
|
resultingHeader, err := api._blockReader.HeaderByNumber(ctx, tx, uint64(blockNum))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if resultingHeader == nil {
|
|
return nil, fmt.Errorf("no header found with header number: %d", blockNum)
|
|
}
|
|
|
|
for resultingHeader.Time > uintTimestamp {
|
|
beforeHeader, err := api._blockReader.HeaderByNumber(ctx, tx, uint64(blockNum)-1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if beforeHeader == nil || beforeHeader.Time < uintTimestamp {
|
|
break
|
|
}
|
|
|
|
blockNum--
|
|
resultingHeader = beforeHeader
|
|
}
|
|
|
|
response, err := buildBlockResponse(api._blockReader, tx, uint64(blockNum), fullTx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return response, nil
|
|
}
|
|
|
|
func buildBlockResponse(br services.FullBlockReader, db kv.Tx, blockNum uint64, fullTx bool) (map[string]interface{}, error) {
|
|
header, err := br.HeaderByNumber(context.Background(), db, blockNum)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if header == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
block, _, err := br.BlockWithSenders(context.Background(), db, header.Hash(), blockNum)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if block == nil {
|
|
return nil, nil
|
|
}
|
|
|
|
additionalFields := make(map[string]interface{})
|
|
td, err := rawdb.ReadTd(db, header.Hash(), header.Number.Uint64())
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if td != nil {
|
|
additionalFields["totalDifficulty"] = (*hexutil.Big)(td)
|
|
}
|
|
|
|
response, err := ethapi.RPCMarshalBlockEx(block, true, fullTx, nil, common.Hash{}, additionalFields)
|
|
|
|
if err == nil && rpc.BlockNumber(block.NumberU64()) == rpc.PendingBlockNumber {
|
|
// Pending blocks need to nil out a few fields
|
|
for _, field := range []string{"hash", "nonce", "miner"} {
|
|
response[field] = nil
|
|
}
|
|
}
|
|
return response, err
|
|
}
|
|
|
|
func (api *ErigonImpl) GetBalanceChangesInBlock(ctx context.Context, blockNrOrHash rpc.BlockNumberOrHash) (map[common.Address]*hexutil.Big, error) {
|
|
tx, err := api.db.BeginRo(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer tx.Rollback()
|
|
|
|
balancesMapping := make(map[common.Address]*hexutil.Big)
|
|
latestState, err := rpchelper.CreateStateReader(ctx, tx, blockNrOrHash, 0, api.filters, api.stateCache, api.historyV3(tx), "")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
blockNumber, _, _, err := rpchelper.GetBlockNumber(blockNrOrHash, tx, api.filters)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if api.historyV3(tx) {
|
|
minTxNum, _ := rawdbv3.TxNums.Min(tx, blockNumber)
|
|
it, err := tx.(kv.TemporalTx).HistoryRange(kv.AccountsHistory, int(minTxNum), -1, order.Asc, -1)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for it.HasNext() {
|
|
addressBytes, v, err := it.Next()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var oldAcc accounts.Account
|
|
if len(v) > 0 {
|
|
if err = accounts.DeserialiseV3(&oldAcc, v); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
oldBalance := oldAcc.Balance
|
|
|
|
address := common.BytesToAddress(addressBytes)
|
|
newAcc, err := latestState.ReadAccountData(address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
newBalance := uint256.NewInt(0)
|
|
if newAcc != nil {
|
|
newBalance = &newAcc.Balance
|
|
}
|
|
|
|
if !oldBalance.Eq(newBalance) {
|
|
newBalanceDesc := (*hexutil.Big)(newBalance.ToBig())
|
|
balancesMapping[address] = newBalanceDesc
|
|
}
|
|
}
|
|
}
|
|
|
|
c, err := tx.Cursor(kv.AccountChangeSet)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
defer c.Close()
|
|
|
|
startkey := hexutility.EncodeTs(blockNumber)
|
|
|
|
decodeFn := historyv2.Mapper[kv.AccountChangeSet].Decode
|
|
|
|
for dbKey, dbValue, err := c.Seek(startkey); bytes.Equal(dbKey, startkey) && dbKey != nil; dbKey, dbValue, err = c.Next() {
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
_, addressBytes, v, err := decodeFn(dbKey, dbValue)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
var oldAcc accounts.Account
|
|
if err = oldAcc.DecodeForStorage(v); err != nil {
|
|
return nil, err
|
|
}
|
|
oldBalance := oldAcc.Balance
|
|
address := common.BytesToAddress(addressBytes)
|
|
|
|
newAcc, err := latestState.ReadAccountData(address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
newBalance := uint256.NewInt(0)
|
|
if newAcc != nil {
|
|
newBalance = &newAcc.Balance
|
|
}
|
|
|
|
if !oldBalance.Eq(newBalance) {
|
|
newBalanceDesc := (*hexutil.Big)(newBalance.ToBig())
|
|
balancesMapping[address] = newBalanceDesc
|
|
}
|
|
}
|
|
|
|
return balancesMapping, nil
|
|
}
|