123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113 |
- package p2c
- import (
- "context"
- "fmt"
- "runtime"
- "strconv"
- "sync"
- "testing"
- "github.com/stretchr/testify/assert"
- "github.com/tal-tech/go-zero/core/logx"
- "github.com/tal-tech/go-zero/core/mathx"
- "google.golang.org/grpc/balancer"
- "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(nil)
- _, _, err := picker.Pick(context.Background(), balancer.PickInfo{
- FullMethodName: "/",
- Ctx: context.Background(),
- })
- assert.NotNil(t, err)
- }
- func TestP2cPicker_Pick(t *testing.T) {
- tests := []struct {
- name string
- candidates int
- threshold float64
- }{
- {
- 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[resolver.Address]balancer.SubConn)
- for i := 0; i < test.candidates; i++ {
- ready[resolver.Address{
- Addr: strconv.Itoa(i),
- }] = new(mockClientConn)
- }
- picker := builder.Build(ready)
- var wg sync.WaitGroup
- wg.Add(total)
- for i := 0; i < total; i++ {
- _, done, err := picker.Pick(context.Background(), balancer.PickInfo{
- FullMethodName: "/",
- Ctx: context.Background(),
- })
- assert.Nil(t, err)
- if i%100 == 0 {
- err = status.Error(codes.DeadlineExceeded, "deadline")
- }
- go func() {
- runtime.Gosched()
- done(balancer.DoneInfo{
- Err: err,
- })
- wg.Done()
- }()
- }
- wg.Wait()
- dist := make(map[interface{}]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))
- })
- }
- }
- type mockClientConn struct {
- }
- func (m mockClientConn) UpdateAddresses(addresses []resolver.Address) {
- }
- func (m mockClientConn) Connect() {
- }
|