From a47d0a64787c825fe5efee465090bc11021be968 Mon Sep 17 00:00:00 2001 From: Silas Davis <silas@erisindustries.com> Date: Mon, 12 Sep 2016 21:37:59 +0100 Subject: [PATCH] Fix some issues with waitForEvent and friends --- rpc/tendermint/test/tests.go | 11 ++++++----- rpc/tendermint/test/ws_helpers.go | 32 +++++++++++++++++++++---------- 2 files changed, 28 insertions(+), 15 deletions(-) diff --git a/rpc/tendermint/test/tests.go b/rpc/tendermint/test/tests.go index 56ea89b2..43769b4b 100644 --- a/rpc/tendermint/test/tests.go +++ b/rpc/tendermint/test/tests.go @@ -202,7 +202,7 @@ func testNameReg(t *testing.T, typ string) { assert.Equal(t, user[0].Address, entry.Owner) // update the data as the owner, make sure still there - numDesiredBlocks = int64(3) + numDesiredBlocks = int64(5) const updatedData = "these are amongst the things I wish to bestow upon the youth of generations come: a safe supply of honey, and a better money. For what else shall they need" amt = fee + numDesiredBlocks*txs.NameByteCostMultiplier*txs.NameBlockCostMultiplier*txs.NameBaseCost(name, updatedData) tx = makeDefaultNameTx(t, typ, name, updatedData, amt, fee) @@ -220,9 +220,11 @@ func testNameReg(t *testing.T, typ string) { _, err := broadcastTxAndWaitForBlock(t, typ, wsc, tx) assert.Error(t, err, "Expected error when updating someone else's unexpired"+ " name registry entry") + assert.Contains(t, err.Error(), "permission denied", "Error should be " + + "permission denied") // Wait a couple of blocks to make sure name registration expires - waitNBlocks(t, wsc, 2) + waitNBlocks(t, wsc, 3) //now the entry should be expired, so we can update as non owner const data2 = "this is not my beautiful house" @@ -309,7 +311,6 @@ func testBlockchainInfo(t *testing.T, typ string) { wsc := newWSClient(t) nBlocks := 4 waitNBlocks(t, wsc, nBlocks) - time.Sleep(time.Millisecond * 200) resp, err := edbcli.BlockchainInfo(client, 0, 0) if err != nil { @@ -319,8 +320,8 @@ func testBlockchainInfo(t *testing.T, typ string) { // NewBlock after saving a block // see https://github.com/tendermint/tendermint/issues/273 //assert.Equal(t, 4, resp.LastHeight, "Last height should be 4 after waiting for first 4 blocks") - assert.Equal(t, nBlocks, len(resp.BlockMetas), - "Should see 4 BlockMetas after waiting for first 4 blocks") + assert.True(t, nBlocks <= len(resp.BlockMetas), + "Should see at least 4 BlockMetas after waiting for first 4 blocks") lastBlockHash := resp.BlockMetas[nBlocks-1].Hash for i := nBlocks - 2; i >= 0; i-- { diff --git a/rpc/tendermint/test/ws_helpers.go b/rpc/tendermint/test/ws_helpers.go index a13635da..1c71a72b 100644 --- a/rpc/tendermint/test/ws_helpers.go +++ b/rpc/tendermint/test/ws_helpers.go @@ -81,7 +81,12 @@ func broadcastTxAndWaitForBlock(t *testing.T, typ string, wsc *client.WSClient, initialHeight = block.Height return false } else { - return block.Height > initialHeight + // TODO: [Silas] remove the + 1 here. It is a workaround for the fact + // that tendermint fires the NewBlock event before it has finalised its + // state updates, so we have to wait for the block after the block we + // want in order for the Tx to be genuinely final. + // This should be addressed by: https://github.com/tendermint/tendermint/pull/265 + return block.Height > initialHeight + 1 } }, func() { @@ -131,10 +136,11 @@ func subscribeAndWaitForNext(t *testing.T, wsc *client.WSClient, event string, // waitForEvent will fail the test. func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, runner func(), - eventDataChecker func(string, txs.EventData) (bool, error)) waitForEventError { + eventDataChecker func(string, txs.EventData) (bool, error)) waitForEventResult { // go routine to wait for websocket msg - goodCh := make(chan txs.EventData) + eventsCh := make(chan txs.EventData) + shutdownEventsCh := make(chan bool, 1) errCh := make(chan error) // do stuff (transactions) @@ -146,6 +152,8 @@ func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, LOOP: for { select { + case <-shutdownEventsCh: + break LOOP case r := <-wsc.ResultsCh: result := new(ctypes.ErisDBResult) wire.ReadJSONPtr(result, r, &err) @@ -155,7 +163,8 @@ func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, } event, ok := (*result).(*ctypes.ResultEvent) if ok && event.Event == eventid { - goodCh <- event.Data + // Keep feeding events + eventsCh <- event.Data } case err := <-wsc.ErrorsCh: errCh <- err @@ -166,19 +175,22 @@ func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, } }() + // Don't block up WSClient + defer func() { shutdownEventsCh <- true }() + for { select { // wait for an event or timeout - case <-time.After(timeoutSeconds * time.Second*3): - return waitForEventError{timeout: true} - case eventData := <-goodCh: + case <-time.After(timeoutSeconds * time.Second): + return waitForEventResult{timeout: true} + case eventData := <-eventsCh: // run the check stopWaiting, err := eventDataChecker(eventid, eventData) if err != nil { t.Fatal(err) // Show the stack trace. } if stopWaiting { - return waitForEventError{} + return waitForEventResult{} } case err := <-errCh: t.Fatal(err) @@ -186,12 +198,12 @@ func waitForEvent(t *testing.T, wsc *client.WSClient, eventid string, } } -type waitForEventError struct { +type waitForEventResult struct { error timeout bool } -func (err waitForEventError) Timeout() bool { +func (err waitForEventResult) Timeout() bool { return err.timeout } -- GitLab