go-zero/zrpc/internal/balancer/p2c/p2c_test.go

141 lines
2.8 KiB
Go
Raw Normal View History

2020-08-06 20:55:38 +08:00
package p2c
2020-08-07 15:28:40 +08:00
import (
"context"
"fmt"
2020-08-21 23:24:07 +08:00
"runtime"
2020-08-07 15:28:40 +08:00
"strconv"
2020-08-20 15:35:13 +08:00
"sync"
2020-08-07 15:28:40 +08:00
"testing"
"github.com/stretchr/testify/assert"
"github.com/zeromicro/go-zero/core/logx"
"github.com/zeromicro/go-zero/core/mathx"
"github.com/zeromicro/go-zero/core/stringx"
2020-08-07 15:28:40 +08:00
"google.golang.org/grpc/balancer"
"google.golang.org/grpc/balancer/base"
2020-08-07 15:28:40 +08:00
"google.golang.org/grpc/codes"
"google.golang.org/grpc/resolver"
"google.golang.org/grpc/status"
)
func init() {
logx.Disable()
}
func TestP2cPicker_PickNil(t *testing.T) {
builder := new(p2cPickerBuilder)
picker := builder.Build(base.PickerBuildInfo{})
_, err := picker.Pick(balancer.PickInfo{
2020-08-07 15:28:40 +08:00
FullMethodName: "/",
Ctx: context.Background(),
})
assert.NotNil(t, err)
}
2020-08-06 20:55:38 +08:00
func TestP2cPicker_Pick(t *testing.T) {
2020-08-07 15:28:40 +08:00
tests := []struct {
name string
candidates int
2021-08-17 17:09:26 +08:00
err error
2020-08-21 23:24:07 +08:00
threshold float64
2020-08-07 15:28:40 +08:00
}{
2021-08-17 17:09:26 +08:00
{
name: "empty",
candidates: 0,
err: balancer.ErrNoSubConnAvailable,
},
2020-08-07 15:28:40 +08:00
{
name: "single",
candidates: 1,
2020-08-21 23:24:07 +08:00
threshold: 0.9,
2020-08-07 15:28:40 +08:00
},
2020-08-20 15:35:13 +08:00
{
name: "two",
candidates: 2,
2020-08-21 23:24:07 +08:00
threshold: 0.5,
2020-08-20 15:35:13 +08:00
},
2020-08-07 15:28:40 +08:00
{
name: "multiple",
candidates: 100,
2020-08-21 23:24:07 +08:00
threshold: 0.95,
2020-08-07 15:28:40 +08:00
},
}
for _, test := range tests {
2020-08-21 16:57:17 +08:00
test := test
2020-08-07 15:28:40 +08:00
t.Run(test.name, func(t *testing.T) {
2020-08-21 16:57:17 +08:00
t.Parallel()
2020-08-21 23:15:45 +08:00
2020-08-21 23:24:07 +08:00
const total = 10000
2020-08-07 15:28:40 +08:00
builder := new(p2cPickerBuilder)
ready := make(map[balancer.SubConn]base.SubConnInfo)
2020-08-07 15:28:40 +08:00
for i := 0; i < test.candidates; i++ {
2021-08-17 17:09:26 +08:00
ready[mockClientConn{
id: stringx.Rand(),
}] = base.SubConnInfo{
Address: resolver.Address{
Addr: strconv.Itoa(i),
},
}
2020-08-07 15:28:40 +08:00
}
picker := builder.Build(base.PickerBuildInfo{
ReadySCs: ready,
})
2020-08-20 15:35:13 +08:00
var wg sync.WaitGroup
wg.Add(total)
for i := 0; i < total; i++ {
result, err := picker.Pick(balancer.PickInfo{
2020-08-07 15:28:40 +08:00
FullMethodName: "/",
Ctx: context.Background(),
})
2021-08-17 17:09:26 +08:00
assert.Equal(t, test.err, err)
if test.err != nil {
return
}
2020-08-07 15:28:40 +08:00
if i%100 == 0 {
err = status.Error(codes.DeadlineExceeded, "deadline")
}
2021-08-17 17:09:26 +08:00
2020-08-20 15:35:13 +08:00
go func() {
2020-08-21 23:24:07 +08:00
runtime.Gosched()
result.Done(balancer.DoneInfo{
2020-08-20 15:35:13 +08:00
Err: err,
})
wg.Done()
}()
2020-08-07 15:28:40 +08:00
}
2020-08-20 15:35:13 +08:00
wg.Wait()
dist := make(map[any]int)
2020-08-07 15:28:40 +08:00
conns := picker.(*p2cPicker).conns
for _, conn := range conns {
dist[conn.addr.Addr] = int(conn.requests)
}
entropy := mathx.CalcEntropy(dist)
2020-08-21 23:24:07 +08:00
assert.True(t, entropy > test.threshold, fmt.Sprintf("entropy is %f, less than %f",
entropy, test.threshold))
2020-08-07 15:28:40 +08:00
})
}
}
2021-08-17 17:09:26 +08:00
type mockClientConn struct {
// add random string member to avoid map key equality.
id string
}
2020-08-07 15:28:40 +08:00
func (m mockClientConn) GetOrBuildProducer(builder balancer.ProducerBuilder) (
p balancer.Producer, close func()) {
return builder.Build(m)
}
2020-08-07 15:28:40 +08:00
func (m mockClientConn) UpdateAddresses(addresses []resolver.Address) {
}
2020-08-06 20:55:38 +08:00
2020-08-07 15:28:40 +08:00
func (m mockClientConn) Connect() {
2020-08-06 20:55:38 +08:00
}