websockets.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. /*
  2. This file is part of go-ethereum
  3. go-ethereum is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 3 of the License, or
  6. (at your option) any later version.
  7. go-ethereum is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.
  13. */
  14. /**
  15. * @authors
  16. * Jeffrey Wilcke <i@jev.io>
  17. */
  18. package utils
  19. import (
  20. "github.com/ethereum/go-ethereum/core"
  21. "github.com/ethereum/go-ethereum/core/types"
  22. "github.com/ethereum/go-ethereum/eth"
  23. "github.com/ethereum/go-ethereum/ethutil"
  24. "github.com/ethereum/go-ethereum/event/filter"
  25. "github.com/ethereum/go-ethereum/logger"
  26. "github.com/ethereum/go-ethereum/state"
  27. "github.com/ethereum/go-ethereum/ui/qt"
  28. "github.com/ethereum/go-ethereum/websocket"
  29. "github.com/ethereum/go-ethereum/xeth"
  30. )
  31. var wslogger = logger.NewLogger("WS")
  32. func args(v ...interface{}) []interface{} {
  33. return v
  34. }
  35. type WebSocketServer struct {
  36. eth *eth.Ethereum
  37. filterManager *filter.FilterManager
  38. }
  39. func NewWebSocketServer(eth *eth.Ethereum) *WebSocketServer {
  40. filterManager := filter.NewFilterManager(eth.EventMux())
  41. go filterManager.Start()
  42. return &WebSocketServer{eth, filterManager}
  43. }
  44. func (self *WebSocketServer) Serv() {
  45. pipe := xeth.NewJSXEth(self.eth)
  46. wsServ := websocket.NewServer("/eth", ":40404")
  47. wsServ.MessageFunc(func(c *websocket.Client, msg *websocket.Message) {
  48. switch msg.Call {
  49. case "compile":
  50. data := ethutil.NewValue(msg.Args)
  51. bcode, err := ethutil.Compile(data.Get(0).Str(), false)
  52. if err != nil {
  53. c.Write(args(nil, err.Error()), msg.Id)
  54. }
  55. code := ethutil.Bytes2Hex(bcode)
  56. c.Write(args(code, nil), msg.Id)
  57. case "eth_blockByNumber":
  58. args := msg.Arguments()
  59. block := pipe.BlockByNumber(int32(args.Get(0).Uint()))
  60. c.Write(block, msg.Id)
  61. case "eth_blockByHash":
  62. args := msg.Arguments()
  63. c.Write(pipe.BlockByHash(args.Get(0).Str()), msg.Id)
  64. case "eth_transact":
  65. if mp, ok := msg.Args[0].(map[string]interface{}); ok {
  66. object := mapToTxParams(mp)
  67. c.Write(
  68. args(pipe.Transact(pipe.Key().PrivateKey, object["to"], object["value"], object["gas"], object["gasPrice"], object["data"])),
  69. msg.Id,
  70. )
  71. }
  72. case "eth_gasPrice":
  73. c.Write("10000000000000", msg.Id)
  74. case "eth_coinbase":
  75. c.Write(pipe.CoinBase(), msg.Id)
  76. case "eth_listening":
  77. c.Write(pipe.IsListening(), msg.Id)
  78. case "eth_mining":
  79. c.Write(pipe.IsMining(), msg.Id)
  80. case "eth_peerCount":
  81. c.Write(pipe.PeerCount(), msg.Id)
  82. case "eth_countAt":
  83. args := msg.Arguments()
  84. c.Write(pipe.TxCountAt(args.Get(0).Str()), msg.Id)
  85. case "eth_codeAt":
  86. args := msg.Arguments()
  87. c.Write(len(pipe.CodeAt(args.Get(0).Str())), msg.Id)
  88. case "eth_storageAt":
  89. args := msg.Arguments()
  90. c.Write(pipe.StorageAt(args.Get(0).Str(), args.Get(1).Str()), msg.Id)
  91. case "eth_balanceAt":
  92. args := msg.Arguments()
  93. c.Write(pipe.BalanceAt(args.Get(0).Str()), msg.Id)
  94. case "eth_accounts":
  95. c.Write(pipe.Accounts(), msg.Id)
  96. case "eth_newFilter":
  97. if mp, ok := msg.Args[0].(map[string]interface{}); ok {
  98. var id int
  99. filter := qt.NewFilterFromMap(mp, self.eth)
  100. filter.MessageCallback = func(messages state.Messages) {
  101. c.Event(toMessages(messages), "eth_changed", id)
  102. }
  103. id = self.filterManager.InstallFilter(filter)
  104. c.Write(id, msg.Id)
  105. }
  106. case "eth_newFilterString":
  107. var id int
  108. filter := core.NewFilter(self.eth)
  109. filter.BlockCallback = func(block *types.Block) {
  110. c.Event(nil, "eth_changed", id)
  111. }
  112. id = self.filterManager.InstallFilter(filter)
  113. c.Write(id, msg.Id)
  114. case "eth_filterLogs":
  115. filter := self.filterManager.GetFilter(int(msg.Arguments().Get(0).Uint()))
  116. if filter != nil {
  117. c.Write(toMessages(filter.Find()), msg.Id)
  118. }
  119. }
  120. })
  121. wsServ.Listen()
  122. }
  123. func toMessages(messages state.Messages) (msgs []xeth.JSMessage) {
  124. msgs = make([]xeth.JSMessage, len(messages))
  125. for i, msg := range messages {
  126. msgs[i] = xeth.NewJSMessage(msg)
  127. }
  128. return
  129. }
  130. func StartWebSockets(eth *eth.Ethereum) {
  131. wslogger.Infoln("Starting WebSockets")
  132. sock := NewWebSocketServer(eth)
  133. go sock.Serv()
  134. }
  135. // TODO This is starting to become a generic method. Move to utils
  136. func mapToTxParams(object map[string]interface{}) map[string]string {
  137. // Default values
  138. if object["from"] == nil {
  139. object["from"] = ""
  140. }
  141. if object["to"] == nil {
  142. object["to"] = ""
  143. }
  144. if object["value"] == nil {
  145. object["value"] = ""
  146. }
  147. if object["gas"] == nil {
  148. object["gas"] = ""
  149. }
  150. if object["gasPrice"] == nil {
  151. object["gasPrice"] = ""
  152. }
  153. var dataStr string
  154. var data []string
  155. if str, ok := object["data"].(string); ok {
  156. data = []string{str}
  157. }
  158. for _, str := range data {
  159. if ethutil.IsHex(str) {
  160. str = str[2:]
  161. if len(str) != 64 {
  162. str = ethutil.LeftPadString(str, 64)
  163. }
  164. } else {
  165. str = ethutil.Bytes2Hex(ethutil.LeftPadBytes(ethutil.Big(str).Bytes(), 32))
  166. }
  167. dataStr += str
  168. }
  169. object["data"] = dataStr
  170. conv := make(map[string]string)
  171. for key, value := range object {
  172. if v, ok := value.(string); ok {
  173. conv[key] = v
  174. }
  175. }
  176. return conv
  177. }