2015-01-21 20:58:06 +00:00
|
|
|
package httpd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
2015-01-21 23:14:51 +00:00
|
|
|
"strconv"
|
2015-01-21 23:00:44 +00:00
|
|
|
"strings"
|
2015-01-21 20:58:06 +00:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type loggingResponseWriter interface {
|
|
|
|
http.ResponseWriter
|
|
|
|
Status() int
|
|
|
|
Size() int
|
|
|
|
}
|
|
|
|
|
|
|
|
// responseLogger is wrapper of http.ResponseWriter that keeps track of its HTTP status
|
|
|
|
// code and body size
|
|
|
|
type responseLogger struct {
|
|
|
|
w http.ResponseWriter
|
|
|
|
status int
|
|
|
|
size int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *responseLogger) Header() http.Header {
|
|
|
|
return l.w.Header()
|
|
|
|
}
|
|
|
|
|
2015-04-30 22:49:49 +00:00
|
|
|
func (l *responseLogger) Flush() {
|
|
|
|
l.w.(http.Flusher).Flush()
|
|
|
|
}
|
|
|
|
|
2015-01-21 20:58:06 +00:00
|
|
|
func (l *responseLogger) Write(b []byte) (int, error) {
|
|
|
|
if l.status == 0 {
|
2015-01-21 21:07:30 +00:00
|
|
|
// Set status if WriteHeader has not been called
|
2015-01-21 20:58:06 +00:00
|
|
|
l.status = http.StatusOK
|
|
|
|
}
|
|
|
|
size, err := l.w.Write(b)
|
|
|
|
l.size += size
|
|
|
|
return size, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *responseLogger) WriteHeader(s int) {
|
|
|
|
l.w.WriteHeader(s)
|
|
|
|
l.status = s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *responseLogger) Status() int {
|
2015-04-08 15:38:17 +00:00
|
|
|
if l.status == 0 {
|
|
|
|
// This can happen if we never actually write data, but only set response headers.
|
|
|
|
l.status = http.StatusOK
|
|
|
|
}
|
2015-01-21 20:58:06 +00:00
|
|
|
return l.status
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *responseLogger) Size() int {
|
|
|
|
return l.size
|
|
|
|
}
|
|
|
|
|
|
|
|
// Common Log Format: http://en.wikipedia.org/wiki/Common_Log_Format
|
|
|
|
|
2015-01-21 22:36:50 +00:00
|
|
|
// buildLogLine creates a common log format
|
|
|
|
// in addittion to the common fields, we also append referrer, user agent and request ID
|
2015-01-21 20:58:06 +00:00
|
|
|
func buildLogLine(l *responseLogger, r *http.Request, start time.Time) string {
|
2015-01-21 23:14:51 +00:00
|
|
|
username := parseUsername(r)
|
|
|
|
|
|
|
|
host, _, err := net.SplitHostPort(r.RemoteAddr)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
host = r.RemoteAddr
|
|
|
|
}
|
|
|
|
|
|
|
|
uri := r.URL.RequestURI()
|
|
|
|
|
2015-01-22 01:13:51 +00:00
|
|
|
referer := r.Referer()
|
|
|
|
|
|
|
|
userAgent := r.UserAgent()
|
|
|
|
|
2015-01-21 23:14:51 +00:00
|
|
|
fields := []string{
|
|
|
|
host,
|
|
|
|
"-",
|
2015-01-22 20:54:19 +00:00
|
|
|
detect(username, "-"),
|
2015-01-21 23:14:51 +00:00
|
|
|
fmt.Sprintf("[%s]", start.Format("02/Jan/2006:15:04:05 -0700")),
|
|
|
|
r.Method,
|
|
|
|
uri,
|
|
|
|
r.Proto,
|
2015-01-22 20:54:19 +00:00
|
|
|
detect(strconv.Itoa(l.Status()), "-"),
|
2015-01-21 23:14:51 +00:00
|
|
|
strconv.Itoa(l.Size()),
|
2015-01-22 20:54:19 +00:00
|
|
|
detect(referer, "-"),
|
|
|
|
detect(userAgent, "-"),
|
2015-01-21 23:14:51 +00:00
|
|
|
r.Header.Get("Request-Id"),
|
2015-03-16 18:41:15 +00:00
|
|
|
fmt.Sprintf("%s", time.Since(start)),
|
2015-01-21 23:14:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(fields, " ")
|
|
|
|
}
|
|
|
|
|
2015-01-22 20:54:19 +00:00
|
|
|
// detect detects the first presense of a non blank string and returns it
|
|
|
|
func detect(values ...string) string {
|
|
|
|
for _, v := range values {
|
|
|
|
if v != "" {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2015-01-21 23:14:51 +00:00
|
|
|
// parses the uesrname either from the url or auth header
|
|
|
|
func parseUsername(r *http.Request) string {
|
2015-01-22 20:54:19 +00:00
|
|
|
var (
|
|
|
|
username = ""
|
|
|
|
url = r.URL
|
|
|
|
)
|
2015-01-21 23:00:44 +00:00
|
|
|
|
|
|
|
// get username from the url if passed there
|
2015-01-21 20:58:06 +00:00
|
|
|
if url.User != nil {
|
|
|
|
if name := url.User.Username(); name != "" {
|
|
|
|
username = name
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-21 23:00:44 +00:00
|
|
|
// Try to get it from the authorization header if set there
|
2015-01-22 20:54:19 +00:00
|
|
|
if username == "" {
|
2015-01-23 17:50:29 +00:00
|
|
|
if u, _, ok := r.BasicAuth(); ok {
|
|
|
|
username = u
|
2015-01-21 23:00:44 +00:00
|
|
|
}
|
|
|
|
}
|
2015-01-21 23:14:51 +00:00
|
|
|
return username
|
2015-01-21 20:58:06 +00:00
|
|
|
}
|