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
|
package main
import (
"flag"
"fmt"
"io"
"log"
"log/slog"
"net"
"sync"
proxyproto "github.com/pires/go-proxyproto"
"within.website/x/internal"
)
var (
httpPort = flag.Int("http-port", 80, "HTTP forwarding port")
httpsPort = flag.Int("https-port", 443, "HTTPS forwarding port")
httpTarget = flag.String("http-target", "10.216.118.119:80", "target address for http traffic")
httpsTarget = flag.String("https-target", "10.216.118.119:443", "target address for https traffic")
)
func main() {
internal.HandleStartup()
slog.Info("starting up", "httpPort", *httpPort, "httpsPort", *httpsPort, "httpTarget", *httpTarget, "httpsTarget", *httpsTarget)
s := &Server{}
httpLn, err := net.Listen("tcp", fmt.Sprintf(":%d", *httpPort))
if err != nil {
log.Fatalf("can't listen to HTTP port %d: %v", *httpPort, err)
return
}
httpsLn, err := net.Listen("tcp", fmt.Sprintf(":%d", *httpsPort))
if err != nil {
log.Fatalf("can't listen to HTTPS port %d: %v", *httpsPort, err)
return
}
go s.Handle(httpsLn, *httpsTarget)
s.Handle(httpLn, *httpTarget)
}
type Server struct{}
func (s *Server) Handle(l net.Listener, dest string) {
defer l.Close()
for {
conn, err := l.Accept()
if err != nil {
slog.Error("can't accept connection", "localAddr", l.Addr().String(), "err", err)
return
}
slog.Debug("accepted connection", "addr", conn.RemoteAddr().String())
go s.HandleConn(conn, dest)
}
}
func (s *Server) HandleConn(conn net.Conn, dest string) {
defer conn.Close()
slog.Debug("dialing remote host", "remoteHost", dest)
destConn, err := net.Dial("tcp", dest)
if err != nil {
slog.Error("can't dial downstream", "err", err)
return
}
defer destConn.Close()
slog.Debug("dialed remote host", "remoteAddr", conn.RemoteAddr().String(), "destAddr", destConn.RemoteAddr().String())
header := proxyproto.HeaderProxyFromAddrs(2, conn.RemoteAddr(), destConn.RemoteAddr())
if _, err := header.WriteTo(destConn); err != nil {
slog.Error("can't write haproxy header to downstream", "header", header, "err", err)
return
}
slog.Debug("wrote proxy header", "header", header)
var wg sync.WaitGroup
wg.Add(2)
go func() {
slog.Debug("copying from dest to src")
defer wg.Done()
io.Copy(conn, destConn)
// Signal peer that no more data is coming.
conn.(*net.TCPConn).CloseWrite()
slog.Debug("done copying from dest to src")
}()
go func() {
slog.Debug("copying from src to dest")
defer wg.Done()
io.Copy(destConn, conn)
// Signal peer that no more data is coming.
destConn.(*net.TCPConn).CloseWrite()
slog.Debug("done copying from src to dest")
}()
wg.Wait()
slog.Debug("done")
}
|