-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver_test.go
146 lines (130 loc) · 2.93 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
// server starts an HTTP Server to run the application
package server
import (
"context"
"net/http"
"reflect"
"sync"
"sync/atomic"
"syscall"
"testing"
"time"
)
func TestNew(t *testing.T) {
t.Parallel()
type args struct {
addr string
}
tests := []struct {
name string
args args
want *Server
}{
{
name: "New",
args: args{addr: "myhost:2000"},
want: &Server{
srv: &http.Server{
Addr: "myhost:2000",
ReadHeaderTimeout: 60 * time.Second,
},
},
},
}
for _, tt := range tests {
tt := tt
t.Run(tt.name, func(t *testing.T) {
t.Parallel()
if got := New(tt.args.addr); !reflect.DeepEqual(got, tt.want) {
t.Errorf("New() = %v, want %v", got, tt.want)
}
})
}
}
func TestServer_Start(t *testing.T) {
type args struct {
responseDelay time.Duration
}
tests := []struct {
name string
args args
wantErr bool
}{
{
name: "Request completes",
args: args{responseDelay: time.Millisecond},
},
{
name: "Request incomplete",
args: args{responseDelay: time.Second * 10},
wantErr: true,
},
}
for _, tt := range tests {
tt := tt
t.Run(tt.name, func(t *testing.T) {
var wg sync.WaitGroup
var handlerCalled atomic.Bool
handler := http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
wg.Done()
if handlerCalled.Load() {
time.Sleep(tt.args.responseDelay)
}
handlerCalled.Store(true)
})
s := &Server{
srv: &http.Server{
Addr: "localhost:23461",
ReadHeaderTimeout: 60 * time.Second,
},
}
done := make(chan bool)
go func() {
defer close(done)
ctx := context.Background()
if err := s.Start(ctx, handler); (err != nil) != tt.wantErr {
t.Errorf("Server.Start() error = %v, wantErr %v", err, tt.wantErr)
}
done <- true
}()
req, err := http.NewRequestWithContext(context.Background(), http.MethodGet, "http://localhost:23461", http.NoBody)
if err != nil {
t.Fatalf("http.NewRequest() error = %v", err)
}
var res *http.Response
for i := 0; i <= 5; i++ {
wg.Add(1)
var err error
res, err = http.DefaultClient.Do(req)
if err == nil {
break
}
if i == 5 {
t.Fatalf("http.Do() error = %v", err)
}
time.Sleep(time.Millisecond * time.Duration(i*20))
wg.Done()
}
if res.StatusCode != http.StatusOK {
t.Fatalf("res.StatusCode = %v, want %v", res.StatusCode, http.StatusOK)
}
if called := handlerCalled.Load(); !called {
t.Fatalf("handlerCalled = %v", called)
}
// second request that will delay via tt.args.responseDelay
wg.Add(1)
go func() {
_, _ = http.DefaultClient.Do(req)
}()
// Wait for handler to signal it has been invoked
wg.Wait()
// send shutdown
_ = syscall.Kill(syscall.Getpid(), syscall.SIGINT)
select {
case <-done:
case <-time.After(time.Second * 10):
t.Fatalf("srv.Shutdown() did not return within 10 seconds")
}
})
}
}