test: make joinLAN/WAN reliable

only return if the members can see each other
This commit is contained in:
Frank Schroeder 2017-07-04 21:17:16 +02:00 committed by Frank Schröder
parent 74d3c4d896
commit 98dc634f17
1 changed files with 65 additions and 5 deletions

View File

@ -1,12 +1,34 @@
package consul package consul
import ( import (
"errors"
"fmt" "fmt"
"testing" "testing"
"github.com/hashicorp/consul/testutil/retry"
"github.com/hashicorp/serf/serf"
) )
func waitForLeader(servers ...*Server) error {
if len(servers) == 0 {
return errors.New("no servers")
}
dc := servers[0].config.Datacenter
for _, s := range servers {
if s.config.Datacenter != dc {
return fmt.Errorf("servers are in different datacenters %s and %s", s.config.Datacenter, dc)
}
}
for _, s := range servers {
if s.IsLeader() {
return nil
}
}
return errors.New("no leader")
}
// wantPeers determines whether the server has the given // wantPeers determines whether the server has the given
// number of peers. // number of raft peers.
func wantPeers(s *Server, peers int) error { func wantPeers(s *Server, peers int) error {
n, err := s.numPeers() n, err := s.numPeers()
if err != nil { if err != nil {
@ -40,6 +62,7 @@ func joinAddrWAN(s *Server) string {
type clientOrServer interface { type clientOrServer interface {
JoinLAN(addrs []string) (int, error) JoinLAN(addrs []string) (int, error)
LANMembers() []serf.Member
} }
// joinLAN is a convenience function for // joinLAN is a convenience function for
@ -49,10 +72,25 @@ func joinLAN(t *testing.T, member clientOrServer, leader *Server) {
if member == nil || leader == nil { if member == nil || leader == nil {
panic("no server") panic("no server")
} }
addr := []string{joinAddrLAN(leader)} var memberAddr string
if _, err := member.JoinLAN(addr); err != nil { switch x := member.(type) {
case *Server:
memberAddr = joinAddrLAN(x)
case *Client:
memberAddr = fmt.Sprintf("127.0.0.1:%d", x.config.SerfLANConfig.MemberlistConfig.BindPort)
}
leaderAddr := joinAddrLAN(leader)
if _, err := member.JoinLAN([]string{leaderAddr}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
retry.Run(t, func(r *retry.R) {
if !seeEachOther(leader.LANMembers(), member.LANMembers(), leaderAddr, memberAddr) {
r.Fatalf("leader and member cannot see each other on LAN")
}
})
if !seeEachOther(leader.LANMembers(), member.LANMembers(), leaderAddr, memberAddr) {
t.Fatalf("leader and member cannot see each other on LAN")
}
} }
// joinWAN is a convenience function for // joinWAN is a convenience function for
@ -62,8 +100,30 @@ func joinWAN(t *testing.T, member, leader *Server) {
if member == nil || leader == nil { if member == nil || leader == nil {
panic("no server") panic("no server")
} }
addr := []string{joinAddrWAN(leader)} leaderAddr, memberAddr := joinAddrWAN(leader), joinAddrWAN(member)
if _, err := member.JoinWAN(addr); err != nil { if _, err := member.JoinWAN([]string{leaderAddr}); err != nil {
t.Fatal(err) t.Fatal(err)
} }
retry.Run(t, func(r *retry.R) {
if !seeEachOther(leader.WANMembers(), member.WANMembers(), leaderAddr, memberAddr) {
r.Fatalf("leader and member cannot see each other on WAN")
}
})
if !seeEachOther(leader.WANMembers(), member.WANMembers(), leaderAddr, memberAddr) {
t.Fatalf("leader and member cannot see each other on WAN")
}
}
func seeEachOther(a, b []serf.Member, addra, addrb string) bool {
return serfMembersContains(a, addrb) && serfMembersContains(b, addra)
}
func serfMembersContains(members []serf.Member, addr string) bool {
for _, m := range members {
maddr := fmt.Sprintf("%s:%d", m.Addr.String(), m.Port)
if maddr == addr {
return true
}
}
return false
} }