Fix request dumper

The existing code had a bug in printing request params.  Simplify the
request logger by using httputil.DumpRequest.

Also print the request before handing it downstream.

Fixes #425
This commit is contained in:
Cameron Moore 2020-05-21 17:47:55 -05:00
parent a75ab4f92f
commit 526c9a20ac

View file

@ -8,13 +8,11 @@ import (
"bytes" "bytes"
"fmt" "fmt"
"io" "io"
"io/ioutil"
"net" "net"
"net/http" "net/http"
"net/http/httputil"
"sort" "sort"
"strings" "strings"
"github.com/gorilla/mux"
) )
// responseDupper tees the response to a buffer and a response writer. // responseDupper tees the response to a buffer and a response writer.
@ -34,70 +32,53 @@ func Dumper(w io.Writer) func(http.Handler) http.Handler {
// Request ID // Request ID
rid := r.Context().Value(RequestIDKey) rid := r.Context().Value(RequestIDKey)
// Request URL // Dump request
buf.WriteString(fmt.Sprintf("> [%s] %s %s", rid, r.Method, r.URL.String()))
// Request Headers bd, err := httputil.DumpRequest(r, true)
keys := make([]string, len(r.Header))
i := 0
for k := range r.Header {
keys[i] = k
i++
}
sort.Strings(keys)
for _, k := range keys {
buf.WriteString(fmt.Sprintf("\n> [%s] %s: %s", rid, k, strings.Join(r.Header[k], ", ")))
}
// Request parameters
params := mux.Vars(r)
keys = make([]string, len(params))
i = 0
for k := range params {
keys[i] = k
i++
}
sort.Strings(keys)
for _, k := range keys {
buf.WriteString(fmt.Sprintf("\n> [%s] %s: %s", rid, k, strings.Join(r.Header[k], ", ")))
}
// Request body
b, err := ioutil.ReadAll(r.Body)
if err != nil { if err != nil {
b = []byte("failed to read body: " + err.Error()) buf.WriteString(fmt.Sprintf("[%s] Error dumping request for debugging: %s\n", rid, err))
} }
if len(b) > 0 {
buf.WriteByte('\n') sc := bufio.NewScanner(bytes.NewBuffer(bd))
lines := strings.Split(string(b), "\n") sc.Split(bufio.ScanLines)
for _, line := range lines { for sc.Scan() {
buf.WriteString(fmt.Sprintf("> [%s] %s\n", rid, line)) buf.WriteString(fmt.Sprintf("> [%s] ", rid))
} buf.WriteString(sc.Text() + "\n")
} }
r.Body = ioutil.NopCloser(bytes.NewBuffer(b))
w.Write(buf.Bytes())
buf.Reset()
// Dump Response
dupper := &responseDupper{ResponseWriter: rw, Buffer: &bytes.Buffer{}} dupper := &responseDupper{ResponseWriter: rw, Buffer: &bytes.Buffer{}}
h.ServeHTTP(dupper, r) h.ServeHTTP(dupper, r)
buf.WriteString(fmt.Sprintf("\n< [%s] %s", rid, http.StatusText(dupper.Status))) // Response Status
keys = make([]string, len(dupper.Header())) buf.WriteString(fmt.Sprintf("< [%s] %d %s\n", rid, dupper.Status, http.StatusText(dupper.Status)))
i = 0
// Response Headers
keys := make([]string, len(dupper.Header()))
i := 0
for k := range dupper.Header() { for k := range dupper.Header() {
keys[i] = k keys[i] = k
i++ i++
} }
sort.Strings(keys) sort.Strings(keys)
for _, k := range keys { for _, k := range keys {
buf.WriteString(fmt.Sprintf("\n< [%s] %s: %s", rid, k, strings.Join(dupper.Header()[k], ", "))) buf.WriteString(fmt.Sprintf("< [%s] %s: %s\n", rid, k, strings.Join(dupper.Header()[k], ", ")))
} }
// Response Body
if dupper.Buffer.Len() > 0 { if dupper.Buffer.Len() > 0 {
buf.WriteByte('\n') buf.WriteString(fmt.Sprintf("< [%s]\n", rid))
lines := strings.Split(dupper.Buffer.String(), "\n") sc = bufio.NewScanner(dupper.Buffer)
for _, line := range lines { sc.Split(bufio.ScanLines)
buf.WriteString(fmt.Sprintf("< [%s] %s\n", rid, line)) for sc.Scan() {
buf.WriteString(fmt.Sprintf("< [%s] ", rid))
buf.WriteString(sc.Text() + "\n")
} }
} }
buf.WriteByte('\n')
w.Write(buf.Bytes()) w.Write(buf.Bytes())
}) })
} }