- package e2e_test
-
- import (
- "bytes"
- "context"
- "fmt"
- "math/rand"
- "testing"
- "time"
-
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
-
- "github.com/tendermint/tendermint/rpc/client/http"
- e2e "github.com/tendermint/tendermint/test/e2e/pkg"
- "github.com/tendermint/tendermint/types"
- )
-
- // Tests that any initial state given in genesis has made it into the app.
- func TestApp_InitialState(t *testing.T) {
- testNode(t, func(ctx context.Context, t *testing.T, node e2e.Node) {
- if len(node.Testnet.InitialState) == 0 {
- return
- }
-
- client, err := node.Client()
- require.NoError(t, err)
- for k, v := range node.Testnet.InitialState {
- resp, err := client.ABCIQuery(ctx, "", []byte(k))
- require.NoError(t, err)
- assert.Equal(t, k, string(resp.Response.Key))
- assert.Equal(t, v, string(resp.Response.Value))
- }
- })
- }
-
- // Tests that the app hash (as reported by the app) matches the last
- // block and the node sync status.
- func TestApp_Hash(t *testing.T) {
- testNode(t, func(ctx context.Context, t *testing.T, node e2e.Node) {
- client, err := node.Client()
- require.NoError(t, err)
- info, err := client.ABCIInfo(ctx)
- require.NoError(t, err)
- require.NotEmpty(t, info.Response.LastBlockAppHash, "expected app to return app hash")
-
- status, err := client.Status(ctx)
- require.NoError(t, err)
- require.NotZero(t, status.SyncInfo.LatestBlockHeight)
-
- block, err := client.Block(ctx, &info.Response.LastBlockHeight)
- require.NoError(t, err)
-
- if info.Response.LastBlockHeight == block.Block.Height {
- require.Equal(t,
- fmt.Sprintf("%x", info.Response.LastBlockAppHash),
- fmt.Sprintf("%x", block.Block.AppHash.Bytes()),
- "app hash does not match last block's app hash")
- }
-
- require.True(t, status.SyncInfo.LatestBlockHeight >= info.Response.LastBlockHeight,
- "status out of sync with application")
- })
- }
-
- // Tests that the app and blockstore have and report the same height.
- func TestApp_Height(t *testing.T) {
- testNode(t, func(ctx context.Context, t *testing.T, node e2e.Node) {
- client, err := node.Client()
- require.NoError(t, err)
- info, err := client.ABCIInfo(ctx)
- require.NoError(t, err)
- require.NotZero(t, info.Response.LastBlockHeight)
-
- status, err := client.Status(ctx)
- require.NoError(t, err)
- require.NotZero(t, status.SyncInfo.LatestBlockHeight)
-
- block, err := client.Block(ctx, &info.Response.LastBlockHeight)
- require.NoError(t, err)
-
- require.Equal(t, info.Response.LastBlockHeight, block.Block.Height)
-
- require.True(t, status.SyncInfo.LatestBlockHeight >= info.Response.LastBlockHeight,
- "status out of sync with application")
- })
- }
-
- // Tests that we can set a value and retrieve it.
- func TestApp_Tx(t *testing.T) {
- type broadcastFunc func(context.Context, types.Tx) error
-
- testCases := []struct {
- Name string
- WaitTime time.Duration
- BroadcastTx func(client *http.HTTP) broadcastFunc
- ShouldSkip bool
- }{
- {
- Name: "Sync",
- WaitTime: time.Minute,
- BroadcastTx: func(client *http.HTTP) broadcastFunc {
- return func(ctx context.Context, tx types.Tx) error {
- _, err := client.BroadcastTxSync(ctx, tx)
- return err
- }
- },
- },
- {
- Name: "Commit",
- WaitTime: 15 * time.Second,
- // TODO: turn this check back on if it can
- // return reliably. Currently these calls have
- // a hard timeout of 10s (server side
- // configured). The Sync check is probably
- // safe.
- ShouldSkip: true,
- BroadcastTx: func(client *http.HTTP) broadcastFunc {
- return func(ctx context.Context, tx types.Tx) error {
- _, err := client.BroadcastTxCommit(ctx, tx)
- return err
- }
- },
- },
- {
- Name: "Async",
- WaitTime: 90 * time.Second,
- // TODO: turn this check back on if there's a
- // way to avoid failures in the case that the
- // transaction doesn't make it into the
- // mempool. (retries?)
- ShouldSkip: true,
- BroadcastTx: func(client *http.HTTP) broadcastFunc {
- return func(ctx context.Context, tx types.Tx) error {
- _, err := client.BroadcastTxAsync(ctx, tx)
- return err
- }
- },
- },
- }
-
- for idx, test := range testCases {
- if test.ShouldSkip {
- continue
- }
- t.Run(test.Name, func(t *testing.T) {
- // testNode calls t.Parallel as well, so we should
- // have a copy of the
- test := testCases[idx]
- testNode(t, func(ctx context.Context, t *testing.T, node e2e.Node) {
- client, err := node.Client()
- require.NoError(t, err)
-
- // Generate a random value, to prevent duplicate tx errors when
- // manually running the test multiple times for a testnet.
- bz := make([]byte, 32)
- _, err = rand.Read(bz)
- require.NoError(t, err)
-
- key := fmt.Sprintf("testapp-tx-%v", node.Name)
- value := fmt.Sprintf("%x", bz)
- tx := types.Tx(fmt.Sprintf("%v=%v", key, value))
-
- require.NoError(t, test.BroadcastTx(client)(ctx, tx))
-
- hash := tx.Hash()
-
- require.Eventuallyf(t, func() bool {
- txResp, err := client.Tx(ctx, hash, false)
- return err == nil && bytes.Equal(txResp.Tx, tx)
- },
- test.WaitTime, // timeout
- time.Second, // interval
- "submitted tx %X wasn't committed after %v",
- hash, test.WaitTime,
- )
-
- abciResp, err := client.ABCIQuery(ctx, "", []byte(key))
- require.NoError(t, err)
- assert.Equal(t, key, string(abciResp.Response.Key))
- assert.Equal(t, value, string(abciResp.Response.Value))
- })
-
- })
-
- }
-
- }
|