Browse Source

consensus: test proposal heartbeat

pull/606/head
Ethan Buchman 7 years ago
parent
commit
c5a657f540
6 changed files with 44 additions and 7 deletions
  1. +4
    -1
      consensus/common_test.go
  2. +3
    -0
      consensus/reactor.go
  3. +32
    -1
      consensus/reactor_test.go
  4. +1
    -1
      mempool/mempool.go
  5. +2
    -2
      mempool/mempool_test.go
  6. +2
    -2
      types/services.go

+ 4
- 1
consensus/common_test.go View File

@ -332,7 +332,7 @@ func consensusLogger() log.Logger {
}) })
} }
func randConsensusNet(nValidators int, testName string, tickerFunc func() TimeoutTicker, appFunc func() abci.Application) []*ConsensusState {
func randConsensusNet(nValidators int, testName string, tickerFunc func() TimeoutTicker, appFunc func() abci.Application, configOpts ...func(*cfg.Config)) []*ConsensusState {
genDoc, privVals := randGenesisDoc(nValidators, false, 10) genDoc, privVals := randGenesisDoc(nValidators, false, 10)
css := make([]*ConsensusState, nValidators) css := make([]*ConsensusState, nValidators)
logger := consensusLogger() logger := consensusLogger()
@ -342,6 +342,9 @@ func randConsensusNet(nValidators int, testName string, tickerFunc func() Timeou
state.SetLogger(logger.With("module", "state", "validator", i)) state.SetLogger(logger.With("module", "state", "validator", i))
state.Save() state.Save()
thisConfig := ResetConfig(Fmt("%s_%d", testName, i)) thisConfig := ResetConfig(Fmt("%s_%d", testName, i))
for _, opt := range configOpts {
opt(thisConfig)
}
ensureDir(path.Dir(thisConfig.Consensus.WalFile()), 0700) // dir for wal ensureDir(path.Dir(thisConfig.Consensus.WalFile()), 0700) // dir for wal
css[i] = newConsensusStateWithConfig(thisConfig, state, privVals[i], appFunc()) css[i] = newConsensusStateWithConfig(thisConfig, state, privVals[i], appFunc())
css[i].SetLogger(logger.With("validator", i)) css[i].SetLogger(logger.With("validator", i))


+ 3
- 0
consensus/reactor.go View File

@ -1178,6 +1178,8 @@ const (
msgTypeHasVote = byte(0x15) msgTypeHasVote = byte(0x15)
msgTypeVoteSetMaj23 = byte(0x16) msgTypeVoteSetMaj23 = byte(0x16)
msgTypeVoteSetBits = byte(0x17) msgTypeVoteSetBits = byte(0x17)
msgTypeProposalHeartbeat = byte(0x20)
) )
// ConsensusMessage is a message that can be sent and received on the ConsensusReactor // ConsensusMessage is a message that can be sent and received on the ConsensusReactor
@ -1194,6 +1196,7 @@ var _ = wire.RegisterInterface(
wire.ConcreteType{&HasVoteMessage{}, msgTypeHasVote}, wire.ConcreteType{&HasVoteMessage{}, msgTypeHasVote},
wire.ConcreteType{&VoteSetMaj23Message{}, msgTypeVoteSetMaj23}, wire.ConcreteType{&VoteSetMaj23Message{}, msgTypeVoteSetMaj23},
wire.ConcreteType{&VoteSetBitsMessage{}, msgTypeVoteSetBits}, wire.ConcreteType{&VoteSetBitsMessage{}, msgTypeVoteSetBits},
wire.ConcreteType{&ProposalHeartbeatMessage{}, msgTypeProposalHeartbeat},
) )
// DecodeMessage decodes the given bytes into a ConsensusMessage. // DecodeMessage decodes the given bytes into a ConsensusMessage.


+ 32
- 1
consensus/reactor_test.go View File

@ -7,9 +7,11 @@ import (
"time" "time"
"github.com/tendermint/abci/example/dummy" "github.com/tendermint/abci/example/dummy"
"github.com/tendermint/tmlibs/events"
cfg "github.com/tendermint/tendermint/config"
"github.com/tendermint/tendermint/p2p" "github.com/tendermint/tendermint/p2p"
"github.com/tendermint/tendermint/types" "github.com/tendermint/tendermint/types"
"github.com/tendermint/tmlibs/events"
) )
func init() { func init() {
@ -76,6 +78,35 @@ func TestReactor(t *testing.T) {
}, css) }, css)
} }
// Ensure a testnet sends proposal heartbeats and makes blocks when there are txs
func TestReactorProposalHeartbeats(t *testing.T) {
N := 4
css := randConsensusNet(N, "consensus_reactor_test", newMockTickerFunc(true), newCounter,
func(c *cfg.Config) {
c.Consensus.CreateEmptyBlocks = false
})
reactors, eventChans := startConsensusNet(t, css, N, false)
defer stopConsensusNet(reactors)
heartbeatChans := make([]chan interface{}, N)
for i := 0; i < N; i++ {
heartbeatChans[i] = subscribeToEvent(css[i].evsw, "tester", types.EventStringProposalHeartbeat(), 1)
}
// wait till everyone sends a proposal heartbeat
timeoutWaitGroup(t, N, func(wg *sync.WaitGroup, j int) {
<-heartbeatChans[j]
wg.Done()
}, css)
// send a tx
css[3].mempool.CheckTx([]byte{1, 2, 3}, nil)
// wait till everyone makes the first new block
timeoutWaitGroup(t, N, func(wg *sync.WaitGroup, j int) {
<-eventChans[j]
wg.Done()
}, css)
}
//------------------------------------------------------------- //-------------------------------------------------------------
// ensure we can make blocks despite cycling a validator set // ensure we can make blocks despite cycling a validator set


