Skip to content

Commit

Permalink
fix flaky test switch balancer and race
Browse files Browse the repository at this point in the history
  • Loading branch information
menghanl committed Oct 3, 2017
1 parent 88af955 commit bd21314
Show file tree
Hide file tree
Showing 2 changed files with 33 additions and 13 deletions.
2 changes: 2 additions & 0 deletions balancer_conn_wrappers.go
Original file line number Diff line number Diff line change
Expand Up @@ -189,7 +189,9 @@ func (ccb *ccBalancerWrapper) NewSubConn(addrs []resolver.Address, opts balancer
return nil, err
}
acbw := &acBalancerWrapper{ac: ac}
acbw.ac.mu.Lock()
ac.acbw = acbw
acbw.ac.mu.Unlock()
ccb.subConns[acbw] = struct{}{}
return acbw, nil
}
Expand Down
44 changes: 31 additions & 13 deletions balancer_switching_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,17 +19,19 @@
package grpc

import (
"fmt"
"math"
"testing"
"time"

"golang.org/x/net/context"
_ "google.golang.org/grpc/grpclog/glogger"
"google.golang.org/grpc/resolver"
"google.golang.org/grpc/resolver/manual"
"google.golang.org/grpc/test/leakcheck"
)

func checkPickFirst(cc *ClientConn, servers []*server, t *testing.T) {
func checkPickFirst(cc *ClientConn, servers []*server) error {
var (
req = "port"
reply string
Expand All @@ -38,37 +40,47 @@ func checkPickFirst(cc *ClientConn, servers []*server, t *testing.T) {
// The second RPC should succeed with the first server.
for i := 0; i < 1000; i++ {
if err = Invoke(context.Background(), "/foo/bar", &req, &reply, cc); err != nil && ErrorDesc(err) == servers[0].port {
return
return nil
}
time.Sleep(time.Millisecond)
}
t.Fatalf("EmptyCall() = _, %v, want _, %v", err, servers[0].port)
return fmt.Errorf("EmptyCall() = _, %v, want _, %v", err, servers[0].port)
}

func checkRoundRobin(cc *ClientConn, servers []*server, t *testing.T) {
func checkRoundRobin(cc *ClientConn, servers []*server) error {
var (
req = "port"
reply string
err error
)

// Make sure connections to all servers are up.
for _, s := range servers {
for {
if err = Invoke(context.Background(), "/foo/bar", &req, &reply, cc); err != nil && ErrorDesc(err) == s.port {
break
for i := 0; i < 2; i++ {
// Do this check twice, otherwise the first RPC's transport may still be
// picked by the closing pickfirst balancer, and the test becomes flaky.
for _, s := range servers {
var up bool
for i := 0; i < 1000; i++ {
if err = Invoke(context.Background(), "/foo/bar", &req, &reply, cc); err != nil && ErrorDesc(err) == s.port {
up = true
break
}
time.Sleep(time.Millisecond)
}
if !up {
return fmt.Errorf("server %v is not up within 1 second", s.port)
}
time.Sleep(time.Millisecond)
}
}

serverCount := len(servers)
for i := 0; i < 3*serverCount; i++ {
err = Invoke(context.Background(), "/foo/bar", &req, &reply, cc)
if ErrorDesc(err) != servers[i%serverCount].port {
t.Fatalf("Index %d: want peer %v, got peer %v", i, servers[i%serverCount].port, err)
return fmt.Errorf("Index %d: want peer %v, got peer %v", i, servers[i%serverCount].port, err)
}
}
return nil
}

func TestSwitchBalancer(t *testing.T) {
Expand All @@ -87,11 +99,17 @@ func TestSwitchBalancer(t *testing.T) {
defer cc.Close()
r.NewAddress([]resolver.Address{{Addr: servers[0].addr}, {Addr: servers[1].addr}})
// The default balancer is pickfirst.
checkPickFirst(cc, servers, t)
if err := checkPickFirst(cc, servers); err != nil {
t.Fatalf("check pickfirst returned non-nil error: %v", err)
}
// Switch to roundrobin.
cc.switchBalancer("roundrobin")
checkRoundRobin(cc, servers, t)
if err := checkRoundRobin(cc, servers); err != nil {
t.Fatalf("check roundrobin returned non-nil error: %v", err)
}
// Switch to pickfirst.
cc.switchBalancer("pickfirst")
checkPickFirst(cc, servers, t)
if err := checkPickFirst(cc, servers); err != nil {
t.Fatalf("check pickfirst returned non-nil error: %v", err)
}
}

0 comments on commit bd21314

Please sign in to comment.