github.com/mydexchain/tendermint@v0.0.4/mempool/reactor_test.go (about)

     1  package mempool
     2  
     3  import (
     4  	"encoding/hex"
     5  	"errors"
     6  	"net"
     7  	"sync"
     8  	"testing"
     9  	"time"
    10  
    11  	"github.com/fortytw2/leaktest"
    12  	"github.com/go-kit/kit/log/term"
    13  	"github.com/stretchr/testify/assert"
    14  	"github.com/stretchr/testify/require"
    15  
    16  	"github.com/mydexchain/tendermint/abci/example/kvstore"
    17  	abci "github.com/mydexchain/tendermint/abci/types"
    18  	cfg "github.com/mydexchain/tendermint/config"
    19  	"github.com/mydexchain/tendermint/libs/log"
    20  	tmrand "github.com/mydexchain/tendermint/libs/rand"
    21  	"github.com/mydexchain/tendermint/p2p"
    22  	"github.com/mydexchain/tendermint/p2p/mock"
    23  	memproto "github.com/mydexchain/tendermint/proto/tendermint/mempool"
    24  	"github.com/mydexchain/tendermint/proxy"
    25  	"github.com/mydexchain/tendermint/types"
    26  )
    27  
    28  const (
    29  	numTxs  = 1000
    30  	timeout = 120 * time.Second // ridiculously high because CircleCI is slow
    31  )
    32  
    33  type peerState struct {
    34  	height int64
    35  }
    36  
    37  func (ps peerState) GetHeight() int64 {
    38  	return ps.height
    39  }
    40  
    41  // Send a bunch of txs to the first reactor's mempool and wait for them all to
    42  // be received in the others.
    43  func TestReactorBroadcastTxsMessage(t *testing.T) {
    44  	config := cfg.TestConfig()
    45  	// if there were more than two reactors, the order of transactions could not be
    46  	// asserted in waitForTxsOnReactors (due to transactions gossiping). If we
    47  	// replace Connect2Switches (full mesh) with a func, which connects first
    48  	// reactor to others and nothing else, this test should also pass with >2 reactors.
    49  	const N = 2
    50  	reactors := makeAndConnectReactors(config, N)
    51  	defer func() {
    52  		for _, r := range reactors {
    53  			if err := r.Stop(); err != nil {
    54  				assert.NoError(t, err)
    55  			}
    56  		}
    57  	}()
    58  	for _, r := range reactors {
    59  		for _, peer := range r.Switch.Peers().List() {
    60  			peer.Set(types.PeerStateKey, peerState{1})
    61  		}
    62  	}
    63  
    64  	txs := checkTxs(t, reactors[0].mempool, numTxs, UnknownPeerID)
    65  	waitForTxsOnReactors(t, txs, reactors)
    66  }
    67  
    68  // regression test for https://github.com/mydexchain/tendermint/issues/5408
    69  func TestReactorConcurrency(t *testing.T) {
    70  	config := cfg.TestConfig()
    71  	const N = 2
    72  	reactors := makeAndConnectReactors(config, N)
    73  	defer func() {
    74  		for _, r := range reactors {
    75  			if err := r.Stop(); err != nil {
    76  				assert.NoError(t, err)
    77  			}
    78  		}
    79  	}()
    80  	for _, r := range reactors {
    81  		for _, peer := range r.Switch.Peers().List() {
    82  			peer.Set(types.PeerStateKey, peerState{1})
    83  		}
    84  	}
    85  	var wg sync.WaitGroup
    86  
    87  	const numTxs = 5
    88  
    89  	for i := 0; i < 1000; i++ {
    90  		wg.Add(2)
    91  
    92  		// 1. submit a bunch of txs
    93  		// 2. update the whole mempool
    94  		txs := checkTxs(t, reactors[0].mempool, numTxs, UnknownPeerID)
    95  		go func() {
    96  			defer wg.Done()
    97  
    98  			reactors[0].mempool.Lock()
    99  			defer reactors[0].mempool.Unlock()
   100  
   101  			deliverTxResponses := make([]*abci.ResponseDeliverTx, len(txs))
   102  			for i := range txs {
   103  				deliverTxResponses[i] = &abci.ResponseDeliverTx{Code: 0}
   104  			}
   105  			err := reactors[0].mempool.Update(1, txs, deliverTxResponses, nil, nil)
   106  			assert.NoError(t, err)
   107  		}()
   108  
   109  		// 1. submit a bunch of txs
   110  		// 2. update none
   111  		_ = checkTxs(t, reactors[1].mempool, numTxs, UnknownPeerID)
   112  		go func() {
   113  			defer wg.Done()
   114  
   115  			reactors[1].mempool.Lock()
   116  			defer reactors[1].mempool.Unlock()
   117  			err := reactors[1].mempool.Update(1, []types.Tx{}, make([]*abci.ResponseDeliverTx, 0), nil, nil)
   118  			assert.NoError(t, err)
   119  		}()
   120  
   121  		// 1. flush the mempool
   122  		reactors[1].mempool.Flush()
   123  	}
   124  
   125  	wg.Wait()
   126  }
   127  
   128  // Send a bunch of txs to the first reactor's mempool, claiming it came from peer
   129  // ensure peer gets no txs.
   130  func TestReactorNoBroadcastToSender(t *testing.T) {
   131  	config := cfg.TestConfig()
   132  	const N = 2
   133  	reactors := makeAndConnectReactors(config, N)
   134  	defer func() {
   135  		for _, r := range reactors {
   136  			if err := r.Stop(); err != nil {
   137  				assert.NoError(t, err)
   138  			}
   139  		}
   140  	}()
   141  	for _, r := range reactors {
   142  		for _, peer := range r.Switch.Peers().List() {
   143  			peer.Set(types.PeerStateKey, peerState{1})
   144  		}
   145  	}
   146  
   147  	const peerID = 1
   148  	checkTxs(t, reactors[0].mempool, numTxs, peerID)
   149  	ensureNoTxs(t, reactors[peerID], 100*time.Millisecond)
   150  }
   151  
   152  func TestReactor_MaxTxBytes(t *testing.T) {
   153  	config := cfg.TestConfig()
   154  
   155  	const N = 2
   156  	reactors := makeAndConnectReactors(config, N)
   157  	defer func() {
   158  		for _, r := range reactors {
   159  			if err := r.Stop(); err != nil {
   160  				assert.NoError(t, err)
   161  			}
   162  		}
   163  	}()
   164  	for _, r := range reactors {
   165  		for _, peer := range r.Switch.Peers().List() {
   166  			peer.Set(types.PeerStateKey, peerState{1})
   167  		}
   168  	}
   169  
   170  	// Broadcast a tx, which has the max size
   171  	// => ensure it's received by the second reactor.
   172  	tx1 := tmrand.Bytes(config.Mempool.MaxTxBytes)
   173  	err := reactors[0].mempool.CheckTx(tx1, nil, TxInfo{SenderID: UnknownPeerID})
   174  	require.NoError(t, err)
   175  	waitForTxsOnReactors(t, []types.Tx{tx1}, reactors)
   176  
   177  	reactors[0].mempool.Flush()
   178  	reactors[1].mempool.Flush()
   179  
   180  	// Broadcast a tx, which is beyond the max size
   181  	// => ensure it's not sent
   182  	tx2 := tmrand.Bytes(config.Mempool.MaxTxBytes + 1)
   183  	err = reactors[0].mempool.CheckTx(tx2, nil, TxInfo{SenderID: UnknownPeerID})
   184  	require.Error(t, err)
   185  }
   186  
   187  func TestBroadcastTxForPeerStopsWhenPeerStops(t *testing.T) {
   188  	if testing.Short() {
   189  		t.Skip("skipping test in short mode.")
   190  	}
   191  
   192  	config := cfg.TestConfig()
   193  	const N = 2
   194  	reactors := makeAndConnectReactors(config, N)
   195  	defer func() {
   196  		for _, r := range reactors {
   197  			if err := r.Stop(); err != nil {
   198  				assert.NoError(t, err)
   199  			}
   200  		}
   201  	}()
   202  
   203  	// stop peer
   204  	sw := reactors[1].Switch
   205  	sw.StopPeerForError(sw.Peers().List()[0], errors.New("some reason"))
   206  
   207  	// check that we are not leaking any go-routines
   208  	// i.e. broadcastTxRoutine finishes when peer is stopped
   209  	leaktest.CheckTimeout(t, 10*time.Second)()
   210  }
   211  
   212  func TestBroadcastTxForPeerStopsWhenReactorStops(t *testing.T) {
   213  	if testing.Short() {
   214  		t.Skip("skipping test in short mode.")
   215  	}
   216  
   217  	config := cfg.TestConfig()
   218  	const N = 2
   219  	reactors := makeAndConnectReactors(config, N)
   220  
   221  	// stop reactors
   222  	for _, r := range reactors {
   223  		if err := r.Stop(); err != nil {
   224  			assert.NoError(t, err)
   225  		}
   226  	}
   227  
   228  	// check that we are not leaking any go-routines
   229  	// i.e. broadcastTxRoutine finishes when reactor is stopped
   230  	leaktest.CheckTimeout(t, 10*time.Second)()
   231  }
   232  
   233  func TestMempoolIDsBasic(t *testing.T) {
   234  	ids := newMempoolIDs()
   235  
   236  	peer := mock.NewPeer(net.IP{127, 0, 0, 1})
   237  
   238  	ids.ReserveForPeer(peer)
   239  	assert.EqualValues(t, 1, ids.GetForPeer(peer))
   240  	ids.Reclaim(peer)
   241  
   242  	ids.ReserveForPeer(peer)
   243  	assert.EqualValues(t, 2, ids.GetForPeer(peer))
   244  	ids.Reclaim(peer)
   245  }
   246  
   247  func TestMempoolIDsPanicsIfNodeRequestsOvermaxActiveIDs(t *testing.T) {
   248  	if testing.Short() {
   249  		return
   250  	}
   251  
   252  	// 0 is already reserved for UnknownPeerID
   253  	ids := newMempoolIDs()
   254  
   255  	for i := 0; i < maxActiveIDs-1; i++ {
   256  		peer := mock.NewPeer(net.IP{127, 0, 0, 1})
   257  		ids.ReserveForPeer(peer)
   258  	}
   259  
   260  	assert.Panics(t, func() {
   261  		peer := mock.NewPeer(net.IP{127, 0, 0, 1})
   262  		ids.ReserveForPeer(peer)
   263  	})
   264  }
   265  
   266  func TestDontExhaustMaxActiveIDs(t *testing.T) {
   267  	config := cfg.TestConfig()
   268  	const N = 1
   269  	reactors := makeAndConnectReactors(config, N)
   270  	defer func() {
   271  		for _, r := range reactors {
   272  			if err := r.Stop(); err != nil {
   273  				assert.NoError(t, err)
   274  			}
   275  		}
   276  	}()
   277  	reactor := reactors[0]
   278  
   279  	for i := 0; i < maxActiveIDs+1; i++ {
   280  		peer := mock.NewPeer(nil)
   281  		reactor.Receive(MempoolChannel, peer, []byte{0x1, 0x2, 0x3})
   282  		reactor.AddPeer(peer)
   283  	}
   284  }
   285  
   286  // mempoolLogger is a TestingLogger which uses a different
   287  // color for each validator ("validator" key must exist).
   288  func mempoolLogger() log.Logger {
   289  	return log.TestingLoggerWithColorFn(func(keyvals ...interface{}) term.FgBgColor {
   290  		for i := 0; i < len(keyvals)-1; i += 2 {
   291  			if keyvals[i] == "validator" {
   292  				return term.FgBgColor{Fg: term.Color(uint8(keyvals[i+1].(int) + 1))}
   293  			}
   294  		}
   295  		return term.FgBgColor{}
   296  	})
   297  }
   298  
   299  // connect N mempool reactors through N switches
   300  func makeAndConnectReactors(config *cfg.Config, n int) []*Reactor {
   301  	reactors := make([]*Reactor, n)
   302  	logger := mempoolLogger()
   303  	for i := 0; i < n; i++ {
   304  		app := kvstore.NewApplication()
   305  		cc := proxy.NewLocalClientCreator(app)
   306  		mempool, cleanup := newMempoolWithApp(cc)
   307  		defer cleanup()
   308  
   309  		reactors[i] = NewReactor(config.Mempool, mempool) // so we dont start the consensus states
   310  		reactors[i].SetLogger(logger.With("validator", i))
   311  	}
   312  
   313  	p2p.MakeConnectedSwitches(config.P2P, n, func(i int, s *p2p.Switch) *p2p.Switch {
   314  		s.AddReactor("MEMPOOL", reactors[i])
   315  		return s
   316  
   317  	}, p2p.Connect2Switches)
   318  	return reactors
   319  }
   320  
   321  func waitForTxsOnReactors(t *testing.T, txs types.Txs, reactors []*Reactor) {
   322  	// wait for the txs in all mempools
   323  	wg := new(sync.WaitGroup)
   324  	for i, reactor := range reactors {
   325  		wg.Add(1)
   326  		go func(r *Reactor, reactorIndex int) {
   327  			defer wg.Done()
   328  			waitForTxsOnReactor(t, txs, r, reactorIndex)
   329  		}(reactor, i)
   330  	}
   331  
   332  	done := make(chan struct{})
   333  	go func() {
   334  		wg.Wait()
   335  		close(done)
   336  	}()
   337  
   338  	timer := time.After(timeout)
   339  	select {
   340  	case <-timer:
   341  		t.Fatal("Timed out waiting for txs")
   342  	case <-done:
   343  	}
   344  }
   345  
   346  func waitForTxsOnReactor(t *testing.T, txs types.Txs, reactor *Reactor, reactorIndex int) {
   347  	mempool := reactor.mempool
   348  	for mempool.Size() < len(txs) {
   349  		time.Sleep(time.Millisecond * 100)
   350  	}
   351  
   352  	reapedTxs := mempool.ReapMaxTxs(len(txs))
   353  	for i, tx := range txs {
   354  		assert.Equalf(t, tx, reapedTxs[i],
   355  			"txs at index %d on reactor %d don't match: %v vs %v", i, reactorIndex, tx, reapedTxs[i])
   356  	}
   357  }
   358  
   359  // ensure no txs on reactor after some timeout
   360  func ensureNoTxs(t *testing.T, reactor *Reactor, timeout time.Duration) {
   361  	time.Sleep(timeout) // wait for the txs in all mempools
   362  	assert.Zero(t, reactor.mempool.Size())
   363  }
   364  
   365  func TestMempoolVectors(t *testing.T) {
   366  	testCases := []struct {
   367  		testName string
   368  		tx       []byte
   369  		expBytes string
   370  	}{
   371  		{"tx 1", []byte{123}, "0a030a017b"},
   372  		{"tx 2", []byte("proto encoding in mempool"), "0a1b0a1970726f746f20656e636f64696e6720696e206d656d706f6f6c"},
   373  	}
   374  
   375  	for _, tc := range testCases {
   376  		tc := tc
   377  
   378  		msg := memproto.Message{
   379  			Sum: &memproto.Message_Txs{
   380  				Txs: &memproto.Txs{Txs: [][]byte{tc.tx}},
   381  			},
   382  		}
   383  		bz, err := msg.Marshal()
   384  		require.NoError(t, err, tc.testName)
   385  
   386  		require.Equal(t, tc.expBytes, hex.EncodeToString(bz), tc.testName)
   387  	}
   388  }