|
@@ -1,7 +1,92 @@
|
|
|
package p2c
|
|
|
|
|
|
-import "testing"
|
|
|
+import (
|
|
|
+ "context"
|
|
|
+ "fmt"
|
|
|
+ "strconv"
|
|
|
+ "testing"
|
|
|
+
|
|
|
+ "zero/core/logx"
|
|
|
+ "zero/core/mathx"
|
|
|
+
|
|
|
+ "github.com/stretchr/testify/assert"
|
|
|
+ "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
|
|
|
+ }{
|
|
|
+ {
|
|
|
+ name: "single",
|
|
|
+ candidates: 1,
|
|
|
+ },
|
|
|
+ {
|
|
|
+ name: "multiple",
|
|
|
+ candidates: 100,
|
|
|
+ },
|
|
|
+ }
|
|
|
+
|
|
|
+ for _, test := range tests {
|
|
|
+ t.Run(test.name, func(t *testing.T) {
|
|
|
+ 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)
|
|
|
+ for i := 0; i < 10000; 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")
|
|
|
+ }
|
|
|
+ done(balancer.DoneInfo{
|
|
|
+ Err: err,
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ 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 > .95, fmt.Sprintf("entropy is %f, less than .95", entropy))
|
|
|
+ })
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+type mockClientConn struct {
|
|
|
+}
|
|
|
+
|
|
|
+func (m mockClientConn) UpdateAddresses(addresses []resolver.Address) {
|
|
|
+}
|
|
|
|
|
|
+func (m mockClientConn) Connect() {
|
|
|
}
|