test: baseapp: add concurrent test for querying GRPCRouter (#10371)
Runs 1,000 concurrent requests for baseapp.GRPCQueryRouter, this test requires that we enable: go test -race. Closes #10324
This commit is contained in:
parent
fb083f9d34
commit
c0cc0529c4
|
@ -3,6 +3,7 @@ package baseapp_test
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"os"
|
"os"
|
||||||
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"github.com/stretchr/testify/require"
|
"github.com/stretchr/testify/require"
|
||||||
|
@ -74,3 +75,98 @@ func TestRegisterQueryServiceTwice(t *testing.T) {
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tests that we don't have data races per
|
||||||
|
// https://github.com/cosmos/cosmos-sdk/issues/10324
|
||||||
|
// but with the same client connection being used concurrently.
|
||||||
|
func TestQueryDataRaces_sameConnectionToSameHandler(t *testing.T) {
|
||||||
|
var mu sync.Mutex
|
||||||
|
var helper *baseapp.QueryServiceTestHelper
|
||||||
|
makeClientConn := func(qr *baseapp.GRPCQueryRouter) *baseapp.QueryServiceTestHelper {
|
||||||
|
mu.Lock()
|
||||||
|
defer mu.Unlock()
|
||||||
|
|
||||||
|
if helper == nil {
|
||||||
|
helper = &baseapp.QueryServiceTestHelper{
|
||||||
|
GRPCQueryRouter: qr,
|
||||||
|
Ctx: sdk.Context{}.WithContext(context.Background()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return helper
|
||||||
|
}
|
||||||
|
testQueryDataRacesSameHandler(t, makeClientConn)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tests that we don't have data races per
|
||||||
|
// https://github.com/cosmos/cosmos-sdk/issues/10324
|
||||||
|
// but with unique client connections requesting from the same handler concurrently.
|
||||||
|
func TestQueryDataRaces_uniqueConnectionsToSameHandler(t *testing.T) {
|
||||||
|
// Return a new handler for every single call.
|
||||||
|
testQueryDataRacesSameHandler(t, func(qr *baseapp.GRPCQueryRouter) *baseapp.QueryServiceTestHelper {
|
||||||
|
return &baseapp.QueryServiceTestHelper{
|
||||||
|
GRPCQueryRouter: qr,
|
||||||
|
Ctx: sdk.Context{}.WithContext(context.Background()),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func testQueryDataRacesSameHandler(t *testing.T, makeClientConn func(*baseapp.GRPCQueryRouter) *baseapp.QueryServiceTestHelper) {
|
||||||
|
t.Parallel()
|
||||||
|
|
||||||
|
qr := baseapp.NewGRPCQueryRouter()
|
||||||
|
interfaceRegistry := testdata.NewTestInterfaceRegistry()
|
||||||
|
qr.SetInterfaceRegistry(interfaceRegistry)
|
||||||
|
testdata.RegisterQueryServer(qr, testdata.QueryImpl{})
|
||||||
|
|
||||||
|
// The goal is to invoke the router concurrently and check for any data races.
|
||||||
|
// 0. Run with: go test -race
|
||||||
|
// 1. Synchronize every one of the 1,000 goroutines waiting to all query at the
|
||||||
|
// same time.
|
||||||
|
// 2. Once the greenlight is given, perform a query through the router.
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
defer wg.Wait()
|
||||||
|
|
||||||
|
greenlight := make(chan bool)
|
||||||
|
n := 1000
|
||||||
|
ready := make(chan bool, n)
|
||||||
|
go func() {
|
||||||
|
for i := 0; i < n; i++ {
|
||||||
|
<-ready
|
||||||
|
}
|
||||||
|
close(greenlight)
|
||||||
|
}()
|
||||||
|
|
||||||
|
for i := 0; i < n; i++ {
|
||||||
|
wg.Add(1)
|
||||||
|
go func() {
|
||||||
|
defer wg.Done()
|
||||||
|
|
||||||
|
// Wait until we get the green light to start.
|
||||||
|
ready <- true
|
||||||
|
<-greenlight
|
||||||
|
|
||||||
|
client := testdata.NewQueryClient(makeClientConn(qr))
|
||||||
|
res, err := client.Echo(context.Background(), &testdata.EchoRequest{Message: "hello"})
|
||||||
|
require.Nil(t, err)
|
||||||
|
require.NotNil(t, res)
|
||||||
|
require.Equal(t, "hello", res.Message)
|
||||||
|
|
||||||
|
require.Panics(t, func() {
|
||||||
|
_, _ = client.Echo(context.Background(), nil)
|
||||||
|
})
|
||||||
|
|
||||||
|
res2, err := client.SayHello(context.Background(), &testdata.SayHelloRequest{Name: "Foo"})
|
||||||
|
require.Nil(t, err)
|
||||||
|
require.NotNil(t, res)
|
||||||
|
require.Equal(t, "Hello Foo!", res2.Greeting)
|
||||||
|
|
||||||
|
spot := &testdata.Dog{Name: "Spot", Size_: "big"}
|
||||||
|
any, err := types.NewAnyWithValue(spot)
|
||||||
|
require.NoError(t, err)
|
||||||
|
res3, err := client.TestAny(context.Background(), &testdata.TestAnyRequest{AnyAnimal: any})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.NotNil(t, res3)
|
||||||
|
require.Equal(t, spot, res3.HasAnimal.Animal.GetCachedValue())
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue