2017-08-15 11:27:12 +01:00
|
|
|
package jail
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
2017-09-14 22:14:31 +02:00
|
|
|
gethrpc "github.com/ethereum/go-ethereum/rpc"
|
2017-08-15 11:27:12 +01:00
|
|
|
"github.com/status-im/status-go/geth/common"
|
2017-09-18 15:13:32 +03:00
|
|
|
"github.com/status-im/status-go/geth/jail/internal/vm"
|
2017-09-04 14:56:58 +02:00
|
|
|
"github.com/status-im/status-go/geth/params"
|
2017-09-14 22:14:31 +02:00
|
|
|
"github.com/status-im/status-go/geth/rpc"
|
2017-09-07 08:49:40 +01:00
|
|
|
)
|
|
|
|
|
2017-08-15 11:27:12 +01:00
|
|
|
// ExecutionPolicy provides a central container for the executions of RPCCall requests for both
|
|
|
|
// remote/upstream processing and internal node processing.
|
|
|
|
type ExecutionPolicy struct {
|
|
|
|
nodeManager common.NodeManager
|
|
|
|
accountManager common.AccountManager
|
2017-09-04 14:56:58 +02:00
|
|
|
txQueueManager common.TxQueueManager
|
2017-08-15 11:27:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewExecutionPolicy returns a new instance of ExecutionPolicy.
|
2017-09-04 14:56:58 +02:00
|
|
|
func NewExecutionPolicy(
|
|
|
|
nodeManager common.NodeManager, accountManager common.AccountManager, txQueueManager common.TxQueueManager,
|
|
|
|
) *ExecutionPolicy {
|
2017-08-15 11:27:12 +01:00
|
|
|
return &ExecutionPolicy{
|
|
|
|
nodeManager: nodeManager,
|
|
|
|
accountManager: accountManager,
|
2017-09-04 14:56:58 +02:00
|
|
|
txQueueManager: txQueueManager,
|
2017-08-15 11:27:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-07 08:49:40 +01:00
|
|
|
// Execute handles the execution of a RPC request and routes appropriately to either a local or remote ethereum node.
|
2017-09-18 15:13:32 +03:00
|
|
|
func (ep *ExecutionPolicy) Execute(req common.RPCCall, vm *vm.VM) (map[string]interface{}, error) {
|
2017-09-14 22:14:31 +02:00
|
|
|
client := ep.nodeManager.RPCClient()
|
2017-09-18 15:13:32 +03:00
|
|
|
return ep.executeWithClient(client, vm, req)
|
2017-08-15 11:27:12 +01:00
|
|
|
}
|
|
|
|
|
2017-09-18 15:13:32 +03:00
|
|
|
func (ep *ExecutionPolicy) executeWithClient(client *rpc.Client, vm *vm.VM, req common.RPCCall) (map[string]interface{}, error) {
|
|
|
|
// Arbitrary JSON-RPC response.
|
|
|
|
var result interface{}
|
2017-08-15 11:27:12 +01:00
|
|
|
|
2017-09-18 15:13:32 +03:00
|
|
|
resp := map[string]interface{}{
|
|
|
|
"jsonrpc": "2.0",
|
|
|
|
"id": req.ID,
|
|
|
|
}
|
2017-08-15 11:27:12 +01:00
|
|
|
|
|
|
|
// do extra request pre processing (persist message id)
|
|
|
|
// within function semaphore will be acquired and released,
|
|
|
|
// so that no more than one client (per cell) can enter
|
2017-09-18 15:13:32 +03:00
|
|
|
messageID, err := preProcessRequest(vm)
|
2017-08-15 11:27:12 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, common.StopRPCCallError{Err: err}
|
|
|
|
}
|
|
|
|
|
2017-09-15 12:35:31 +02:00
|
|
|
if client == nil {
|
2017-09-18 15:13:32 +03:00
|
|
|
resp = newErrorResponse("RPC client is not available. Node is stopped?", &req.ID)
|
2017-09-15 12:35:31 +02:00
|
|
|
} else {
|
2017-09-25 19:04:40 +03:00
|
|
|
// TODO(adam): check if context is used
|
|
|
|
ctx := context.WithValue(context.Background(), common.MessageIDKey, messageID)
|
|
|
|
err = client.CallContext(ctx, &result, req.Method, req.Params...)
|
2017-09-15 12:35:31 +02:00
|
|
|
if err != nil {
|
|
|
|
if err2, ok := err.(gethrpc.Error); ok {
|
2017-09-18 15:13:32 +03:00
|
|
|
resp["error"] = map[string]interface{}{
|
2017-09-15 12:35:31 +02:00
|
|
|
"code": err2.ErrorCode(),
|
|
|
|
"message": err2.Error(),
|
2017-09-18 15:13:32 +03:00
|
|
|
}
|
2017-09-15 12:35:31 +02:00
|
|
|
} else {
|
2017-09-18 15:13:32 +03:00
|
|
|
resp = newErrorResponse(err.Error(), &req.ID)
|
2017-09-15 12:35:31 +02:00
|
|
|
}
|
2017-08-15 11:27:12 +01:00
|
|
|
}
|
2017-09-14 22:14:31 +02:00
|
|
|
}
|
2017-08-15 11:27:12 +01:00
|
|
|
|
2017-09-14 22:14:31 +02:00
|
|
|
if result == nil {
|
|
|
|
// Special case null because it is decoded as an empty
|
|
|
|
// raw message for some reason.
|
2017-09-18 15:13:32 +03:00
|
|
|
resp["result"] = ""
|
2017-09-14 22:14:31 +02:00
|
|
|
} else {
|
2017-09-18 15:13:32 +03:00
|
|
|
resp["result"] = result
|
2017-08-15 11:27:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// do extra request post processing (setting back tx context)
|
2017-09-18 15:13:32 +03:00
|
|
|
postProcessRequest(vm, req, messageID)
|
2017-08-15 11:27:12 +01:00
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
2017-09-04 14:56:58 +02:00
|
|
|
|
|
|
|
// preProcessRequest pre-processes a given RPC call to a given Otto VM
|
2017-09-18 15:13:32 +03:00
|
|
|
func preProcessRequest(vm *vm.VM) (string, error) {
|
|
|
|
messageID := currentMessageID(vm)
|
2017-09-04 14:56:58 +02:00
|
|
|
|
|
|
|
return messageID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// postProcessRequest post-processes a given RPC call to a given Otto VM
|
2017-09-18 15:13:32 +03:00
|
|
|
func postProcessRequest(vm *vm.VM, req common.RPCCall, messageID string) {
|
2017-09-04 14:56:58 +02:00
|
|
|
if len(messageID) > 0 {
|
|
|
|
vm.Call("addContext", nil, messageID, common.MessageIDKey, messageID) // nolint: errcheck
|
|
|
|
}
|
|
|
|
|
|
|
|
// set extra markers for queued transaction requests
|
|
|
|
if req.Method == params.SendTransactionMethodName {
|
|
|
|
vm.Call("addContext", nil, messageID, params.SendTransactionMethodName, true) // nolint: errcheck
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// currentMessageID looks for `status.message_id` variable in current JS context
|
2017-09-18 15:13:32 +03:00
|
|
|
func currentMessageID(vm *vm.VM) string {
|
|
|
|
msgID, err := vm.Run("status.message_id")
|
|
|
|
if err != nil {
|
|
|
|
return ""
|
2017-09-04 14:56:58 +02:00
|
|
|
}
|
2017-09-18 15:13:32 +03:00
|
|
|
return msgID.String()
|
2017-09-04 14:56:58 +02:00
|
|
|
}
|