-
Notifications
You must be signed in to change notification settings - Fork 157
/
logger.go
112 lines (105 loc) · 3.48 KB
/
logger.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
package main
import (
"io"
"log"
"net/http"
"os"
"time"
)
type (
// Options is a struct for specifying configuration parameters for the Logger middleware.
Options struct {
// Prefix is the outputted keyword in front of the log message. Logger automatically wraps the prefix in square brackets (ie. [myApp] ) unless the `DisableAutoBrackets` is set to true. A blank value will not have brackets added. Default is blank (with no brackets).
Prefix string
// DisableAutoBrackets if set to true, will remove the prefix and square brackets. Default is false.
DisableAutoBrackets bool
// RemoteAddressHeaders is a list of header keys that Logger will look at to determine the proper remote address. Useful when using a proxy like Nginx: `[]string{"X-Forwarded-Proto"}`. Default is an empty slice, and thus will use `reqeust.RemoteAddr`.
RemoteAddressHeaders []string
// Out is the destination to which the logged data will be written too. Default is `os.Stdout`.
Out io.Writer
// OutputFlags defines the logging properties. See http://golang.org/pkg/log/#pkg-constants. To disable all flags, set this to `-1`. Defaults to log.LstdFlags (2009/01/23 01:23:23).
OutputFlags int
// IgnoredRequestURIs is a list of path values we do not want logged out. Exact match only!
IgnoredRequestURIs []string
}
// Logger is a HTTP middleware handler that logs a request. Outputted information includes status, method, URL, remote address, size, and the time it took to process the request.
Logger struct {
*log.Logger
opt Options
}
// customResponseWriter is a wrapper around golang's standard ResponseWriter to include a status code and response size
customResponseWriter struct {
http.ResponseWriter
status int
size int
}
)
// NewLogger returns a reference to a configured instance of Logger
func NewLogger(opts ...Options) *Logger {
var o Options
if len(opts) == 0 {
o = Options{}
} else {
o = opts[0]
}
// Determine prefix.
prefix := o.Prefix
if len(prefix) > 0 && o.DisableAutoBrackets == false {
prefix = "[" + prefix + "] "
}
// Determine output writer.
var output io.Writer
if o.Out != nil {
output = o.Out
} else {
// Default is stdout.
output = os.Stdout
}
// Determine output flags.
flags := log.LstdFlags
if o.OutputFlags == -1 {
flags = 0
} else if o.OutputFlags != 0 {
flags = o.OutputFlags
}
return &Logger{
Logger: log.New(output, prefix, flags),
opt: o,
}
}
func (l *Logger) logFunc(rw http.ResponseWriter, r *http.Request, next http.HandlerFunc) {
start := time.Now()
crw := newCustomResponseWriter(rw)
next(crw, r)
end := time.Now()
for _, ignoredURI := range l.opt.IgnoredRequestURIs {
if ignoredURI == r.RequestURI {
return
}
}
addr := r.RemoteAddr
headersToCheck := []string{"X-Real-Ip", "X-Forwarded-For"}
for _, headerKey := range headersToCheck {
if val := r.Header.Get(headerKey); len(val) > 0 {
addr = val
break
}
}
l.Printf("| %3d | %13v | %50s | %8d | %5s %s\n", crw.status, end.Sub(start), addr, crw.size, r.Method, r.RequestURI)
}
func (c *customResponseWriter) WriteHeader(status int) {
c.status = status
c.ResponseWriter.WriteHeader(status)
}
func (c *customResponseWriter) Write(b []byte) (int, error) {
size, err := c.ResponseWriter.Write(b)
c.size += size
return size, err
}
func newCustomResponseWriter(w http.ResponseWriter) *customResponseWriter {
// When WriteHeader is not called, it's safe to assume the status will be 200.
return &customResponseWriter{
ResponseWriter: w,
status: 200,
}
}