tendermint/rpc/test/helpers.go

160 lines
3.9 KiB
Go
Raw Normal View History

2016-01-10 13:33:52 -08:00
package rpctest
import (
"testing"
"time"
. "github.com/tendermint/go-common"
2016-05-08 15:00:58 -07:00
cfg "github.com/tendermint/go-config"
2016-01-10 13:33:52 -08:00
"github.com/tendermint/go-p2p"
"github.com/tendermint/go-wire"
2016-01-12 13:54:27 -08:00
client "github.com/tendermint/go-rpc/client"
2016-05-08 15:00:58 -07:00
"github.com/tendermint/tendermint/config/tendermint_test"
2016-01-10 13:33:52 -08:00
nm "github.com/tendermint/tendermint/node"
2016-01-13 15:38:55 -08:00
ctypes "github.com/tendermint/tendermint/rpc/core/types"
2016-01-10 13:33:52 -08:00
)
// global variables for use across all tests
var (
2016-05-08 15:00:58 -07:00
config cfg.Config
node *nm.Node
chainID string
rpcAddr string
requestAddr string
websocketAddr string
websocketEndpoint string
clientURI *client.ClientURI
clientJSON *client.ClientJSONRPC
2016-01-10 13:33:52 -08:00
)
2016-01-12 13:54:27 -08:00
// initialize config and create new node
2016-05-08 15:00:58 -07:00
func init() {
config = tendermint_test.ResetConfig("rpc_test_client_test")
2016-01-12 13:54:27 -08:00
chainID = config.GetString("chain_id")
rpcAddr = config.GetString("rpc_laddr")
2016-02-18 18:06:11 -08:00
requestAddr = rpcAddr
websocketAddr = rpcAddr
websocketEndpoint = "/websocket"
2016-01-12 13:54:27 -08:00
clientURI = client.NewClientURI(requestAddr)
clientJSON = client.NewClientJSONRPC(requestAddr)
// TODO: change consensus/state.go timeouts to be shorter
// start a node
ready := make(chan struct{})
go newNode(ready)
<-ready
}
2016-01-10 13:33:52 -08:00
// create a new node and sleep forever
func newNode(ready chan struct{}) {
// Create & start node
node = nm.NewNodeDefault(config)
protocol, address := nm.ProtocolAndAddress(config.GetString("node_laddr"))
l := p2p.NewDefaultListener(protocol, address, true)
2016-01-10 13:33:52 -08:00
node.AddListener(l)
node.Start()
// Run the RPC server.
node.StartRPC()
ready <- struct{}{}
// Sleep forever
ch := make(chan struct{})
<-ch
}
//--------------------------------------------------------------------------------
// Utilities for testing the websocket service
// create a new connection
2016-01-13 18:20:25 -08:00
func newWSClient(t *testing.T) *client.WSClient {
2016-02-18 18:06:11 -08:00
wsc := client.NewWSClient(websocketAddr, websocketEndpoint)
2016-01-13 18:20:25 -08:00
if _, err := wsc.Start(); err != nil {
panic(err)
2016-01-10 13:33:52 -08:00
}
2016-01-13 18:20:25 -08:00
return wsc
2016-01-10 13:33:52 -08:00
}
// subscribe to an event
2016-01-13 18:20:25 -08:00
func subscribe(t *testing.T, wsc *client.WSClient, eventid string) {
if err := wsc.Subscribe(eventid); err != nil {
panic(err)
2016-01-10 13:33:52 -08:00
}
}
// unsubscribe from an event
2016-01-13 18:20:25 -08:00
func unsubscribe(t *testing.T, wsc *client.WSClient, eventid string) {
if err := wsc.Unsubscribe(eventid); err != nil {
panic(err)
2016-01-10 13:33:52 -08:00
}
}
// wait for an event; do things that might trigger events, and check them when they are received
// the check function takes an event id and the byte slice read off the ws
2016-01-13 18:20:25 -08:00
func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, dieOnTimeout bool, f func(), check func(string, interface{}) error) {
2016-01-10 13:33:52 -08:00
// go routine to wait for webscoket msg
2016-01-13 18:20:25 -08:00
goodCh := make(chan interface{})
2016-01-10 13:33:52 -08:00
errCh := make(chan error)
// Read message
go func() {
2016-01-13 18:20:25 -08:00
var err error
LOOP:
2016-01-10 13:33:52 -08:00
for {
2016-01-13 18:20:25 -08:00
select {
case r := <-wsc.ResultsCh:
2016-01-13 15:38:55 -08:00
result := new(ctypes.TMResult)
2016-01-13 18:20:25 -08:00
wire.ReadJSONPtr(result, r, &err)
2016-01-10 13:33:52 -08:00
if err != nil {
errCh <- err
2016-01-13 18:20:25 -08:00
break LOOP
2016-01-10 13:33:52 -08:00
}
2016-01-13 15:38:55 -08:00
event, ok := (*result).(*ctypes.ResultEvent)
if ok && event.Name == eventid {
2016-01-13 18:20:25 -08:00
goodCh <- event.Data
break LOOP
2016-01-10 13:33:52 -08:00
}
2016-01-13 18:20:25 -08:00
case err := <-wsc.ErrorsCh:
errCh <- err
break LOOP
case <-wsc.Quit:
break LOOP
2016-01-10 13:33:52 -08:00
}
}
}()
// do stuff (transactions)
f()
// wait for an event or timeout
timeout := time.NewTimer(10 * time.Second)
select {
case <-timeout.C:
if dieOnTimeout {
2016-01-13 18:20:25 -08:00
wsc.Stop()
panic(Fmt("%s event was not received in time", eventid))
2016-01-10 13:33:52 -08:00
}
// else that's great, we didn't hear the event
// and we shouldn't have
2016-01-13 18:20:25 -08:00
case eventData := <-goodCh:
2016-01-10 13:33:52 -08:00
if dieOnTimeout {
// message was received and expected
// run the check
2016-01-13 18:20:25 -08:00
if err := check(eventid, eventData); err != nil {
panic(err) // Show the stack trace.
2016-01-10 13:33:52 -08:00
}
} else {
2016-01-13 18:20:25 -08:00
wsc.Stop()
panic(Fmt("%s event was not expected", eventid))
2016-01-10 13:33:52 -08:00
}
case err := <-errCh:
panic(err) // Show the stack trace.
}
}
2016-01-13 18:20:25 -08:00
//--------------------------------------------------------------------------------