204 lines
4.6 KiB
Go
204 lines
4.6 KiB
Go
package tests
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"time"
|
|
|
|
"strings"
|
|
|
|
amino "github.com/tendermint/go-amino"
|
|
tmclient "github.com/tendermint/tendermint/rpc/client"
|
|
ctypes "github.com/tendermint/tendermint/rpc/core/types"
|
|
rpcclient "github.com/tendermint/tendermint/rpc/lib/client"
|
|
)
|
|
|
|
// Wait for the next tendermint block from the Tendermint RPC
|
|
// on localhost
|
|
func WaitForNextHeightTM(port string) {
|
|
WaitForNextNBlocksTM(1, port)
|
|
}
|
|
|
|
// Wait for N tendermint blocks to pass using the Tendermint RPC
|
|
// on localhost
|
|
func WaitForNextNBlocksTM(n int64, port string) {
|
|
|
|
// get the latest block and wait for n more
|
|
url := fmt.Sprintf("http://localhost:%v", port)
|
|
cl := tmclient.NewHTTP(url, "/websocket")
|
|
resBlock, err := cl.Block(nil)
|
|
var height int64
|
|
if err != nil || resBlock.Block == nil {
|
|
// wait for the first block to exist
|
|
WaitForHeightTM(1, port)
|
|
height = 1 + n
|
|
} else {
|
|
height = resBlock.Block.Height + n
|
|
}
|
|
waitForHeightTM(height, url)
|
|
}
|
|
|
|
// Wait for the given height from the Tendermint RPC
|
|
// on localhost
|
|
func WaitForHeightTM(height int64, port string) {
|
|
url := fmt.Sprintf("http://localhost:%v", port)
|
|
waitForHeightTM(height, url)
|
|
}
|
|
|
|
func waitForHeightTM(height int64, url string) {
|
|
cl := tmclient.NewHTTP(url, "/websocket")
|
|
for {
|
|
// get url, try a few times
|
|
var resBlock *ctypes.ResultBlock
|
|
var err error
|
|
INNER:
|
|
for i := 0; i < 5; i++ {
|
|
resBlock, err = cl.Block(nil)
|
|
if err == nil {
|
|
break INNER
|
|
}
|
|
time.Sleep(time.Millisecond * 200)
|
|
}
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if resBlock.Block != nil &&
|
|
resBlock.Block.Height >= height {
|
|
return
|
|
}
|
|
time.Sleep(time.Millisecond * 100)
|
|
}
|
|
}
|
|
|
|
// Wait for height from the LCD API on localhost
|
|
func WaitForHeight(height int64, port string) {
|
|
url := fmt.Sprintf("http://localhost:%v/blocks/latest", port)
|
|
waitForHeight(height, url)
|
|
}
|
|
|
|
// Whether or not an HTTP status code was "successful"
|
|
func StatusOK(statusCode int) bool {
|
|
switch statusCode {
|
|
case http.StatusOK:
|
|
case http.StatusCreated:
|
|
case http.StatusNoContent:
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func waitForHeight(height int64, url string) {
|
|
var res *http.Response
|
|
var err error
|
|
for {
|
|
res, err = http.Get(url)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
body, err := ioutil.ReadAll(res.Body)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = res.Body.Close()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
var resultBlock ctypes.ResultBlock
|
|
err = cdc.UnmarshalJSON(body, &resultBlock)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if resultBlock.Block != nil &&
|
|
resultBlock.Block.Height >= height {
|
|
return
|
|
}
|
|
time.Sleep(time.Millisecond * 100)
|
|
}
|
|
}
|
|
|
|
// wait for tendermint to start by querying the LCD
|
|
func WaitForLCDStart(port string) {
|
|
url := fmt.Sprintf("http://localhost:%v/blocks/latest", port)
|
|
WaitForStart(url)
|
|
}
|
|
|
|
// wait for tendermint to start by querying tendermint
|
|
func WaitForTMStart(port string) {
|
|
url := fmt.Sprintf("http://localhost:%v/block", port)
|
|
WaitForStart(url)
|
|
}
|
|
|
|
// WaitForStart waits for the node to start by pinging the url
|
|
// every 100ms for 10s until it returns 200. If it takes longer than 5s,
|
|
// it panics.
|
|
func WaitForStart(url string) {
|
|
var err error
|
|
|
|
// ping the status endpoint a few times a second
|
|
// for a few seconds until we get a good response.
|
|
// otherwise something probably went wrong
|
|
for i := 0; i < 100; i++ {
|
|
time.Sleep(time.Millisecond * 100)
|
|
|
|
var res *http.Response
|
|
res, err = http.Get(url)
|
|
if err != nil || res == nil {
|
|
continue
|
|
}
|
|
// body, _ := ioutil.ReadAll(res.Body)
|
|
// fmt.Println("BODY", string(body))
|
|
err = res.Body.Close()
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
if res.StatusCode == http.StatusOK {
|
|
// good!
|
|
return
|
|
}
|
|
}
|
|
// still haven't started up?! panic!
|
|
panic(err)
|
|
}
|
|
|
|
// TODO: these functions just print to Stdout.
|
|
// consider using the logger.
|
|
|
|
// Wait for the RPC server to respond to /status
|
|
func WaitForRPC(laddr string) {
|
|
fmt.Println("LADDR", laddr)
|
|
client := rpcclient.NewJSONRPCClient(laddr)
|
|
ctypes.RegisterAmino(client.Codec())
|
|
result := new(ctypes.ResultStatus)
|
|
for {
|
|
_, err := client.Call("status", map[string]interface{}{}, result)
|
|
if err == nil {
|
|
return
|
|
}
|
|
fmt.Printf("Waiting for RPC server to start on %s:%v\n", laddr, err)
|
|
time.Sleep(time.Millisecond)
|
|
}
|
|
}
|
|
|
|
// ExtractPortFromAddress extract port from listenAddress
|
|
// The listenAddress must be some strings like tcp://0.0.0.0:12345
|
|
func ExtractPortFromAddress(listenAddress string) string {
|
|
stringList := strings.Split(listenAddress, ":")
|
|
length := len(stringList)
|
|
if length != 3 {
|
|
panic(fmt.Errorf("expected listen address: tcp://0.0.0.0:12345, got %s", listenAddress))
|
|
}
|
|
return stringList[2]
|
|
}
|
|
|
|
var cdc = amino.NewCodec()
|
|
|
|
func init() {
|
|
ctypes.RegisterAmino(cdc)
|
|
}
|