Files
tendermint/internal/p2p/channel_test.go
Sam Kleinman d5fb82e414 p2p: make p2p.Channel an interface (#8967)
This is (#8446) pulled from the `main/libp2p` branch but without any
of the libp2p content, and is perhaps the easiest first step to enable
pluggability at the peer layer, and makes it possible hoist shims
(including for, say 0.34) into tendermint without touching the reactors.
2022-07-11 20:22:40 +00:00

222 lines
5.1 KiB
Go

package p2p
import (
"context"
"errors"
"testing"
"time"
"github.com/fortytw2/leaktest"
"github.com/stretchr/testify/require"
)
type channelInternal struct {
In chan Envelope
Out chan Envelope
Error chan PeerError
}
func testChannel(size int) (*channelInternal, *legacyChannel) {
in := &channelInternal{
In: make(chan Envelope, size),
Out: make(chan Envelope, size),
Error: make(chan PeerError, size),
}
ch := &legacyChannel{
inCh: in.In,
outCh: in.Out,
errCh: in.Error,
}
return in, ch
}
func TestChannel(t *testing.T) {
t.Cleanup(leaktest.Check(t))
bctx, bcancel := context.WithCancel(context.Background())
defer bcancel()
testCases := []struct {
Name string
Case func(context.Context, *testing.T)
}{
{
Name: "Send",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
require.NoError(t, ch.Send(ctx, Envelope{From: "kip", To: "merlin"}))
res, ok := <-ins.Out
require.True(t, ok)
require.EqualValues(t, "kip", res.From)
require.EqualValues(t, "merlin", res.To)
},
},
{
Name: "SendError",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
require.NoError(t, ch.SendError(ctx, PeerError{NodeID: "kip", Err: errors.New("merlin")}))
res, ok := <-ins.Error
require.True(t, ok)
require.EqualValues(t, "kip", res.NodeID)
require.EqualValues(t, "merlin", res.Err.Error())
},
},
{
Name: "SendWithCanceledContext",
Case: func(ctx context.Context, t *testing.T) {
_, ch := testChannel(0)
cctx, ccancel := context.WithCancel(ctx)
ccancel()
require.Error(t, ch.Send(cctx, Envelope{From: "kip", To: "merlin"}))
},
},
{
Name: "SendErrorWithCanceledContext",
Case: func(ctx context.Context, t *testing.T) {
_, ch := testChannel(0)
cctx, ccancel := context.WithCancel(ctx)
ccancel()
require.Error(t, ch.SendError(cctx, PeerError{NodeID: "kip", Err: errors.New("merlin")}))
},
},
{
Name: "ReceiveEmptyIteratorBlocks",
Case: func(ctx context.Context, t *testing.T) {
_, ch := testChannel(1)
iter := ch.Receive(ctx)
require.NotNil(t, iter)
out := make(chan bool)
go func() {
defer close(out)
select {
case <-ctx.Done():
case out <- iter.Next(ctx):
}
}()
select {
case <-time.After(10 * time.Millisecond):
case <-out:
require.Fail(t, "iterator should not advance")
}
require.Nil(t, iter.Envelope())
},
},
{
Name: "ReceiveWithData",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
ins.In <- Envelope{From: "kip", To: "merlin"}
iter := ch.Receive(ctx)
require.NotNil(t, iter)
require.True(t, iter.Next(ctx))
res := iter.Envelope()
require.EqualValues(t, "kip", res.From)
require.EqualValues(t, "merlin", res.To)
},
},
{
Name: "ReceiveWithCanceledContext",
Case: func(ctx context.Context, t *testing.T) {
_, ch := testChannel(0)
cctx, ccancel := context.WithCancel(ctx)
ccancel()
iter := ch.Receive(cctx)
require.NotNil(t, iter)
require.False(t, iter.Next(cctx))
require.Nil(t, iter.Envelope())
},
},
{
Name: "IteratorWithCanceledContext",
Case: func(ctx context.Context, t *testing.T) {
_, ch := testChannel(0)
iter := ch.Receive(ctx)
require.NotNil(t, iter)
cctx, ccancel := context.WithCancel(ctx)
ccancel()
require.False(t, iter.Next(cctx))
require.Nil(t, iter.Envelope())
},
},
{
Name: "IteratorCanceledAfterFirstUseBecomesNil",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
ins.In <- Envelope{From: "kip", To: "merlin"}
iter := ch.Receive(ctx)
require.NotNil(t, iter)
require.True(t, iter.Next(ctx))
res := iter.Envelope()
require.EqualValues(t, "kip", res.From)
require.EqualValues(t, "merlin", res.To)
cctx, ccancel := context.WithCancel(ctx)
ccancel()
require.False(t, iter.Next(cctx))
require.Nil(t, iter.Envelope())
},
},
{
Name: "IteratorMultipleNextCalls",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
ins.In <- Envelope{From: "kip", To: "merlin"}
iter := ch.Receive(ctx)
require.NotNil(t, iter)
require.True(t, iter.Next(ctx))
res := iter.Envelope()
require.EqualValues(t, "kip", res.From)
require.EqualValues(t, "merlin", res.To)
res1 := iter.Envelope()
require.Equal(t, res, res1)
},
},
{
Name: "IteratorProducesNilObjectBeforeNext",
Case: func(ctx context.Context, t *testing.T) {
ins, ch := testChannel(1)
iter := ch.Receive(ctx)
require.NotNil(t, iter)
require.Nil(t, iter.Envelope())
ins.In <- Envelope{From: "kip", To: "merlin"}
require.NotNil(t, iter)
require.True(t, iter.Next(ctx))
res := iter.Envelope()
require.NotNil(t, res)
require.EqualValues(t, "kip", res.From)
require.EqualValues(t, "merlin", res.To)
},
},
}
for _, tc := range testCases {
t.Run(tc.Name, func(t *testing.T) {
t.Cleanup(leaktest.Check(t))
ctx, cancel := context.WithCancel(bctx)
defer cancel()
tc.Case(ctx, t)
})
}
}