+ 1
- 1
mempool/mempool.go View File

@ -277,7 +277,7 @@ func (mem *Mempool) resCbRecheck(req *abci.Request, res *abci.Response) {
// TxsAvailable returns a channel which fires once for every height, // TxsAvailable returns a channel which fires once for every height,
// and only when transactions are available in the mempool. // and only when transactions are available in the mempool.
// NOTE: the returned channel may be nil if EnableTxsAvailable was not called. // NOTE: the returned channel may be nil if EnableTxsAvailable was not called.
func (mem *Mempool) TxsAvailable() chan int {
func (mem *Mempool) TxsAvailable() <-chan int {
return mem.txsAvailable return mem.txsAvailable
} }


+ 2
- 2
mempool/mempool_test.go View File

@ -28,7 +28,7 @@ func newMempoolWithApp(t *testing.T, cc proxy.ClientCreator) *Mempool {
return mempool return mempool
} }
func ensureNoFire(t *testing.T, ch chan int, timeoutMS int) {
func ensureNoFire(t *testing.T, ch <-chan int, timeoutMS int) {
timer := time.NewTimer(time.Duration(timeoutMS) * time.Millisecond) timer := time.NewTimer(time.Duration(timeoutMS) * time.Millisecond)
select { select {
case <-ch: case <-ch:
@ -37,7 +37,7 @@ func ensureNoFire(t *testing.T, ch chan int, timeoutMS int) {
} }
} }
func ensureFire(t *testing.T, ch chan int, timeoutMS int) {
func ensureFire(t *testing.T, ch <-chan int, timeoutMS int) {
timer := time.NewTimer(time.Duration(timeoutMS) * time.Millisecond) timer := time.NewTimer(time.Duration(timeoutMS) * time.Millisecond)
select { select {
case <-ch: case <-ch:


+ 2
- 2
types/services.go View File

@ -24,7 +24,7 @@ type Mempool interface {
Update(height int, txs Txs) Update(height int, txs Txs)
Flush() Flush()
TxsAvailable() chan int
TxsAvailable() <-chan int
EnableTxsAvailable() EnableTxsAvailable()
} }
@ -38,7 +38,7 @@ func (m MockMempool) CheckTx(tx Tx, cb func(*abci.Response)) error { return nil
func (m MockMempool) Reap(n int) Txs { return Txs{} } func (m MockMempool) Reap(n int) Txs { return Txs{} }
func (m MockMempool) Update(height int, txs Txs) {} func (m MockMempool) Update(height int, txs Txs) {}
func (m MockMempool) Flush() {} func (m MockMempool) Flush() {}
func (m MockMempool) TxsAvailable() chan int { return make(chan int) }
func (m MockMempool) TxsAvailable() <-chan int { return make(chan int) }
func (m MockMempool) EnableTxsAvailable() {} func (m MockMempool) EnableTxsAvailable() {}
//------------------------------------------------------ //------------------------------------------------------


Loading…
Cancel
Save