|
|
|
package node
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/event"
|
|
|
|
"github.com/ethereum/go-ethereum/rpc"
|
|
|
|
"github.com/harmony-one/harmony/core/types"
|
|
|
|
"github.com/harmony-one/harmony/hmy"
|
|
|
|
nodeconfig "github.com/harmony-one/harmony/internal/configs/node"
|
|
|
|
"github.com/harmony-one/harmony/internal/hmyapi"
|
|
|
|
"github.com/harmony-one/harmony/internal/hmyapi/apiv1"
|
|
|
|
"github.com/harmony-one/harmony/internal/hmyapi/apiv2"
|
|
|
|
"github.com/harmony-one/harmony/internal/hmyapi/filters"
|
|
|
|
"github.com/harmony-one/harmony/internal/utils"
|
|
|
|
staking "github.com/harmony-one/harmony/staking/types"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
rpcHTTPPortOffset = 500
|
|
|
|
rpcWSPortOffset = 800
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// HTTP RPC
|
|
|
|
httpListener net.Listener
|
|
|
|
httpHandler *rpc.Server
|
|
|
|
httpEndpoint = ""
|
|
|
|
wsEndpoint = ""
|
|
|
|
httpModules = []string{"hmy", "hmyv2", "net", "netv2", "explorer"}
|
|
|
|
httpVirtualHosts = []string{"*"}
|
|
|
|
httpTimeouts = rpc.DefaultHTTPTimeouts
|
|
|
|
httpOrigins = []string{"*"}
|
|
|
|
wsModules = []string{"hmy", "hmyv2", "net", "netv2", "web3"}
|
|
|
|
wsOrigins = []string{"*"}
|
|
|
|
harmony *hmy.Harmony
|
|
|
|
)
|
|
|
|
|
|
|
|
// IsCurrentlyLeader exposes if node is currently the leader node
|
|
|
|
func (node *Node) IsCurrentlyLeader() bool {
|
|
|
|
return node.Consensus.IsLeader()
|
|
|
|
}
|
|
|
|
|
|
|
|
// PendingCXReceipts returns node.pendingCXReceiptsProof
|
|
|
|
func (node *Node) PendingCXReceipts() []*types.CXReceiptsProof {
|
|
|
|
cxReceipts := make([]*types.CXReceiptsProof, len(node.pendingCXReceipts))
|
|
|
|
i := 0
|
|
|
|
for _, cxReceipt := range node.pendingCXReceipts {
|
|
|
|
cxReceipts[i] = cxReceipt
|
|
|
|
i++
|
|
|
|
}
|
|
|
|
return cxReceipts
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErroredStakingTransactionSink is the inmemory failed staking transactions this node has
|
|
|
|
func (node *Node) ErroredStakingTransactionSink() []staking.RPCTransactionError {
|
|
|
|
node.errorSink.Lock()
|
|
|
|
defer node.errorSink.Unlock()
|
|
|
|
result := []staking.RPCTransactionError{}
|
|
|
|
node.errorSink.failedStakingTxns.Do(func(d interface{}) {
|
|
|
|
if d != nil {
|
|
|
|
result = append(result, d.(staking.RPCTransactionError))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErroredTransactionSink is the inmemory failed transactions this node has
|
|
|
|
func (node *Node) ErroredTransactionSink() []types.RPCTransactionError {
|
|
|
|
node.errorSink.Lock()
|
|
|
|
defer node.errorSink.Unlock()
|
|
|
|
result := []types.RPCTransactionError{}
|
|
|
|
node.errorSink.failedTxns.Do(func(d interface{}) {
|
|
|
|
if d != nil {
|
|
|
|
result = append(result, d.(types.RPCTransactionError))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
// StartRPC start RPC service
|
|
|
|
func (node *Node) StartRPC(nodePort string) error {
|
|
|
|
// Gather all the possible APIs to surface
|
|
|
|
harmony, _ = hmy.New(
|
|
|
|
node, node.TxPool, node.CxPool, new(event.TypeMux), node.Consensus.ShardID,
|
|
|
|
)
|
|
|
|
|
|
|
|
apis := node.APIs()
|
|
|
|
|
|
|
|
for _, service := range node.serviceManager.GetServices() {
|
|
|
|
apis = append(apis, service.APIs()...)
|
|
|
|
}
|
|
|
|
|
|
|
|
port, _ := strconv.Atoi(nodePort)
|
|
|
|
|
|
|
|
ip := ""
|
|
|
|
if !nodeconfig.GetPublicRPC() {
|
|
|
|
ip = "127.0.0.1"
|
|
|
|
}
|
|
|
|
httpEndpoint = fmt.Sprintf("%v:%v", ip, port+rpcHTTPPortOffset)
|
|
|
|
|
|
|
|
if err := node.startHTTP(httpEndpoint, apis, httpModules, httpOrigins, httpVirtualHosts, httpTimeouts); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
wsEndpoint = fmt.Sprintf("%v:%v", ip, port+rpcWSPortOffset)
|
|
|
|
if err := node.startWS(wsEndpoint, apis, wsModules, wsOrigins, true); err != nil {
|
|
|
|
node.stopHTTP()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// startHTTP initializes and starts the HTTP RPC endpoint.
|
|
|
|
func (node *Node) startHTTP(endpoint string, apis []rpc.API, modules []string, cors []string, vhosts []string, timeouts rpc.HTTPTimeouts) error {
|
|
|
|
// Short circuit if the HTTP endpoint isn't being exposed
|
|
|
|
if endpoint == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
listener, handler, err := rpc.StartHTTPEndpoint(endpoint, apis, modules, cors, vhosts, timeouts)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
utils.Logger().Info().
|
|
|
|
Str("url", fmt.Sprintf("http://%s", endpoint)).
|
|
|
|
Str("cors", strings.Join(cors, ",")).
|
|
|
|
Str("vhosts", strings.Join(vhosts, ",")).
|
|
|
|
Msg("HTTP endpoint opened")
|
|
|
|
// All listeners booted successfully
|
|
|
|
httpListener = listener
|
|
|
|
httpHandler = handler
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// stopHTTP terminates the HTTP RPC endpoint.
|
|
|
|
func (node *Node) stopHTTP() {
|
|
|
|
if httpListener != nil {
|
|
|
|
httpListener.Close()
|
|
|
|
httpListener = nil
|
|
|
|
utils.Logger().Info().Str("url", fmt.Sprintf("http://%s", httpEndpoint)).Msg("HTTP endpoint closed")
|
|
|
|
}
|
|
|
|
if httpHandler != nil {
|
|
|
|
httpHandler.Stop()
|
|
|
|
httpHandler = nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// startWS initializes and starts the websocket RPC endpoint.
|
|
|
|
func (node *Node) startWS(
|
|
|
|
endpoint string, apis []rpc.API, modules []string, wsOrigins []string, exposeAll bool,
|
|
|
|
) error {
|
|
|
|
// Short circuit if the WS endpoint isn't being exposed
|
|
|
|
if endpoint == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
listener, _, err := rpc.StartWSEndpoint(endpoint, apis, modules, wsOrigins, exposeAll)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
utils.Logger().Info().
|
|
|
|
Str("url", fmt.Sprintf("ws://%s", listener.Addr())).
|
|
|
|
Msg("WebSocket endpoint opened")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// APIs return the collection of RPC services the ethereum package offers.
|
|
|
|
// NOTE, some of these services probably need to be moved to somewhere else.
|
|
|
|
func (node *Node) APIs() []rpc.API {
|
|
|
|
// Gather all the possible APIs to surface
|
|
|
|
apis := hmyapi.GetAPIs(harmony.APIBackend)
|
|
|
|
// Append all the local APIs and return
|
|
|
|
return append(apis, []rpc.API{
|
|
|
|
{
|
|
|
|
Namespace: "hmy",
|
|
|
|
Version: "1.0",
|
|
|
|
Service: filters.NewPublicFilterAPI(harmony.APIBackend, false),
|
|
|
|
Public: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Namespace: "net",
|
|
|
|
Version: "1.0",
|
|
|
|
Service: apiv1.NewPublicNetAPI(node.host, harmony.APIBackend.NetVersion()),
|
|
|
|
Public: true,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Namespace: "netv2",
|
|
|
|
Version: "1.0",
|
|
|
|
Service: apiv2.NewPublicNetAPI(node.host, harmony.APIBackend.NetVersion()),
|
|
|
|
Public: true,
|
|
|
|
},
|
|
|
|
}...)
|
|
|
|
}
|