2023-07-08 17:01:26 +00:00
|
|
|
package jsonrpc
|
2023-02-20 11:23:06 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2023-10-21 23:17:18 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/common/hexutil"
|
2023-02-20 11:23:06 +00:00
|
|
|
"math/big"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
|
|
"github.com/ledgerwatch/erigon/rpc"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/adapter/ethapi"
|
|
|
|
"github.com/ledgerwatch/erigon/turbo/rpchelper"
|
|
|
|
)
|
|
|
|
|
|
|
|
type GraphQLAPI interface {
|
|
|
|
GetBlockDetails(ctx context.Context, number rpc.BlockNumber) (map[string]interface{}, error)
|
|
|
|
GetChainID(ctx context.Context) (*big.Int, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
type GraphQLAPIImpl struct {
|
|
|
|
*BaseAPI
|
|
|
|
db kv.RoDB
|
|
|
|
}
|
|
|
|
|
|
|
|
func NewGraphQLAPI(base *BaseAPI, db kv.RoDB) *GraphQLAPIImpl {
|
|
|
|
return &GraphQLAPIImpl{
|
|
|
|
BaseAPI: base,
|
|
|
|
db: db,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api *GraphQLAPIImpl) GetChainID(ctx context.Context) (*big.Int, error) {
|
|
|
|
tx, err := api.db.BeginRo(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
|
|
|
response, err := api.chainConfig(tx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return response.ChainID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api *GraphQLAPIImpl) GetBlockDetails(ctx context.Context, blockNumber rpc.BlockNumber) (map[string]interface{}, error) {
|
|
|
|
tx, err := api.db.BeginRo(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
|
|
|
block, senders, err := api.getBlockWithSenders(ctx, blockNumber, tx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if block == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
getBlockRes, err := api.delegateGetBlockByNumber(tx, block, blockNumber, false)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
chainConfig, err := api.chainConfig(tx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
receipts, err := api.getReceipts(ctx, tx, chainConfig, block, senders)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("getReceipts error: %w", err)
|
|
|
|
}
|
2023-06-02 20:26:19 +00:00
|
|
|
|
2023-02-20 11:23:06 +00:00
|
|
|
result := make([]map[string]interface{}, 0, len(receipts))
|
|
|
|
for _, receipt := range receipts {
|
|
|
|
txn := block.Transactions()[receipt.TransactionIndex]
|
2023-03-06 18:58:10 +00:00
|
|
|
|
2023-06-02 20:26:19 +00:00
|
|
|
transaction := marshalReceipt(receipt, txn, chainConfig, block.HeaderNoCopy(), txn.Hash(), true)
|
2023-03-06 18:58:10 +00:00
|
|
|
transaction["nonce"] = txn.GetNonce()
|
|
|
|
transaction["value"] = txn.GetValue()
|
|
|
|
transaction["data"] = txn.GetData()
|
2023-03-09 02:22:41 +00:00
|
|
|
transaction["logs"] = receipt.Logs
|
2023-03-06 18:58:10 +00:00
|
|
|
result = append(result, transaction)
|
2023-02-20 11:23:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
response := map[string]interface{}{}
|
|
|
|
response["block"] = getBlockRes
|
|
|
|
response["receipts"] = result
|
|
|
|
|
|
|
|
return response, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api *GraphQLAPIImpl) getBlockWithSenders(ctx context.Context, number rpc.BlockNumber, tx kv.Tx) (*types.Block, []common.Address, error) {
|
|
|
|
if number == rpc.PendingBlockNumber {
|
|
|
|
return api.pendingBlock(), nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
blockHeight, blockHash, _, err := rpchelper.GetBlockNumber(rpc.BlockNumberOrHashWithNumber(number), tx, api.filters)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
block, senders, err := api._blockReader.BlockWithSenders(ctx, tx, blockHash, blockHeight)
|
|
|
|
return block, senders, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (api *GraphQLAPIImpl) delegateGetBlockByNumber(tx kv.Tx, b *types.Block, number rpc.BlockNumber, inclTx bool) (map[string]interface{}, error) {
|
|
|
|
td, err := rawdb.ReadTd(tx, b.Hash(), b.NumberU64())
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
additionalFields := make(map[string]interface{})
|
|
|
|
response, err := ethapi.RPCMarshalBlock(b, inclTx, inclTx, additionalFields)
|
|
|
|
if !inclTx {
|
|
|
|
delete(response, "transactions") // workaround for https://github.com/ledgerwatch/erigon/issues/4989#issuecomment-1218415666
|
|
|
|
}
|
|
|
|
response["totalDifficulty"] = (*hexutil.Big)(td)
|
|
|
|
response["transactionCount"] = b.Transactions().Len()
|
|
|
|
|
|
|
|
if err == nil && number == rpc.PendingBlockNumber {
|
|
|
|
// Pending blocks need to nil out a few fields
|
|
|
|
for _, field := range []string{"hash", "nonce", "miner"} {
|
|
|
|
response[field] = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return response, err
|
|
|
|
}
|