tendermint/rpc/test/helpers.go

164 lines
4.1 KiB
Go
Raw Normal View History

2016-01-10 16:33:52 -05: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 16:33:52 -05:00
"github.com/tendermint/go-p2p"
"github.com/tendermint/go-wire"
2016-01-12 16:54:27 -05: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 16:33:52 -05:00
nm "github.com/tendermint/tendermint/node"
2016-01-13 18:38:55 -05:00
ctypes "github.com/tendermint/tendermint/rpc/core/types"
2016-01-10 16:33:52 -05:00
"github.com/tendermint/tendermint/types"
)
// global variables for use across all tests
var (
2016-05-08 15:00:58 -07:00
config cfg.Config
node *nm.Node
mempoolCount = 0
chainID string
rpcAddr string
requestAddr string
websocketAddr string
websocketEndpoint string
clientURI *client.ClientURI
clientJSON *client.ClientJSONRPC
2016-01-10 16:33:52 -05:00
)
2016-01-12 16:54:27 -05: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 16:54:27 -05:00
chainID = config.GetString("chain_id")
rpcAddr = config.GetString("rpc_laddr")
2016-02-19 02:06:11 +00:00
requestAddr = rpcAddr
websocketAddr = rpcAddr
websocketEndpoint = "/websocket"
2016-01-12 16:54:27 -05: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 16:33:52 -05:00
// create a new node and sleep forever
func newNode(ready chan struct{}) {
// Create & start node
2016-01-12 19:30:31 -05:00
privValidatorFile := config.GetString("priv_validator_file")
privValidator := types.LoadOrGenPrivValidator(privValidatorFile)
2016-05-08 15:00:58 -07:00
node = nm.NewNode(config, privValidator, nm.GetProxyApp)
2016-01-10 16:33:52 -05:00
l := p2p.NewDefaultListener("tcp", config.GetString("node_laddr"), true)
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 21:20:25 -05:00
func newWSClient(t *testing.T) *client.WSClient {
2016-02-19 02:06:11 +00:00
wsc := client.NewWSClient(websocketAddr, websocketEndpoint)
2016-01-13 21:20:25 -05:00
if _, err := wsc.Start(); err != nil {
2016-01-10 16:33:52 -05:00
t.Fatal(err)
}
2016-01-13 21:20:25 -05:00
return wsc
2016-01-10 16:33:52 -05:00
}
// subscribe to an event
2016-01-13 21:20:25 -05:00
func subscribe(t *testing.T, wsc *client.WSClient, eventid string) {
if err := wsc.Subscribe(eventid); err != nil {
2016-01-10 16:33:52 -05:00
t.Fatal(err)
}
}
// unsubscribe from an event
2016-01-13 21:20:25 -05:00
func unsubscribe(t *testing.T, wsc *client.WSClient, eventid string) {
if err := wsc.Unsubscribe(eventid); err != nil {
2016-01-10 16:33:52 -05:00
t.Fatal(err)
}
}
// 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 21:20:25 -05:00
func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, dieOnTimeout bool, f func(), check func(string, interface{}) error) {
2016-01-10 16:33:52 -05:00
// go routine to wait for webscoket msg
2016-01-13 21:20:25 -05:00
goodCh := make(chan interface{})
2016-01-10 16:33:52 -05:00
errCh := make(chan error)
// Read message
go func() {
2016-01-13 21:20:25 -05:00
var err error
LOOP:
2016-01-10 16:33:52 -05:00
for {
2016-01-13 21:20:25 -05:00
select {
case r := <-wsc.ResultsCh:
2016-01-13 18:38:55 -05:00
result := new(ctypes.TMResult)
2016-01-13 21:20:25 -05:00
wire.ReadJSONPtr(result, r, &err)
2016-01-10 16:33:52 -05:00
if err != nil {
errCh <- err
2016-01-13 21:20:25 -05:00
break LOOP
2016-01-10 16:33:52 -05:00
}
2016-01-13 18:38:55 -05:00
event, ok := (*result).(*ctypes.ResultEvent)
if ok && event.Name == eventid {
2016-01-13 21:20:25 -05:00
goodCh <- event.Data
break LOOP
2016-01-10 16:33:52 -05:00
}
2016-01-13 21:20:25 -05:00
case err := <-wsc.ErrorsCh:
errCh <- err
break LOOP
case <-wsc.Quit:
break LOOP
2016-01-10 16:33:52 -05: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 21:20:25 -05:00
wsc.Stop()
2016-01-10 16:33:52 -05:00
t.Fatalf("%s event was not received in time", eventid)
}
// else that's great, we didn't hear the event
// and we shouldn't have
2016-01-13 21:20:25 -05:00
case eventData := <-goodCh:
2016-01-10 16:33:52 -05:00
if dieOnTimeout {
// message was received and expected
// run the check
2016-01-13 21:20:25 -05:00
if err := check(eventid, eventData); err != nil {
t.Fatal(err) // Show the stack trace.
2016-01-10 16:33:52 -05:00
}
} else {
2016-01-13 21:20:25 -05:00
wsc.Stop()
2016-01-10 16:33:52 -05:00
t.Fatalf("%s event was not expected", eventid)
}
case err := <-errCh:
t.Fatal(err)
panic(err) // Show the stack trace.
}
}
2016-01-13 21:20:25 -05:00
//--------------------------------------------------------------------------------