-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlisten_proxy_protocol_test.go
124 lines (117 loc) · 3.33 KB
/
listen_proxy_protocol_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
package net
import (
"context"
"fmt"
"io"
"net"
"net/http"
"strings"
"testing"
"time"
"github.com/go-chi/chi/v5"
proxyproto "github.com/pires/go-proxyproto"
)
func TestImpl_ListenAndServe_ProxyProtocol(t *testing.T) {
const (
testAddress, testNetwork = "127.0.0.1:8080", "tcp"
uri = "http" + "://" + testAddress + "/"
testProxyAddress, testProxyPort = "111.222.21.22", 43210
)
var (
nut Interface
conf *Configuration
testTransport http.RoundTripper
origTransport http.RoundTripper
)
// Конфигурация.
conf, _ = parseAddress(testAddress, testNetwork)
conf.ProxyProtocol = true
// Сервер.
nut = New().
Handler(func(l net.Listener) (err error) {
var (
route *chi.Mux
srv *http.Server
)
route = chi.NewRouter()
route.Get("/", func(wr http.ResponseWriter, rq *http.Request) {
wr.Header().Set("Content-Type", "text/plain")
_, _ = io.WriteString(wr, rq.RemoteAddr)
})
srv = &http.Server{
Addr: testAddress,
Handler: route,
}
err = srv.Serve(l)
return
}).
ListenAndServeWithConfig(conf)
if nut.Error() != nil {
t.Errorf("запуск сервера прерван ошибкой: %s", nut.Error())
return
}
// Тестовый транспорт.
testTransport = &http.Transport{
Proxy: http.ProxyFromEnvironment,
DialContext: func(ctx context.Context, network, addr string) (conn net.Conn, err error) {
var (
target *net.TCPAddr
header *proxyproto.Header
)
if target, err = net.ResolveTCPAddr(testNetwork, testAddress); err != nil {
return
}
conn, err = net.DialTCP(testNetwork, nil, target)
header = &proxyproto.Header{
Version: 1,
Command: proxyproto.PROXY,
TransportProtocol: proxyproto.TCPv4,
SourceAddr: &net.TCPAddr{
IP: net.ParseIP(testProxyAddress),
Port: testProxyPort,
},
DestinationAddr: &net.TCPAddr{
IP: net.ParseIP("127.0.0.1"),
Port: 8080,
},
}
if _, err = header.WriteTo(conn); err != nil {
return
}
return
},
ForceAttemptHTTP2: true,
MaxIdleConns: 100,
IdleConnTimeout: 90 * time.Second,
TLSHandshakeTimeout: 10 * time.Second,
ExpectContinueTimeout: 1 * time.Second,
}
// Временная подмена транспорта.
origTransport = http.DefaultTransport
http.DefaultTransport = testTransport
defer func() { http.DefaultTransport = origTransport }()
// Клиент.
go func() {
defer func() { nut.Stop() }() // Остановка сервера.
rq, e := http.Get(uri)
if e != nil {
t.Errorf("запрос к %q прерван ошибкой: %s", uri, e)
}
defer func() { _ = rq.Body.Close() }()
buf, e := io.ReadAll(rq.Body)
if e != nil {
t.Errorf("чтение ответа на запрос к %q прервано ошибкой: %s", uri, e)
}
testProxyProto := fmt.Sprintf("%s:%d", testProxyAddress, testProxyPort)
if rq.StatusCode != 200 || !strings.Contains(string(buf), testProxyProto) {
t.Errorf("получен ответ сервера: %q, ожидался: %q", string(buf), testProxyProto)
}
}()
// Ожидание завершения сервера.
if err := nut.Wait().
Error(); err != nil {
t.Errorf("сервер завершился с ошибкой: %s", err)
return
}
nut.Stop()
}