77ed4aa754
* Store eth tx index separately Closes: #1075 Solution: - run a optional indexer service - adapt the json-rpc to the more efficient query changelog changelog fix lint fix backward compatibility fix lint timeout better strconv fix linter fix package name add cli command to index old tx fix for loop indexer cmd don't have access to local rpc workaround exceed block gas limit situation add unit tests for indexer refactor polish the indexer module Update server/config/toml.go Co-authored-by: Federico Kunze Küllmer <31522760+fedekunze@users.noreply.github.com> improve comments share code between GetTxByEthHash and GetTxByIndex fix unit test Update server/indexer.go Co-authored-by: Freddy Caceres <facs95@gmail.com> * Apply suggestions from code review * test enable-indexer in integration test * fix go lint * address review suggestions * fix linter * address review suggestions - test indexer in backend unit test - add comments * fix build * fix test * service name Co-authored-by: Freddy Caceres <facs95@gmail.com> Co-authored-by: Federico Kunze Küllmer <31522760+fedekunze@users.noreply.github.com>
167 lines
5.2 KiB
Go
167 lines
5.2 KiB
Go
// Package rpc contains RPC handler methods and utilities to start
|
|
// Ethermint's Web3-compatibly JSON-RPC server.
|
|
package rpc
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/cosmos/cosmos-sdk/client"
|
|
"github.com/cosmos/cosmos-sdk/server"
|
|
|
|
"github.com/ethereum/go-ethereum/rpc"
|
|
|
|
"github.com/evmos/ethermint/rpc/backend"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/debug"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/eth"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/eth/filters"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/miner"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/net"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/personal"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/txpool"
|
|
"github.com/evmos/ethermint/rpc/namespaces/ethereum/web3"
|
|
ethermint "github.com/evmos/ethermint/types"
|
|
|
|
rpcclient "github.com/tendermint/tendermint/rpc/jsonrpc/client"
|
|
)
|
|
|
|
// RPC namespaces and API version
|
|
const (
|
|
// Cosmos namespaces
|
|
|
|
CosmosNamespace = "cosmos"
|
|
|
|
// Ethereum namespaces
|
|
|
|
Web3Namespace = "web3"
|
|
EthNamespace = "eth"
|
|
PersonalNamespace = "personal"
|
|
NetNamespace = "net"
|
|
TxPoolNamespace = "txpool"
|
|
DebugNamespace = "debug"
|
|
MinerNamespace = "miner"
|
|
|
|
apiVersion = "1.0"
|
|
)
|
|
|
|
// APICreator creates the JSON-RPC API implementations.
|
|
type APICreator = func(
|
|
ctx *server.Context,
|
|
clientCtx client.Context,
|
|
tendermintWebsocketClient *rpcclient.WSClient,
|
|
allowUnprotectedTxs bool,
|
|
indexer ethermint.EVMTxIndexer,
|
|
) []rpc.API
|
|
|
|
// apiCreators defines the JSON-RPC API namespaces.
|
|
var apiCreators map[string]APICreator
|
|
|
|
func init() {
|
|
apiCreators = map[string]APICreator{
|
|
EthNamespace: func(ctx *server.Context, clientCtx client.Context, tmWSClient *rpcclient.WSClient, allowUnprotectedTxs bool, indexer ethermint.EVMTxIndexer) []rpc.API {
|
|
evmBackend := backend.NewBackend(ctx, ctx.Logger, clientCtx, allowUnprotectedTxs, indexer)
|
|
return []rpc.API{
|
|
{
|
|
Namespace: EthNamespace,
|
|
Version: apiVersion,
|
|
Service: eth.NewPublicAPI(ctx.Logger, evmBackend),
|
|
Public: true,
|
|
},
|
|
{
|
|
Namespace: EthNamespace,
|
|
Version: apiVersion,
|
|
Service: filters.NewPublicAPI(ctx.Logger, clientCtx, tmWSClient, evmBackend),
|
|
Public: true,
|
|
},
|
|
}
|
|
},
|
|
Web3Namespace: func(*server.Context, client.Context, *rpcclient.WSClient, bool, ethermint.EVMTxIndexer) []rpc.API {
|
|
return []rpc.API{
|
|
{
|
|
Namespace: Web3Namespace,
|
|
Version: apiVersion,
|
|
Service: web3.NewPublicAPI(),
|
|
Public: true,
|
|
},
|
|
}
|
|
},
|
|
NetNamespace: func(_ *server.Context, clientCtx client.Context, _ *rpcclient.WSClient, _ bool, _ ethermint.EVMTxIndexer) []rpc.API {
|
|
return []rpc.API{
|
|
{
|
|
Namespace: NetNamespace,
|
|
Version: apiVersion,
|
|
Service: net.NewPublicAPI(clientCtx),
|
|
Public: true,
|
|
},
|
|
}
|
|
},
|
|
PersonalNamespace: func(ctx *server.Context, clientCtx client.Context, _ *rpcclient.WSClient, allowUnprotectedTxs bool, indexer ethermint.EVMTxIndexer) []rpc.API {
|
|
evmBackend := backend.NewBackend(ctx, ctx.Logger, clientCtx, allowUnprotectedTxs, indexer)
|
|
return []rpc.API{
|
|
{
|
|
Namespace: PersonalNamespace,
|
|
Version: apiVersion,
|
|
Service: personal.NewAPI(ctx.Logger, evmBackend),
|
|
Public: false,
|
|
},
|
|
}
|
|
},
|
|
TxPoolNamespace: func(ctx *server.Context, _ client.Context, _ *rpcclient.WSClient, _ bool, _ ethermint.EVMTxIndexer) []rpc.API {
|
|
return []rpc.API{
|
|
{
|
|
Namespace: TxPoolNamespace,
|
|
Version: apiVersion,
|
|
Service: txpool.NewPublicAPI(ctx.Logger),
|
|
Public: true,
|
|
},
|
|
}
|
|
},
|
|
DebugNamespace: func(ctx *server.Context, clientCtx client.Context, _ *rpcclient.WSClient, allowUnprotectedTxs bool, indexer ethermint.EVMTxIndexer) []rpc.API {
|
|
evmBackend := backend.NewBackend(ctx, ctx.Logger, clientCtx, allowUnprotectedTxs, indexer)
|
|
return []rpc.API{
|
|
{
|
|
Namespace: DebugNamespace,
|
|
Version: apiVersion,
|
|
Service: debug.NewAPI(ctx, evmBackend),
|
|
Public: true,
|
|
},
|
|
}
|
|
},
|
|
MinerNamespace: func(ctx *server.Context, clientCtx client.Context, _ *rpcclient.WSClient, allowUnprotectedTxs bool, indexer ethermint.EVMTxIndexer) []rpc.API {
|
|
evmBackend := backend.NewBackend(ctx, ctx.Logger, clientCtx, allowUnprotectedTxs, indexer)
|
|
return []rpc.API{
|
|
{
|
|
Namespace: MinerNamespace,
|
|
Version: apiVersion,
|
|
Service: miner.NewPrivateAPI(ctx, evmBackend),
|
|
Public: false,
|
|
},
|
|
}
|
|
},
|
|
}
|
|
}
|
|
|
|
// GetRPCAPIs returns the list of all APIs
|
|
func GetRPCAPIs(ctx *server.Context, clientCtx client.Context, tmWSClient *rpcclient.WSClient, allowUnprotectedTxs bool, indexer ethermint.EVMTxIndexer, selectedAPIs []string) []rpc.API {
|
|
var apis []rpc.API
|
|
|
|
for _, ns := range selectedAPIs {
|
|
if creator, ok := apiCreators[ns]; ok {
|
|
apis = append(apis, creator(ctx, clientCtx, tmWSClient, allowUnprotectedTxs, indexer)...)
|
|
} else {
|
|
ctx.Logger.Error("invalid namespace value", "namespace", ns)
|
|
}
|
|
}
|
|
|
|
return apis
|
|
}
|
|
|
|
// RegisterAPINamespace registers a new API namespace with the API creator.
|
|
// This function fails if the namespace is already registered.
|
|
func RegisterAPINamespace(ns string, creator APICreator) error {
|
|
if _, ok := apiCreators[ns]; ok {
|
|
return fmt.Errorf("duplicated api namespace %s", ns)
|
|
}
|
|
apiCreators[ns] = creator
|
|
return nil
|
|
}
|