2017-05-28 16:57:30 +03:00
|
|
|
package node
|
|
|
|
|
|
|
|
import (
|
2017-08-15 11:27:12 +01:00
|
|
|
"bytes"
|
2017-05-28 16:57:30 +03:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2017-08-15 11:27:12 +01:00
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
2017-05-28 16:57:30 +03:00
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
2017-08-15 11:27:12 +01:00
|
|
|
"net/url"
|
|
|
|
|
2017-05-28 16:57:30 +03:00
|
|
|
"github.com/ethereum/go-ethereum/les/status"
|
|
|
|
"github.com/status-im/status-go/geth/common"
|
2017-08-10 15:35:58 +02:00
|
|
|
"github.com/status-im/status-go/geth/log"
|
2017-08-15 11:27:12 +01:00
|
|
|
"github.com/status-im/status-go/geth/params"
|
2017-05-28 16:57:30 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2017-08-10 15:35:58 +02:00
|
|
|
jsonrpcVersion = "2.0"
|
2017-05-28 16:57:30 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
type jsonRequest struct {
|
|
|
|
Method string `json:"method"`
|
|
|
|
Version string `json:"jsonrpc"`
|
|
|
|
ID int `json:"id,omitempty"`
|
|
|
|
Payload json.RawMessage `json:"params,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type jsonError struct {
|
|
|
|
Code int `json:"code"`
|
|
|
|
Message string `json:"message"`
|
|
|
|
Data interface{} `json:"data,omitempty"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type jsonErrResponse struct {
|
|
|
|
Version string `json:"jsonrpc"`
|
|
|
|
ID interface{} `json:"id,omitempty"`
|
|
|
|
Error jsonError `json:"error"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// RPCManager abstract RPC management API (for both client and server)
|
|
|
|
type RPCManager struct {
|
|
|
|
sync.Mutex
|
|
|
|
requestID int
|
|
|
|
nodeManager common.NodeManager
|
|
|
|
}
|
|
|
|
|
|
|
|
// errors
|
|
|
|
var (
|
|
|
|
ErrInvalidMethod = errors.New("method does not exist")
|
|
|
|
ErrRPCServerTimeout = errors.New("RPC server cancelled call due to timeout")
|
|
|
|
ErrRPCServerCallFailed = errors.New("RPC server cannot complete request")
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewRPCManager returns new instance of RPC client
|
|
|
|
func NewRPCManager(nodeManager common.NodeManager) *RPCManager {
|
|
|
|
return &RPCManager{
|
|
|
|
nodeManager: nodeManager,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Call executes RPC request on node's in-proc RPC server
|
|
|
|
func (c *RPCManager) Call(inputJSON string) string {
|
2017-08-15 11:27:12 +01:00
|
|
|
config, err := c.nodeManager.NodeConfig()
|
2017-05-28 16:57:30 +03:00
|
|
|
if err != nil {
|
|
|
|
return c.makeJSONErrorResponse(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// allow HTTP requests to block w/o
|
|
|
|
outputJSON := make(chan string, 1)
|
2017-08-15 11:27:12 +01:00
|
|
|
|
2017-05-28 16:57:30 +03:00
|
|
|
go func() {
|
2017-08-15 11:27:12 +01:00
|
|
|
body := bytes.NewBufferString(inputJSON)
|
|
|
|
|
|
|
|
var err error
|
|
|
|
var res []byte
|
|
|
|
|
|
|
|
if config.UpstreamConfig.Enabled {
|
|
|
|
log.Info("Making RPC JSON Request to upstream RPCServer")
|
|
|
|
res, err = c.callUpstreamStream(config, body)
|
|
|
|
} else {
|
|
|
|
log.Info("Making RPC JSON Request to internal RPCServer")
|
|
|
|
res, err = c.callNodeStream(body)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
outputJSON <- c.makeJSONErrorResponse(err)
|
2017-05-28 16:57:30 +03:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-08-15 11:27:12 +01:00
|
|
|
outputJSON <- string(res)
|
|
|
|
return
|
2017-05-28 16:57:30 +03:00
|
|
|
}()
|
|
|
|
|
|
|
|
// wait till call is complete
|
|
|
|
select {
|
|
|
|
case out := <-outputJSON:
|
|
|
|
return out
|
|
|
|
case <-time.After((status.DefaultTxSendCompletionTimeout + 10) * time.Minute): // give up eventually
|
|
|
|
// pass
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.makeJSONErrorResponse(ErrRPCServerTimeout)
|
|
|
|
}
|
|
|
|
|
2017-08-15 11:27:12 +01:00
|
|
|
// callNodeStream delivers giving request and body content to the external ethereum
|
|
|
|
// (infura) RPCServer to process the request and returns response.
|
|
|
|
func (c *RPCManager) callUpstreamStream(config *params.NodeConfig, body io.Reader) ([]byte, error) {
|
|
|
|
upstreamURL, err := url.Parse(config.UpstreamConfig.URL)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequest("POST", upstreamURL.String(), body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
httpClient := http.Client{
|
|
|
|
Timeout: 20 * time.Second,
|
|
|
|
}
|
|
|
|
|
|
|
|
res, err := httpClient.Do(httpReq)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
defer res.Body.Close()
|
|
|
|
|
|
|
|
if respStatusCode := res.StatusCode; respStatusCode != http.StatusOK {
|
|
|
|
log.Error("handler returned wrong status code", "got", respStatusCode, "want", http.StatusOK)
|
|
|
|
return nil, ErrRPCServerCallFailed
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.ReadAll(res.Body)
|
|
|
|
}
|
|
|
|
|
|
|
|
// callNodeStream delivers giving request and body content to the internal ethereum
|
|
|
|
// RPCServer to process the request.
|
|
|
|
func (c *RPCManager) callNodeStream(body io.Reader) ([]byte, error) {
|
|
|
|
server, err := c.nodeManager.RPCServer()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
httpReq, err := http.NewRequest("POST", "/", body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
rr := httptest.NewRecorder()
|
|
|
|
|
|
|
|
server.ServeHTTP(rr, httpReq)
|
|
|
|
|
|
|
|
// Check the status code is what we expect.
|
|
|
|
if respStatus := rr.Code; respStatus != http.StatusOK {
|
|
|
|
log.Error("handler returned wrong status code", "got", respStatus, "want", http.StatusOK)
|
|
|
|
// outputJSON <- c.makeJSONErrorResponse(ErrRPCServerCallFailed)
|
|
|
|
return nil, ErrRPCServerCallFailed
|
|
|
|
}
|
|
|
|
|
|
|
|
return rr.Body.Bytes(), nil
|
|
|
|
}
|
|
|
|
|
2017-05-28 16:57:30 +03:00
|
|
|
// makeJSONErrorResponse returns error as JSON response
|
|
|
|
func (c *RPCManager) makeJSONErrorResponse(err error) string {
|
|
|
|
response := jsonErrResponse{
|
|
|
|
Version: jsonrpcVersion,
|
|
|
|
Error: jsonError{
|
|
|
|
Message: err.Error(),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
outBytes, _ := json.Marshal(&response)
|
|
|
|
return string(outBytes)
|
|
|
|
}
|