jeth.go 2.2 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. package rpc
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "github.com/ethereum/go-ethereum/jsre"
  6. "github.com/ethereum/go-ethereum/rpc/comms"
  7. "github.com/ethereum/go-ethereum/rpc/shared"
  8. "github.com/robertkrimen/otto"
  9. )
  10. type Jeth struct {
  11. ethApi shared.EthereumApi
  12. re *jsre.JSRE
  13. client comms.EthereumClient
  14. }
  15. func NewJeth(ethApi shared.EthereumApi, re *jsre.JSRE, client comms.EthereumClient) *Jeth {
  16. return &Jeth{ethApi, re, client}
  17. }
  18. func (self *Jeth) err(call otto.FunctionCall, code int, msg string, id interface{}) (response otto.Value) {
  19. errObj := fmt.Sprintf("{\"message\": \"%s\", \"code\": %d}", msg, code)
  20. retResponse := fmt.Sprintf("ret_response = JSON.parse('{\"jsonrpc\": \"%s\", \"id\": %v, \"error\": %s}');", shared.JsonRpcVersion, id, errObj)
  21. call.Otto.Run("ret_error = " + errObj)
  22. res, _ := call.Otto.Run(retResponse)
  23. return res
  24. }
  25. func (self *Jeth) Send(call otto.FunctionCall) (response otto.Value) {
  26. reqif, err := call.Argument(0).Export()
  27. if err != nil {
  28. return self.err(call, -32700, err.Error(), nil)
  29. }
  30. jsonreq, err := json.Marshal(reqif)
  31. var reqs []shared.Request
  32. batch := true
  33. err = json.Unmarshal(jsonreq, &reqs)
  34. if err != nil {
  35. reqs = make([]shared.Request, 1)
  36. err = json.Unmarshal(jsonreq, &reqs[0])
  37. batch = false
  38. }
  39. call.Otto.Set("response_len", len(reqs))
  40. call.Otto.Run("var ret_response = new Array(response_len);")
  41. for i, req := range reqs {
  42. var respif interface{}
  43. err := self.client.Send(&req)
  44. if err != nil {
  45. return self.err(call, -32603, err.Error(), req.Id)
  46. }
  47. respif, err = self.client.Recv()
  48. if err != nil {
  49. return self.err(call, -32603, err.Error(), req.Id)
  50. }
  51. call.Otto.Set("ret_jsonrpc", shared.JsonRpcVersion)
  52. call.Otto.Set("ret_id", req.Id)
  53. res, _ := json.Marshal(respif)
  54. call.Otto.Set("ret_result", string(res))
  55. call.Otto.Set("response_idx", i)
  56. response, err = call.Otto.Run(`
  57. ret_response[response_idx] = { jsonrpc: ret_jsonrpc, id: ret_id, result: JSON.parse(ret_result) };
  58. `)
  59. }
  60. if !batch {
  61. call.Otto.Run("ret_response = ret_response[0];")
  62. }
  63. if call.Argument(1).IsObject() {
  64. call.Otto.Set("callback", call.Argument(1))
  65. call.Otto.Run(`
  66. if (Object.prototype.toString.call(callback) == '[object Function]') {
  67. callback(null, ret_response);
  68. }
  69. `)
  70. }
  71. return
  72. }