-
Notifications
You must be signed in to change notification settings - Fork 14
/
Copy pathserver_test.go
162 lines (137 loc) · 4.01 KB
/
server_test.go
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
159
160
161
162
package admissioncontrol
import (
"context"
"fmt"
"math"
"net"
"net/http"
"net/http/httptest"
"testing"
"time"
)
// noopLogger is a no-op type that satifies the kit.Logger interface
type noopLogger struct{}
// Log logs nothing. Nada. Zilch.
func (nl *noopLogger) Log(keyvals ...interface{}) error {
return nil
}
type testServer struct {
srv *AdmissionServer
client *http.Client
url string
}
func newTestServer(ctx context.Context, t *testing.T) *testServer {
testHandler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintln(w, "OK")
})
testSrv := httptest.NewUnstartedServer(testHandler)
testSrv.Start()
// We start the test server, copy its config out, and close it down so we can
// start our own server.
srv := &http.Server{
Addr: testSrv.Listener.Addr().String(),
Handler: testHandler,
}
admissionServer, err := NewServer(srv, &noopLogger{})
if err != nil {
t.Fatalf("admission server creation failed: %s", err)
return nil
}
testSrv.Close()
go func() {
if err := admissionServer.Run(ctx); err != nil {
t.Logf("server stopped: %s", err)
}
}()
// Wait for our listener to be ready for testing before we return a running
// test server.
var (
backoffFactor = 1.25
waitTime = time.Millisecond * 50
maxAttempts = 5
dialTimeout = time.Second * 1
)
for attempt := 1; attempt <= maxAttempts; attempt++ {
conn, err := net.DialTimeout(
"tcp",
admissionServer.srv.Addr,
dialTimeout,
)
if err != nil {
time.Sleep(waitTime)
newWait := float64(waitTime) * math.Pow(backoffFactor, float64(attempt))
waitTime = time.Duration(newWait)
continue
}
if err := conn.Close(); err != nil {
t.Fatalf("failed to close the test connection: %v", err)
}
break
}
return &testServer{srv: admissionServer, client: testSrv.Client(), url: testSrv.URL}
}
// Test that we can start a minimal AdmissionServer and handle a request.
func TestAdmissionServer(t *testing.T) {
t.Run("AdmissionServer should return an error w/o a *http.Server", func(t *testing.T) {
t.Parallel()
_, err := NewServer(nil, &noopLogger{})
if err == nil {
t.Fatalf("nil *http.Server did not return an error")
}
})
t.Run("AdmissionServer should return an error w/o a log.Logger", func(t *testing.T) {
t.Parallel()
_, err := NewServer(&http.Server{}, nil)
if err == nil {
t.Fatalf("nil log.Logger did not return an error")
}
})
t.Run("AdmissionServer starts & accepts HTTP requests", func(t *testing.T) {
t.Parallel()
testSrv := newTestServer(context.TODO(), t)
defer testSrv.srv.Stop()
client := testSrv.client
req, err := http.NewRequest(http.MethodGet, testSrv.url, nil)
if err != nil {
t.Fatalf("request creation failed: %s", err)
}
resp, err := client.Do(req)
if err != nil {
t.Fatalf("failed to make a request: %s", err)
}
if status := resp.StatusCode; status != http.StatusOK {
t.Fatalf("unexpected status code: got %d (wanted %d)", status, http.StatusOK)
}
})
t.Run("AdmissionServer.Stop() stops the server", func(t *testing.T) {
t.Parallel()
testSrv := newTestServer(context.TODO(), t)
testSrv.srv.GracePeriod = time.Microsecond * 1
// Force a shutdown
testSrv.srv.Stop()
time.Sleep(testSrv.srv.GracePeriod)
if err := testSrv.srv.srv.ListenAndServeTLS("", ""); err != http.ErrServerClosed {
t.Fatalf(
"server did not shutdown after a cancellation signal was received: got %v (want %v)",
err,
http.ErrServerClosed,
)
}
})
t.Run("AdmissionServer handles a cancellation context and shuts down.", func(t *testing.T) {
t.Parallel()
ctx, cancel := context.WithCancel(context.Background())
testSrv := newTestServer(ctx, t)
testSrv.srv.GracePeriod = time.Microsecond * 1
// Cancel the context
cancel()
time.Sleep(testSrv.srv.GracePeriod + time.Second)
if err := testSrv.srv.srv.ListenAndServeTLS("", ""); err != http.ErrServerClosed {
t.Fatalf(
"server did not shutdown after a cancellation signal was received: got %v (want %v)",
err,
http.ErrServerClosed,
)
}
})
}