Welcome to mirror list, hosted at ThFree Co, Russian Federation.

testhelper_test.go « dnsresolver « internal - gitlab.com/gitlab-org/gitaly.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
blob: 819967837ebb32d5e751eb8348ae896806c98b2a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
package dnsresolver

import (
	"context"
	"fmt"
	"net/url"
	"sync"
	"testing"
	"time"

	"github.com/stretchr/testify/require"
	"gitlab.com/gitlab-org/gitaly/v15/internal/testhelper"
	"google.golang.org/grpc/resolver"
	"google.golang.org/grpc/serviceconfig"
)

func TestMain(m *testing.M) {
	testhelper.Run(m)
}

// fakeClientConn stubs resolver.ClientConn. It captures all states and errors received from the
// resolver. As the resolver runs asynchronously, we cannot control when and how many states are
// pushed to the connection. For testing purpose, the fake connection doesn't need to capture all
// states and errors. Instead, it captures some first states and errors (configured via waitState
// and waitError). The caller is expected to call Wait(). This method returns when enough data
// arrives. Afterward, further data are rejected.
type fakeClientConn struct {
	customUpdateState func(resolver.State) error
	states            []resolver.State
	errors            []error

	waitState     int
	waitStateChan chan struct{}

	waitError     int
	waitErrorChan chan struct{}
}

func newFakeClientConn(waitState int, waitError int) *fakeClientConn {
	return &fakeClientConn{
		waitState:     waitState,
		waitStateChan: make(chan struct{}, waitState),
		waitError:     waitError,
		waitErrorChan: make(chan struct{}, waitError),
	}
}

func (c *fakeClientConn) UpdateState(state resolver.State) error {
	if c.customUpdateState != nil {
		return c.customUpdateState(state)
	}
	return c.doUpdateState(state)
}

func (c *fakeClientConn) doUpdateState(state resolver.State) error {
	// Do nothing if received enough states
	if len(c.states) >= c.waitState {
		return nil
	}
	c.states = append(c.states, state)
	c.waitStateChan <- struct{}{}

	return nil
}

func (c *fakeClientConn) Wait() {
	for i := 0; i < c.waitState; i++ {
		<-c.waitStateChan
	}
	for i := 0; i < c.waitError; i++ {
		<-c.waitErrorChan
	}
}

func (c *fakeClientConn) ReportError(err error) {
	// Do nothing if received enough errors
	if len(c.errors) >= c.waitError {
		return
	}
	c.errors = append(c.errors, err)
	c.waitErrorChan <- struct{}{}
}

func (c *fakeClientConn) NewAddress(_ []resolver.Address) { panic("deprecated") }

func (c *fakeClientConn) NewServiceConfig(_ string) { panic("deprecated") }

func (c *fakeClientConn) ParseServiceConfig(_ string) *serviceconfig.ParseResult { panic("deprecated") }

// fakeBackoff stubs the exponential Backoff strategy. It always returns 0 regardless of the retry
// attempts.
type fakeBackoff struct{}

func (c *fakeBackoff) Backoff(uint) time.Duration {
	return 0
}

// fakeLookup stubs the DNS lookup. It wraps around a real DNS lookup. The caller can return an
// alternative addresses, errors, or fallback to use the real DNS lookup if needed.
type fakeLookup struct {
	realLookup dnsLookuper
	stubLookup func(context.Context, string) ([]string, error)
}

func (f *fakeLookup) LookupHost(ctx context.Context, s string) ([]string, error) {
	return f.stubLookup(ctx, s)
}

func newFakeLookup(t *testing.T, authority string) *fakeLookup {
	lookup, err := findDNSLookup(authority)
	require.NoError(t, err)

	return &fakeLookup{realLookup: lookup}
}

func buildResolverTarget(s *testhelper.FakeDNSServer, addr string) resolver.Target {
	return resolver.Target{URL: url.URL{
		Scheme: "dns",
		Host:   s.Addr(),
		Path:   fmt.Sprintf("/%s", addr),
	}}
}

func newTestDNSBuilder(t *testing.T) *Builder {
	return NewBuilder(&BuilderConfig{
		RefreshRate: 0,
		Logger:      testhelper.NewDiscardingLogger(t),
		Backoff:     &fakeBackoff{},
	})
}

// ipList simulates the list of IPs returned by the DNS server in order
type ipList struct {
	ips   [][]string
	mutex sync.Mutex
	index int
}

func (list *ipList) next() []string {
	list.mutex.Lock()
	defer list.mutex.Unlock()

	if list.index < len(list.ips) {
		list.index++
		return list.ips[list.index-1]
	}
	return nil
}

func (list *ipList) peek() []string {
	list.mutex.Lock()
	defer list.mutex.Unlock()

	if list.index < len(list.ips) {
		return list.ips[list.index]
	}
	return nil
}