|
|
@ -0,0 +1,126 @@ |
|
|
|
package rpcclient |
|
|
|
|
|
|
|
import ( |
|
|
|
"net/http" |
|
|
|
"strings" |
|
|
|
"time" |
|
|
|
|
|
|
|
"github.com/gorilla/websocket" |
|
|
|
. "github.com/tendermint/go-common" |
|
|
|
"github.com/tendermint/go-wire" |
|
|
|
ctypes "github.com/tendermint/tendermint/rpc/core/types" |
|
|
|
"github.com/tendermint/tendermint/rpc/types" |
|
|
|
) |
|
|
|
|
|
|
|
const ( |
|
|
|
wsEventsChannelCapacity = 10 |
|
|
|
wsResultsChannelCapacity = 10 |
|
|
|
wsWriteTimeoutSeconds = 10 |
|
|
|
) |
|
|
|
|
|
|
|
type WSClient struct { |
|
|
|
QuitService |
|
|
|
Address string |
|
|
|
*websocket.Conn |
|
|
|
EventsCh chan ctypes.ResultEvent // closes upon WSClient.Stop()
|
|
|
|
ResultsCh chan ctypes.Result // closes upon WSClient.Stop()
|
|
|
|
} |
|
|
|
|
|
|
|
// create a new connection
|
|
|
|
func NewWSClient(addr string) *WSClient { |
|
|
|
wsClient := &WSClient{ |
|
|
|
Address: addr, |
|
|
|
Conn: nil, |
|
|
|
EventsCh: make(chan ctypes.ResultEvent, wsEventsChannelCapacity), |
|
|
|
ResultsCh: make(chan ctypes.Result, wsResultsChannelCapacity), |
|
|
|
} |
|
|
|
wsClient.QuitService = *NewQuitService(log, "WSClient", wsClient) |
|
|
|
return wsClient |
|
|
|
} |
|
|
|
|
|
|
|
func (wsc *WSClient) OnStart() error { |
|
|
|
wsc.QuitService.OnStart() |
|
|
|
err := wsc.dial() |
|
|
|
if err != nil { |
|
|
|
return err |
|
|
|
} |
|
|
|
go wsc.receiveEventsRoutine() |
|
|
|
return nil |
|
|
|
} |
|
|
|
|
|
|
|
func (wsc *WSClient) dial() error { |
|
|
|
// Dial
|
|
|
|
dialer := websocket.DefaultDialer |
|
|
|
rHeader := http.Header{} |
|
|
|
con, _, err := dialer.Dial(wsc.Address, rHeader) |
|
|
|
if err != nil { |
|
|
|
return err |
|
|
|
} |
|
|
|
// Set the ping/pong handlers
|
|
|
|
con.SetPingHandler(func(m string) error { |
|
|
|
// NOTE: https://github.com/gorilla/websocket/issues/97
|
|
|
|
go con.WriteControl(websocket.PongMessage, []byte(m), time.Now().Add(time.Second*wsWriteTimeoutSeconds)) |
|
|
|
return nil |
|
|
|
}) |
|
|
|
con.SetPongHandler(func(m string) error { |
|
|
|
// NOTE: https://github.com/gorilla/websocket/issues/97
|
|
|
|
return nil |
|
|
|
}) |
|
|
|
wsc.Conn = con |
|
|
|
return nil |
|
|
|
} |
|
|
|
|
|
|
|
func (wsc *WSClient) OnStop() { |
|
|
|
wsc.QuitService.OnStop() |
|
|
|
// EventsCh and ResultsCh are closed in receiveEventsRoutine.
|
|
|
|
} |
|
|
|
|
|
|
|
func (wsc *WSClient) receiveEventsRoutine() { |
|
|
|
for { |
|
|
|
_, data, err := wsc.ReadMessage() |
|
|
|
if err != nil { |
|
|
|
log.Info("WSClient failed to read message", "error", err, "data", string(data)) |
|
|
|
wsc.Stop() |
|
|
|
break |
|
|
|
} else { |
|
|
|
var response ctypes.Response |
|
|
|
wire.ReadJSON(&response, data, &err) |
|
|
|
if err != nil { |
|
|
|
log.Info("WSClient failed to parse message", "error", err) |
|
|
|
wsc.Stop() |
|
|
|
break |
|
|
|
} |
|
|
|
if strings.HasSuffix(response.ID, "#event") { |
|
|
|
wsc.EventsCh <- *response.Result.(*ctypes.ResultEvent) |
|
|
|
} else { |
|
|
|
wsc.ResultsCh <- response.Result |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
// Cleanup
|
|
|
|
close(wsc.EventsCh) |
|
|
|
close(wsc.ResultsCh) |
|
|
|
} |
|
|
|
|
|
|
|
// subscribe to an event
|
|
|
|
func (wsc *WSClient) Subscribe(eventid string) error { |
|
|
|
err := wsc.WriteJSON(rpctypes.RPCRequest{ |
|
|
|
JSONRPC: "2.0", |
|
|
|
ID: "", |
|
|
|
Method: "subscribe", |
|
|
|
Params: []interface{}{eventid}, |
|
|
|
}) |
|
|
|
return err |
|
|
|
} |
|
|
|
|
|
|
|
// unsubscribe from an event
|
|
|
|
func (wsc *WSClient) Unsubscribe(eventid string) error { |
|
|
|
err := wsc.WriteJSON(rpctypes.RPCRequest{ |
|
|
|
JSONRPC: "2.0", |
|
|
|
ID: "", |
|
|
|
Method: "unsubscribe", |
|
|
|
Params: []interface{}{eventid}, |
|
|
|
}) |
|
|
|
return err |
|
|
|
} |