-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlogged_handler.go
150 lines (126 loc) · 3.7 KB
/
logged_handler.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
package vproxy
import (
"bufio"
"crypto/tls"
"fmt"
"io"
"log"
"net"
"net/http"
"strings"
"time"
)
var defaultTLSHost = "vproxy.local"
// LoggedHandler is an http.Server implementation which multiplexes requests to the
// vhost backends (via a handler) and logs each request.
type LoggedHandler struct {
*http.ServeMux
VhostLogListeners map[string]chan string
vhostMux *VhostMux
defaultHost string
defaultCert string
defaultKey string
}
// NewLoggedHandler wraps the given handler with a request/response logger
func NewLoggedHandler(vm *VhostMux) *LoggedHandler {
lh := &LoggedHandler{
ServeMux: http.NewServeMux(),
VhostLogListeners: make(map[string]chan string),
vhostMux: vm,
}
lh.defaultHost = defaultTLSHost
lh.createDefaultCert()
// Map all requests, by default, to the appropriate vhost
lh.Handle("/", vm)
return lh
}
func (lh *LoggedHandler) createDefaultCert() {
var err error
lh.defaultCert, lh.defaultKey, err = MakeCert(lh.defaultHost)
if err != nil {
log.Fatalf("failed to create default cert for vproxy.local: %s", err)
}
}
func (lh *LoggedHandler) AddVhost(vhost *Vhost, listener chan string) {
lh.VhostLogListeners[vhost.Host] = listener
lh.vhostMux.Servers[vhost.Host] = vhost
}
func (lh *LoggedHandler) RemoveVhost(host string) {
delete(lh.VhostLogListeners, host)
delete(lh.vhostMux.Servers, host)
}
// DumpServers to the given writer
func (lh *LoggedHandler) DumpServers(w io.Writer) {
fmt.Fprintf(w, "%d vhosts:\n", len(lh.vhostMux.Servers))
for _, v := range lh.vhostMux.Servers {
fmt.Fprintf(w, "%s -> %s:%d\n", v.Host, v.ServiceHost, v.Port)
}
}
// Create multi-certificate TLS config from vhost config
func (lh *LoggedHandler) CreateTLSConfig() *tls.Config {
cfg := &tls.Config{}
// Add default internal cert
cert, err := tls.LoadX509KeyPair(lh.defaultCert, lh.defaultKey)
if err != nil {
log.Fatal("failed to load keypair:", err)
}
cfg.Certificates = append(cfg.Certificates, cert)
// add cert for each vhost
for _, server := range lh.vhostMux.Servers {
cert, err := tls.LoadX509KeyPair(server.Cert, server.Key)
if err != nil {
log.Fatal("failed to load keypair:", err)
}
cfg.Certificates = append(cfg.Certificates, cert)
}
// build cn and return
cfg.BuildNameToCertificate()
return cfg
}
func (lh *LoggedHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
record := &LogRecord{
ResponseWriter: w,
}
// serve request and capture timings
startTime := time.Now()
lh.ServeMux.ServeHTTP(record, r)
finishTime := time.Now()
elapsedTime := finishTime.Sub(startTime)
host := getHostName(r.Host)
l := fmt.Sprintf("%s [%s] %s [ %d ] %s %d %s", r.RemoteAddr, host, r.Method, record.status, r.URL, r.ContentLength, elapsedTime)
log.Println(l)
if listener, ok := lh.VhostLogListeners[host]; ok {
listener <- l
}
}
// ignore port num, if any
func getHostName(input string) string {
s := strings.Split(input, ":")
return s[0]
}
// LogRecord is a thin wrapper around http.ResponseWriter which allows us to
// capture the number of response bytes written and the http status code.
type LogRecord struct {
http.ResponseWriter
status int
responseBytes int64
}
// Write wrapper that counts bytes
func (r *LogRecord) Write(p []byte) (int, error) {
written, err := r.ResponseWriter.Write(p)
r.responseBytes += int64(written)
return written, err
}
// WriteHeader wrapper that captures status code
func (r *LogRecord) WriteHeader(status int) {
r.status = status
r.ResponseWriter.WriteHeader(status)
}
// Hijack wrapper
func (r *LogRecord) Hijack() (rwc net.Conn, buf *bufio.ReadWriter, err error) {
hj, ok := r.ResponseWriter.(http.Hijacker)
if !ok {
log.Fatal("error: expected a hijacker here")
}
return hj.Hijack()
}