- package conn
-
- import (
- "encoding/hex"
- "net"
- "testing"
- "time"
-
- "github.com/fortytw2/leaktest"
- "github.com/gogo/protobuf/proto"
- "github.com/stretchr/testify/assert"
- "github.com/stretchr/testify/require"
-
- "github.com/tendermint/tendermint/libs/log"
- "github.com/tendermint/tendermint/libs/protoio"
- tmp2p "github.com/tendermint/tendermint/proto/tendermint/p2p"
- "github.com/tendermint/tendermint/proto/tendermint/types"
- )
-
- const maxPingPongPacketSize = 1024 // bytes
-
- func createTestMConnection(conn net.Conn) *MConnection {
- onReceive := func(chID byte, msgBytes []byte) {
- }
- onError := func(r interface{}) {
- }
- c := createMConnectionWithCallbacks(conn, onReceive, onError)
- c.SetLogger(log.TestingLogger())
- return c
- }
-
- func createMConnectionWithCallbacks(
- conn net.Conn,
- onReceive func(chID byte, msgBytes []byte),
- onError func(r interface{}),
- ) *MConnection {
- cfg := DefaultMConnConfig()
- cfg.PingInterval = 90 * time.Millisecond
- cfg.PongTimeout = 45 * time.Millisecond
- chDescs := []*ChannelDescriptor{{ID: 0x01, Priority: 1, SendQueueCapacity: 1}}
- c := NewMConnectionWithConfig(conn, chDescs, onReceive, onError, cfg)
- c.SetLogger(log.TestingLogger())
- return c
- }
-
- func TestMConnectionSendFlushStop(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- clientConn := createTestMConnection(client)
- err := clientConn.Start()
- require.Nil(t, err)
- defer clientConn.Stop() // nolint:errcheck // ignore for tests
-
- msg := []byte("abc")
- assert.True(t, clientConn.Send(0x01, msg))
-
- msgLength := 14
-
- // start the reader in a new routine, so we can flush
- errCh := make(chan error)
- go func() {
- msgB := make([]byte, msgLength)
- _, err := server.Read(msgB)
- if err != nil {
- t.Error(err)
- return
- }
- errCh <- err
- }()
-
- // stop the conn - it should flush all conns
- clientConn.FlushStop()
-
- timer := time.NewTimer(3 * time.Second)
- select {
- case <-errCh:
- case <-timer.C:
- t.Error("timed out waiting for msgs to be read")
- }
- }
-
- func TestMConnectionSend(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- mconn := createTestMConnection(client)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- msg := []byte("Ant-Man")
- assert.True(t, mconn.Send(0x01, msg))
- // Note: subsequent Send/TrySend calls could pass because we are reading from
- // the send queue in a separate goroutine.
- _, err = server.Read(make([]byte, len(msg)))
- if err != nil {
- t.Error(err)
- }
- assert.True(t, mconn.CanSend(0x01))
-
- msg = []byte("Spider-Man")
- assert.True(t, mconn.TrySend(0x01, msg))
- _, err = server.Read(make([]byte, len(msg)))
- if err != nil {
- t.Error(err)
- }
-
- assert.False(t, mconn.CanSend(0x05), "CanSend should return false because channel is unknown")
- assert.False(t, mconn.Send(0x05, []byte("Absorbing Man")), "Send should return false because channel is unknown")
- }
-
- func TestMConnectionReceive(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn1 := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn1.Start()
- require.Nil(t, err)
- defer mconn1.Stop() // nolint:errcheck // ignore for tests
-
- mconn2 := createTestMConnection(server)
- err = mconn2.Start()
- require.Nil(t, err)
- defer mconn2.Stop() // nolint:errcheck // ignore for tests
-
- msg := []byte("Cyclops")
- assert.True(t, mconn2.Send(0x01, msg))
-
- select {
- case receivedBytes := <-receivedCh:
- assert.Equal(t, msg, receivedBytes)
- case err := <-errorsCh:
- t.Fatalf("Expected %s, got %+v", msg, err)
- case <-time.After(500 * time.Millisecond):
- t.Fatalf("Did not receive %s message in 500ms", msg)
- }
- }
-
- func TestMConnectionStatus(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- mconn := createTestMConnection(client)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- status := mconn.Status()
- assert.NotNil(t, status)
- assert.Zero(t, status.Channels[0].SendQueueSize)
- }
-
- func TestMConnectionPongTimeoutResultsInError(t *testing.T) {
- server, client := net.Pipe()
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- serverGotPing := make(chan struct{})
- go func() {
- // read ping
- var pkt tmp2p.Packet
- err := protoio.NewDelimitedReader(server, maxPingPongPacketSize).ReadMsg(&pkt)
- require.NoError(t, err)
- serverGotPing <- struct{}{}
- }()
- <-serverGotPing
-
- pongTimerExpired := mconn.config.PongTimeout + 200*time.Millisecond
- select {
- case msgBytes := <-receivedCh:
- t.Fatalf("Expected error, but got %v", msgBytes)
- case err := <-errorsCh:
- assert.NotNil(t, err)
- case <-time.After(pongTimerExpired):
- t.Fatalf("Expected to receive error after %v", pongTimerExpired)
- }
- }
-
- func TestMConnectionMultiplePongsInTheBeginning(t *testing.T) {
- server, client := net.Pipe()
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- // sending 3 pongs in a row (abuse)
- protoWriter := protoio.NewDelimitedWriter(server)
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
-
- serverGotPing := make(chan struct{})
- go func() {
- // read ping (one byte)
- var packet tmp2p.Packet
- err := protoio.NewDelimitedReader(server, maxPingPongPacketSize).ReadMsg(&packet)
- require.NoError(t, err)
- serverGotPing <- struct{}{}
-
- // respond with pong
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
- }()
- <-serverGotPing
-
- pongTimerExpired := mconn.config.PongTimeout + 20*time.Millisecond
- select {
- case msgBytes := <-receivedCh:
- t.Fatalf("Expected no data, but got %v", msgBytes)
- case err := <-errorsCh:
- t.Fatalf("Expected no error, but got %v", err)
- case <-time.After(pongTimerExpired):
- assert.True(t, mconn.IsRunning())
- }
- }
-
- func TestMConnectionMultiplePings(t *testing.T) {
- server, client := net.Pipe()
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- // sending 3 pings in a row (abuse)
- // see https://github.com/tendermint/tendermint/issues/1190
- protoReader := protoio.NewDelimitedReader(server, maxPingPongPacketSize)
- protoWriter := protoio.NewDelimitedWriter(server)
- var pkt tmp2p.Packet
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPing{}))
- require.NoError(t, err)
-
- err = protoReader.ReadMsg(&pkt)
- require.NoError(t, err)
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPing{}))
- require.NoError(t, err)
-
- err = protoReader.ReadMsg(&pkt)
- require.NoError(t, err)
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPing{}))
- require.NoError(t, err)
-
- err = protoReader.ReadMsg(&pkt)
- require.NoError(t, err)
-
- assert.True(t, mconn.IsRunning())
- }
-
- func TestMConnectionPingPongs(t *testing.T) {
- // check that we are not leaking any go-routines
- defer leaktest.CheckTimeout(t, 10*time.Second)()
-
- server, client := net.Pipe()
-
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- serverGotPing := make(chan struct{})
- go func() {
- protoReader := protoio.NewDelimitedReader(server, maxPingPongPacketSize)
- protoWriter := protoio.NewDelimitedWriter(server)
- var pkt tmp2p.PacketPing
-
- // read ping
- err = protoReader.ReadMsg(&pkt)
- require.NoError(t, err)
- serverGotPing <- struct{}{}
-
- // respond with pong
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
-
- time.Sleep(mconn.config.PingInterval)
-
- // read ping
- err = protoReader.ReadMsg(&pkt)
- require.NoError(t, err)
- serverGotPing <- struct{}{}
-
- // respond with pong
- _, err = protoWriter.WriteMsg(mustWrapPacket(&tmp2p.PacketPong{}))
- require.NoError(t, err)
- }()
- <-serverGotPing
- <-serverGotPing
-
- pongTimerExpired := (mconn.config.PongTimeout + 20*time.Millisecond) * 2
- select {
- case msgBytes := <-receivedCh:
- t.Fatalf("Expected no data, but got %v", msgBytes)
- case err := <-errorsCh:
- t.Fatalf("Expected no error, but got %v", err)
- case <-time.After(2 * pongTimerExpired):
- assert.True(t, mconn.IsRunning())
- }
- }
-
- func TestMConnectionStopsAndReturnsError(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- receivedCh := make(chan []byte)
- errorsCh := make(chan interface{})
- onReceive := func(chID byte, msgBytes []byte) {
- receivedCh <- msgBytes
- }
- onError := func(r interface{}) {
- errorsCh <- r
- }
- mconn := createMConnectionWithCallbacks(client, onReceive, onError)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- if err := client.Close(); err != nil {
- t.Error(err)
- }
-
- select {
- case receivedBytes := <-receivedCh:
- t.Fatalf("Expected error, got %v", receivedBytes)
- case err := <-errorsCh:
- assert.NotNil(t, err)
- assert.False(t, mconn.IsRunning())
- case <-time.After(500 * time.Millisecond):
- t.Fatal("Did not receive error in 500ms")
- }
- }
-
- func newClientAndServerConnsForReadErrors(t *testing.T, chOnErr chan struct{}) (*MConnection, *MConnection) {
- server, client := NetPipe()
-
- onReceive := func(chID byte, msgBytes []byte) {}
- onError := func(r interface{}) {}
-
- // create client conn with two channels
- chDescs := []*ChannelDescriptor{
- {ID: 0x01, Priority: 1, SendQueueCapacity: 1},
- {ID: 0x02, Priority: 1, SendQueueCapacity: 1},
- }
- mconnClient := NewMConnection(client, chDescs, onReceive, onError)
- mconnClient.SetLogger(log.TestingLogger().With("module", "client"))
- err := mconnClient.Start()
- require.Nil(t, err)
-
- // create server conn with 1 channel
- // it fires on chOnErr when there's an error
- serverLogger := log.TestingLogger().With("module", "server")
- onError = func(r interface{}) {
- chOnErr <- struct{}{}
- }
- mconnServer := createMConnectionWithCallbacks(server, onReceive, onError)
- mconnServer.SetLogger(serverLogger)
- err = mconnServer.Start()
- require.Nil(t, err)
- return mconnClient, mconnServer
- }
-
- func expectSend(ch chan struct{}) bool {
- after := time.After(time.Second * 5)
- select {
- case <-ch:
- return true
- case <-after:
- return false
- }
- }
-
- func TestMConnectionReadErrorBadEncoding(t *testing.T) {
- chOnErr := make(chan struct{})
- mconnClient, mconnServer := newClientAndServerConnsForReadErrors(t, chOnErr)
-
- client := mconnClient.conn
-
- // Write it.
- _, err := client.Write([]byte{1, 2, 3, 4, 5})
- require.NoError(t, err)
- assert.True(t, expectSend(chOnErr), "badly encoded msgPacket")
-
- t.Cleanup(func() {
- if err := mconnClient.Stop(); err != nil {
- t.Log(err)
- }
- })
-
- t.Cleanup(func() {
- if err := mconnServer.Stop(); err != nil {
- t.Log(err)
- }
- })
- }
-
- func TestMConnectionReadErrorUnknownChannel(t *testing.T) {
- chOnErr := make(chan struct{})
- mconnClient, mconnServer := newClientAndServerConnsForReadErrors(t, chOnErr)
-
- msg := []byte("Ant-Man")
-
- // fail to send msg on channel unknown by client
- assert.False(t, mconnClient.Send(0x03, msg))
-
- // send msg on channel unknown by the server.
- // should cause an error
- assert.True(t, mconnClient.Send(0x02, msg))
- assert.True(t, expectSend(chOnErr), "unknown channel")
-
- t.Cleanup(func() {
- if err := mconnClient.Stop(); err != nil {
- t.Log(err)
- }
- })
-
- t.Cleanup(func() {
- if err := mconnServer.Stop(); err != nil {
- t.Log(err)
- }
- })
- }
-
- func TestMConnectionReadErrorLongMessage(t *testing.T) {
- chOnErr := make(chan struct{})
- chOnRcv := make(chan struct{})
-
- mconnClient, mconnServer := newClientAndServerConnsForReadErrors(t, chOnErr)
- defer mconnClient.Stop() // nolint:errcheck // ignore for tests
- defer mconnServer.Stop() // nolint:errcheck // ignore for tests
-
- mconnServer.onReceive = func(chID byte, msgBytes []byte) {
- chOnRcv <- struct{}{}
- }
-
- client := mconnClient.conn
- protoWriter := protoio.NewDelimitedWriter(client)
-
- // send msg thats just right
- var packet = tmp2p.PacketMsg{
- ChannelID: 0x01,
- EOF: true,
- Data: make([]byte, mconnClient.config.MaxPacketMsgPayloadSize),
- }
-
- _, err := protoWriter.WriteMsg(mustWrapPacket(&packet))
- require.NoError(t, err)
- assert.True(t, expectSend(chOnRcv), "msg just right")
-
- // send msg thats too long
- packet = tmp2p.PacketMsg{
- ChannelID: 0x01,
- EOF: true,
- Data: make([]byte, mconnClient.config.MaxPacketMsgPayloadSize+100),
- }
-
- _, err = protoWriter.WriteMsg(mustWrapPacket(&packet))
- require.Error(t, err)
- assert.True(t, expectSend(chOnErr), "msg too long")
- }
-
- func TestMConnectionReadErrorUnknownMsgType(t *testing.T) {
- chOnErr := make(chan struct{})
- mconnClient, mconnServer := newClientAndServerConnsForReadErrors(t, chOnErr)
- defer mconnClient.Stop() // nolint:errcheck // ignore for tests
- defer mconnServer.Stop() // nolint:errcheck // ignore for tests
-
- // send msg with unknown msg type
- _, err := protoio.NewDelimitedWriter(mconnClient.conn).WriteMsg(&types.Header{ChainID: "x"})
- require.NoError(t, err)
- assert.True(t, expectSend(chOnErr), "unknown msg type")
- }
-
- func TestMConnectionTrySend(t *testing.T) {
- server, client := NetPipe()
- defer server.Close()
- defer client.Close()
-
- mconn := createTestMConnection(client)
- err := mconn.Start()
- require.Nil(t, err)
- defer mconn.Stop() // nolint:errcheck // ignore for tests
-
- msg := []byte("Semicolon-Woman")
- resultCh := make(chan string, 2)
- assert.True(t, mconn.TrySend(0x01, msg))
- _, err = server.Read(make([]byte, len(msg)))
- require.NoError(t, err)
- assert.True(t, mconn.CanSend(0x01))
- assert.True(t, mconn.TrySend(0x01, msg))
- assert.False(t, mconn.CanSend(0x01))
- go func() {
- mconn.TrySend(0x01, msg)
- resultCh <- "TrySend"
- }()
- assert.False(t, mconn.CanSend(0x01))
- assert.False(t, mconn.TrySend(0x01, msg))
- assert.Equal(t, "TrySend", <-resultCh)
- }
-
- // nolint:lll //ignore line length for tests
- func TestConnVectors(t *testing.T) {
-
- testCases := []struct {
- testName string
- msg proto.Message
- expBytes string
- }{
- {"PacketPing", &tmp2p.PacketPing{}, "0a00"},
- {"PacketPong", &tmp2p.PacketPong{}, "1200"},
- {"PacketMsg", &tmp2p.PacketMsg{ChannelID: 1, EOF: false, Data: []byte("data transmitted over the wire")}, "1a2208011a1e64617461207472616e736d6974746564206f766572207468652077697265"},
- }
-
- for _, tc := range testCases {
- tc := tc
-
- pm := mustWrapPacket(tc.msg)
- bz, err := pm.Marshal()
- require.NoError(t, err, tc.testName)
-
- require.Equal(t, tc.expBytes, hex.EncodeToString(bz), tc.testName)
- }
- }
|