You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
150 lines
3.0 KiB
Go
150 lines
3.0 KiB
Go
package p2c
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"runtime"
|
|
"strconv"
|
|
"sync"
|
|
"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"
|
|
"google.golang.org/grpc/balancer"
|
|
"google.golang.org/grpc/balancer/base"
|
|
"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{
|
|
FullMethodName: "/",
|
|
Ctx: context.Background(),
|
|
})
|
|
assert.NotNil(t, err)
|
|
}
|
|
|
|
func TestP2cPicker_Pick(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
candidates int
|
|
err error
|
|
threshold float64
|
|
}{
|
|
{
|
|
name: "empty",
|
|
candidates: 0,
|
|
err: balancer.ErrNoSubConnAvailable,
|
|
},
|
|
{
|
|
name: "single",
|
|
candidates: 1,
|
|
threshold: 0.9,
|
|
},
|
|
{
|
|
name: "two",
|
|
candidates: 2,
|
|
threshold: 0.5,
|
|
},
|
|
{
|
|
name: "multiple",
|
|
candidates: 100,
|
|
threshold: 0.95,
|
|
},
|
|
}
|
|
|
|
for _, test := range tests {
|
|
test := test
|
|
t.Run(test.name, func(t *testing.T) {
|
|
t.Parallel()
|
|
|
|
const total = 10000
|
|
builder := new(p2cPickerBuilder)
|
|
ready := make(map[balancer.SubConn]base.SubConnInfo)
|
|
for i := 0; i < test.candidates; i++ {
|
|
ready[mockClientConn{
|
|
id: stringx.Rand(),
|
|
}] = base.SubConnInfo{
|
|
Address: resolver.Address{
|
|
Addr: strconv.Itoa(i),
|
|
},
|
|
}
|
|
}
|
|
|
|
picker := builder.Build(base.PickerBuildInfo{
|
|
ReadySCs: ready,
|
|
})
|
|
var wg sync.WaitGroup
|
|
wg.Add(total)
|
|
for i := 0; i < total; i++ {
|
|
result, err := picker.Pick(balancer.PickInfo{
|
|
FullMethodName: "/",
|
|
Ctx: context.Background(),
|
|
})
|
|
assert.Equal(t, test.err, err)
|
|
|
|
if test.err != nil {
|
|
return
|
|
}
|
|
|
|
if i%100 == 0 {
|
|
err = status.Error(codes.DeadlineExceeded, "deadline")
|
|
}
|
|
|
|
go func() {
|
|
runtime.Gosched()
|
|
result.Done(balancer.DoneInfo{
|
|
Err: err,
|
|
})
|
|
wg.Done()
|
|
}()
|
|
}
|
|
|
|
wg.Wait()
|
|
dist := make(map[any]int)
|
|
conns := picker.(*p2cPicker).conns
|
|
for _, conn := range conns {
|
|
dist[conn.addr.Addr] = int(conn.requests)
|
|
}
|
|
|
|
entropy := mathx.CalcEntropy(dist)
|
|
assert.True(t, entropy > test.threshold, fmt.Sprintf("entropy is %f, less than %f",
|
|
entropy, test.threshold))
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestPickerWithEmptyConns(t *testing.T) {
|
|
var picker p2cPicker
|
|
_, err := picker.Pick(balancer.PickInfo{
|
|
FullMethodName: "/",
|
|
Ctx: context.Background(),
|
|
})
|
|
assert.ErrorIs(t, err, balancer.ErrNoSubConnAvailable)
|
|
}
|
|
|
|
type mockClientConn struct {
|
|
// add random string member to avoid map key equality.
|
|
id string
|
|
}
|
|
|
|
func (m mockClientConn) GetOrBuildProducer(builder balancer.ProducerBuilder) (
|
|
p balancer.Producer, close func()) {
|
|
return builder.Build(m)
|
|
}
|
|
|
|
func (m mockClientConn) UpdateAddresses(addresses []resolver.Address) {
|
|
}
|
|
|
|
func (m mockClientConn) Connect() {
|
|
}
|