- package eventbus_test
-
- import (
- "context"
- "fmt"
- mrand "math/rand"
- "testing"
- "time"
-
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
-
- abci "github.com/tendermint/tendermint/abci/types"
- "github.com/tendermint/tendermint/internal/eventbus"
- tmpubsub "github.com/tendermint/tendermint/internal/pubsub"
- tmquery "github.com/tendermint/tendermint/internal/pubsub/query"
- "github.com/tendermint/tendermint/libs/log"
- "github.com/tendermint/tendermint/types"
- )
-
- func TestEventBusPublishEventTx(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
-
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- tx := types.Tx("foo")
- result := abci.ResponseDeliverTx{
- Data: []byte("bar"),
- Events: []abci.Event{
- {Type: "testType", Attributes: []abci.EventAttribute{{Key: "baz", Value: "1"}}},
- },
- }
-
- // PublishEventTx adds 3 composite keys, so the query below should work
- query := fmt.Sprintf("tm.event='Tx' AND tx.height=1 AND tx.hash='%X' AND testType.baz=1", tx.Hash())
- txsSub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: "test",
- Query: tmquery.MustCompile(query),
- })
- require.NoError(t, err)
-
- done := make(chan struct{})
- go func() {
- defer close(done)
- msg, err := txsSub.Next(ctx)
- assert.NoError(t, err)
-
- edt := msg.Data().(types.EventDataTx)
- assert.Equal(t, int64(1), edt.Height)
- assert.Equal(t, uint32(0), edt.Index)
- assert.EqualValues(t, tx, edt.Tx)
- assert.Equal(t, result, edt.Result)
- }()
-
- err = eventBus.PublishEventTx(ctx, types.EventDataTx{
- TxResult: abci.TxResult{
- Height: 1,
- Index: 0,
- Tx: tx,
- Result: result,
- },
- })
- assert.NoError(t, err)
-
- select {
- case <-done:
- case <-time.After(1 * time.Second):
- t.Fatal("did not receive a transaction after 1 sec.")
- }
- }
-
- func TestEventBusPublishEventNewBlock(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- block := types.MakeBlock(0, []types.Tx{}, nil, []types.Evidence{})
- bps, err := block.MakePartSet(types.BlockPartSizeBytes)
- require.NoError(t, err)
- blockID := types.BlockID{Hash: block.Hash(), PartSetHeader: bps.Header()}
- resultFinalizeBlock := abci.ResponseFinalizeBlock{
- Events: []abci.Event{
- {Type: "testType", Attributes: []abci.EventAttribute{
- {Key: "baz", Value: "1"},
- {Key: "foz", Value: "2"},
- }},
- },
- }
-
- // PublishEventNewBlock adds the tm.event compositeKey, so the query below should work
- query := "tm.event='NewBlock' AND testType.baz=1 AND testType.foz=2"
- blocksSub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: "test",
- Query: tmquery.MustCompile(query),
- })
- require.NoError(t, err)
-
- done := make(chan struct{})
- go func() {
- defer close(done)
- msg, err := blocksSub.Next(ctx)
- assert.NoError(t, err)
-
- edt := msg.Data().(types.EventDataNewBlock)
- assert.Equal(t, block, edt.Block)
- assert.Equal(t, blockID, edt.BlockID)
- assert.Equal(t, resultFinalizeBlock, edt.ResultFinalizeBlock)
- }()
-
- err = eventBus.PublishEventNewBlock(ctx, types.EventDataNewBlock{
- Block: block,
- BlockID: blockID,
- ResultFinalizeBlock: resultFinalizeBlock,
- })
- assert.NoError(t, err)
-
- select {
- case <-done:
- case <-time.After(1 * time.Second):
- t.Fatal("did not receive a block after 1 sec.")
- }
- }
-
- func TestEventBusPublishEventTxDuplicateKeys(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- tx := types.Tx("foo")
- result := abci.ResponseDeliverTx{
- Data: []byte("bar"),
- Events: []abci.Event{
- {
- Type: "transfer",
- Attributes: []abci.EventAttribute{
- {Key: "sender", Value: "foo"},
- {Key: "recipient", Value: "bar"},
- {Key: "amount", Value: "5"},
- },
- },
- {
- Type: "transfer",
- Attributes: []abci.EventAttribute{
- {Key: "sender", Value: "baz"},
- {Key: "recipient", Value: "cat"},
- {Key: "amount", Value: "13"},
- },
- },
- {
- Type: "withdraw.rewards",
- Attributes: []abci.EventAttribute{
- {Key: "address", Value: "bar"},
- {Key: "source", Value: "iceman"},
- {Key: "amount", Value: "33"},
- },
- },
- },
- }
-
- testCases := []struct {
- query string
- expectResults bool
- }{
- {
- "tm.event='Tx' AND tx.height=1 AND transfer.sender='DoesNotExist'",
- false,
- },
- {
- "tm.event='Tx' AND tx.height=1 AND transfer.sender='foo'",
- true,
- },
- {
- "tm.event='Tx' AND tx.height=1 AND transfer.sender='baz'",
- true,
- },
- {
- "tm.event='Tx' AND tx.height=1 AND transfer.sender='foo' AND transfer.sender='baz'",
- true,
- },
- {
- "tm.event='Tx' AND tx.height=1 AND transfer.sender='foo' AND transfer.sender='DoesNotExist'",
- false,
- },
- }
-
- for i, tc := range testCases {
- var name string
-
- if tc.expectResults {
- name = fmt.Sprintf("ExpetedResultsCase%d", i)
- } else {
- name = fmt.Sprintf("NoResultsCase%d", i)
- }
-
- t.Run(name, func(t *testing.T) {
-
- sub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: fmt.Sprintf("client-%d", i),
- Query: tmquery.MustCompile(tc.query),
- })
- require.NoError(t, err)
-
- gotResult := make(chan bool, 1)
- go func() {
- defer close(gotResult)
- tctx, cancel := context.WithTimeout(ctx, 1*time.Second)
- defer cancel()
- msg, err := sub.Next(tctx)
- if err == nil {
- data := msg.Data().(types.EventDataTx)
- assert.Equal(t, int64(1), data.Height)
- assert.Equal(t, uint32(0), data.Index)
- assert.EqualValues(t, tx, data.Tx)
- assert.Equal(t, result, data.Result)
- gotResult <- true
- }
- }()
-
- assert.NoError(t, eventBus.PublishEventTx(ctx, types.EventDataTx{
- TxResult: abci.TxResult{
- Height: 1,
- Index: 0,
- Tx: tx,
- Result: result,
- },
- }))
-
- require.NoError(t, ctx.Err(), "context should not have been canceled")
-
- if got := <-gotResult; got != tc.expectResults {
- require.Failf(t, "Wrong transaction result",
- "got a tx: %v, wanted a tx: %v", got, tc.expectResults)
- }
- })
-
- }
- }
-
- func TestEventBusPublishEventNewBlockHeader(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
-
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- block := types.MakeBlock(0, []types.Tx{}, nil, []types.Evidence{})
- resultFinalizeBlock := abci.ResponseFinalizeBlock{
- Events: []abci.Event{
- {Type: "testType", Attributes: []abci.EventAttribute{
- {Key: "baz", Value: "1"},
- {Key: "foz", Value: "2"},
- }},
- },
- }
-
- // PublishEventNewBlockHeader adds the tm.event compositeKey, so the query below should work
- query := "tm.event='NewBlockHeader' AND testType.baz=1 AND testType.foz=2"
- headersSub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: "test",
- Query: tmquery.MustCompile(query),
- })
- require.NoError(t, err)
-
- done := make(chan struct{})
- go func() {
- defer close(done)
- msg, err := headersSub.Next(ctx)
- assert.NoError(t, err)
-
- edt := msg.Data().(types.EventDataNewBlockHeader)
- assert.Equal(t, block.Header, edt.Header)
- assert.Equal(t, resultFinalizeBlock, edt.ResultFinalizeBlock)
- }()
-
- err = eventBus.PublishEventNewBlockHeader(ctx, types.EventDataNewBlockHeader{
- Header: block.Header,
- ResultFinalizeBlock: resultFinalizeBlock,
- })
- assert.NoError(t, err)
-
- select {
- case <-done:
- case <-time.After(1 * time.Second):
- t.Fatal("did not receive a block header after 1 sec.")
- }
- }
-
- func TestEventBusPublishEventNewEvidence(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
-
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- ev, err := types.NewMockDuplicateVoteEvidence(ctx, 1, time.Now(), "test-chain-id")
- require.NoError(t, err)
-
- const query = `tm.event='NewEvidence'`
- evSub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: "test",
- Query: tmquery.MustCompile(query),
- })
- require.NoError(t, err)
-
- done := make(chan struct{})
- go func() {
- defer close(done)
- msg, err := evSub.Next(ctx)
- assert.NoError(t, err)
-
- edt := msg.Data().(types.EventDataNewEvidence)
- assert.Equal(t, ev, edt.Evidence)
- assert.Equal(t, int64(4), edt.Height)
- }()
-
- err = eventBus.PublishEventNewEvidence(ctx, types.EventDataNewEvidence{
- Evidence: ev,
- Height: 4,
- })
- assert.NoError(t, err)
-
- select {
- case <-done:
- case <-time.After(1 * time.Second):
- t.Fatal("did not receive a block header after 1 sec.")
- }
- }
-
- func TestEventBusPublish(t *testing.T) {
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
-
- eventBus := eventbus.NewDefault(log.TestingLogger())
- err := eventBus.Start(ctx)
- require.NoError(t, err)
-
- const numEventsExpected = 14
-
- sub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: "test",
- Query: tmquery.All,
- Limit: numEventsExpected,
- })
- require.NoError(t, err)
-
- count := make(chan int, 1)
- go func() {
- defer close(count)
- ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
- defer cancel()
-
- for n := 0; ; n++ {
- if _, err := sub.Next(ctx); err != nil {
- count <- n
- return
- }
- }
- }()
-
- require.NoError(t, eventBus.Publish(ctx, types.EventNewBlockHeaderValue,
- types.EventDataNewBlockHeader{}))
- require.NoError(t, eventBus.PublishEventNewBlock(ctx, types.EventDataNewBlock{}))
- require.NoError(t, eventBus.PublishEventNewBlockHeader(ctx, types.EventDataNewBlockHeader{}))
- require.NoError(t, eventBus.PublishEventVote(ctx, types.EventDataVote{}))
- require.NoError(t, eventBus.PublishEventNewRoundStep(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventTimeoutPropose(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventTimeoutWait(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventNewRound(ctx, types.EventDataNewRound{}))
- require.NoError(t, eventBus.PublishEventCompleteProposal(ctx, types.EventDataCompleteProposal{}))
- require.NoError(t, eventBus.PublishEventPolka(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventRelock(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventLock(ctx, types.EventDataRoundState{}))
- require.NoError(t, eventBus.PublishEventValidatorSetUpdates(ctx, types.EventDataValidatorSetUpdates{}))
- require.NoError(t, eventBus.PublishEventBlockSyncStatus(ctx, types.EventDataBlockSyncStatus{}))
- require.NoError(t, eventBus.PublishEventStateSyncStatus(ctx, types.EventDataStateSyncStatus{}))
-
- require.GreaterOrEqual(t, <-count, numEventsExpected)
- }
-
- func BenchmarkEventBus(b *testing.B) {
- benchmarks := []struct {
- name string
- numClients int
- randQueries bool
- randEvents bool
- }{
- {"10Clients1Query1Event", 10, false, false},
- {"100Clients", 100, false, false},
- {"1000Clients", 1000, false, false},
-
- {"10ClientsRandQueries1Event", 10, true, false},
- {"100Clients", 100, true, false},
- {"1000Clients", 1000, true, false},
-
- {"10ClientsRandQueriesRandEvents", 10, true, true},
- {"100Clients", 100, true, true},
- {"1000Clients", 1000, true, true},
-
- {"10Clients1QueryRandEvents", 10, false, true},
- {"100Clients", 100, false, true},
- {"1000Clients", 1000, false, true},
- }
-
- for _, bm := range benchmarks {
- bm := bm
- b.Run(bm.name, func(b *testing.B) {
- benchmarkEventBus(bm.numClients, bm.randQueries, bm.randEvents, b)
- })
- }
- }
-
- func benchmarkEventBus(numClients int, randQueries bool, randEvents bool, b *testing.B) {
- // for random* functions
- mrand.Seed(time.Now().Unix())
-
- ctx, cancel := context.WithCancel(context.Background())
- defer cancel()
-
- eventBus := eventbus.NewDefault(log.TestingLogger()) // set buffer capacity to 0 so we are not testing cache
- err := eventBus.Start(ctx)
- if err != nil {
- b.Error(err)
- }
- b.Cleanup(eventBus.Wait)
-
- q := types.EventQueryNewBlock
-
- for i := 0; i < numClients; i++ {
- if randQueries {
- q = randQuery()
- }
- sub, err := eventBus.SubscribeWithArgs(ctx, tmpubsub.SubscribeArgs{
- ClientID: fmt.Sprintf("client-%d", i),
- Query: q,
- })
- if err != nil {
- b.Fatal(err)
- }
- go func() {
- for {
- if _, err := sub.Next(ctx); err != nil {
- return
- }
- }
- }()
- }
-
- eventValue := types.EventNewBlockValue
-
- b.ReportAllocs()
- b.ResetTimer()
- for i := 0; i < b.N; i++ {
- if randEvents {
- eventValue = randEventValue()
- }
-
- err := eventBus.Publish(ctx, eventValue, types.EventDataString("Gamora"))
- if err != nil {
- b.Error(err)
- }
- }
- }
-
- var events = []string{
- types.EventNewBlockValue,
- types.EventNewBlockHeaderValue,
- types.EventNewRoundValue,
- types.EventNewRoundStepValue,
- types.EventTimeoutProposeValue,
- types.EventCompleteProposalValue,
- types.EventPolkaValue,
- types.EventLockValue,
- types.EventRelockValue,
- types.EventTimeoutWaitValue,
- types.EventVoteValue,
- types.EventBlockSyncStatusValue,
- types.EventStateSyncStatusValue,
- }
-
- func randEventValue() string {
- return events[mrand.Intn(len(events))]
- }
-
- var queries = []*tmquery.Query{
- types.EventQueryNewBlock,
- types.EventQueryNewBlockHeader,
- types.EventQueryNewRound,
- types.EventQueryNewRoundStep,
- types.EventQueryTimeoutPropose,
- types.EventQueryCompleteProposal,
- types.EventQueryPolka,
- types.EventQueryLock,
- types.EventQueryRelock,
- types.EventQueryTimeoutWait,
- types.EventQueryVote,
- types.EventQueryBlockSyncStatus,
- types.EventQueryStateSyncStatus,
- }
-
- func randQuery() *tmquery.Query {
- return queries[mrand.Intn(len(queries))]
- }
|