123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238 |
- // Copyright 2015 The etcd Authors
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- package srv
- import (
- "errors"
- "net"
- "reflect"
- "strings"
- "testing"
- "go.etcd.io/etcd/pkg/testutil"
- )
- func TestSRVGetCluster(t *testing.T) {
- defer func() {
- lookupSRV = net.LookupSRV
- resolveTCPAddr = net.ResolveTCPAddr
- }()
- name := "dnsClusterTest"
- dns := map[string]string{
- "1.example.com.:2480": "10.0.0.1:2480",
- "2.example.com.:2480": "10.0.0.2:2480",
- "3.example.com.:2480": "10.0.0.3:2480",
- "4.example.com.:2380": "10.0.0.3:2380",
- }
- srvAll := []*net.SRV{
- {Target: "1.example.com.", Port: 2480},
- {Target: "2.example.com.", Port: 2480},
- {Target: "3.example.com.", Port: 2480},
- }
- tests := []struct {
- scheme string
- records []*net.SRV
- urls []string
- expected string
- }{
- {
- "https",
- []*net.SRV{},
- nil,
- "",
- },
- {
- "https",
- srvAll,
- nil,
- "0=https://1.example.com:2480,1=https://2.example.com:2480,2=https://3.example.com:2480",
- },
- {
- "http",
- srvAll,
- nil,
- "0=http://1.example.com:2480,1=http://2.example.com:2480,2=http://3.example.com:2480",
- },
- {
- "https",
- srvAll,
- []string{"https://10.0.0.1:2480"},
- "dnsClusterTest=https://1.example.com:2480,0=https://2.example.com:2480,1=https://3.example.com:2480",
- },
- // matching local member with resolved addr and return unresolved hostnames
- {
- "https",
- srvAll,
- []string{"https://10.0.0.1:2480"},
- "dnsClusterTest=https://1.example.com:2480,0=https://2.example.com:2480,1=https://3.example.com:2480",
- },
- // reject if apurls are TLS but SRV is only http
- {
- "http",
- srvAll,
- []string{"https://10.0.0.1:2480"},
- "0=http://2.example.com:2480,1=http://3.example.com:2480",
- },
- }
- resolveTCPAddr = func(network, addr string) (*net.TCPAddr, error) {
- if strings.Contains(addr, "10.0.0.") {
- // accept IP addresses when resolving apurls
- return net.ResolveTCPAddr(network, addr)
- }
- if dns[addr] == "" {
- return nil, errors.New("missing dns record")
- }
- return net.ResolveTCPAddr(network, dns[addr])
- }
- for i, tt := range tests {
- lookupSRV = func(service string, proto string, domain string) (string, []*net.SRV, error) {
- return "", tt.records, nil
- }
- urls := testutil.MustNewURLs(t, tt.urls)
- str, err := GetCluster(tt.scheme, "etcd-server", name, "example.com", urls)
- if err != nil {
- t.Fatalf("%d: err: %#v", i, err)
- }
- if strings.Join(str, ",") != tt.expected {
- t.Errorf("#%d: cluster = %s, want %s", i, str, tt.expected)
- }
- }
- }
- func TestSRVDiscover(t *testing.T) {
- defer func() { lookupSRV = net.LookupSRV }()
- tests := []struct {
- withSSL []*net.SRV
- withoutSSL []*net.SRV
- expected []string
- }{
- {
- []*net.SRV{},
- []*net.SRV{},
- []string{},
- },
- {
- []*net.SRV{
- {Target: "10.0.0.1", Port: 2480},
- {Target: "10.0.0.2", Port: 2480},
- {Target: "10.0.0.3", Port: 2480},
- },
- []*net.SRV{},
- []string{"https://10.0.0.1:2480", "https://10.0.0.2:2480", "https://10.0.0.3:2480"},
- },
- {
- []*net.SRV{
- {Target: "10.0.0.1", Port: 2480},
- {Target: "10.0.0.2", Port: 2480},
- {Target: "10.0.0.3", Port: 2480},
- },
- []*net.SRV{
- {Target: "10.0.0.1", Port: 7001},
- },
- []string{"https://10.0.0.1:2480", "https://10.0.0.2:2480", "https://10.0.0.3:2480", "http://10.0.0.1:7001"},
- },
- {
- []*net.SRV{
- {Target: "10.0.0.1", Port: 2480},
- {Target: "10.0.0.2", Port: 2480},
- {Target: "10.0.0.3", Port: 2480},
- },
- []*net.SRV{
- {Target: "10.0.0.1", Port: 7001},
- },
- []string{"https://10.0.0.1:2480", "https://10.0.0.2:2480", "https://10.0.0.3:2480", "http://10.0.0.1:7001"},
- },
- {
- []*net.SRV{
- {Target: "a.example.com", Port: 2480},
- {Target: "b.example.com", Port: 2480},
- {Target: "c.example.com", Port: 2480},
- },
- []*net.SRV{},
- []string{"https://a.example.com:2480", "https://b.example.com:2480", "https://c.example.com:2480"},
- },
- }
- for i, tt := range tests {
- lookupSRV = func(service string, proto string, domain string) (string, []*net.SRV, error) {
- if service == "etcd-client-ssl" {
- return "", tt.withSSL, nil
- }
- if service == "etcd-client" {
- return "", tt.withoutSSL, nil
- }
- return "", nil, errors.New("Unknown service in mock")
- }
- srvs, err := GetClient("etcd-client", "example.com", "")
- if err != nil {
- t.Fatalf("%d: err: %#v", i, err)
- }
- if !reflect.DeepEqual(srvs.Endpoints, tt.expected) {
- t.Errorf("#%d: endpoints = %v, want %v", i, srvs.Endpoints, tt.expected)
- }
- }
- }
- func TestGetSRVService(t *testing.T) {
- tests := []struct {
- scheme string
- serviceName string
- expected string
- }{
- {
- "https",
- "",
- "etcd-client-ssl",
- },
- {
- "http",
- "",
- "etcd-client",
- },
- {
- "https",
- "foo",
- "etcd-client-ssl-foo",
- },
- {
- "http",
- "bar",
- "etcd-client-bar",
- },
- }
- for i, tt := range tests {
- service := GetSRVService("etcd-client", tt.serviceName, tt.scheme)
- if strings.Compare(service, tt.expected) != 0 {
- t.Errorf("#%d: service = %s, want %s", i, service, tt.expected)
- }
- }
- }
|