forked from mushorg/glutton
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathhttp.go
69 lines (64 loc) · 1.73 KB
/
http.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
package glutton
import (
"bufio"
"bytes"
"context"
"encoding/hex"
"fmt"
"net"
"net/http"
"strings"
)
// formatRequest generates ascii representation of a request
func formatRequest(r *http.Request) string {
// Create return string
var request []string
// Add the request string
url := fmt.Sprintf("%v %v %v", r.Method, r.URL, r.Proto)
request = append(request, url)
// Add the host
request = append(request, fmt.Sprintf("Host: %v", r.Host))
// Loop through headers
for name, headers := range r.Header {
name = strings.ToLower(name)
for _, h := range headers {
request = append(request, fmt.Sprintf("%v: %v", name, h))
}
}
// If this is a POST, add post data
if r.Method == "POST" {
r.ParseForm()
request = append(request, "\n")
request = append(request, r.Form.Encode())
}
// Return the request as a string
return strings.Join(request, "\n")
}
// HandleHTTP takes a net.Conn and does basic HTTP communication
func (g *Glutton) HandleHTTP(ctx context.Context, conn net.Conn) (err error) {
defer func() {
err = conn.Close()
if err != nil {
g.logger.Error(fmt.Sprintf("[http ] error: %v", err))
}
}()
req, err := http.ReadRequest(bufio.NewReader(conn))
if err != nil {
g.logger.Error(fmt.Sprintf("[http ] error: %v", err))
return err
}
g.logger.Info(fmt.Sprintf("[http ] %s", formatRequest(req)))
if req.ContentLength > 0 {
defer req.Body.Close()
buf := bytes.NewBuffer(make([]byte, 0, req.ContentLength))
_, err = buf.ReadFrom(req.Body)
if err != nil {
g.logger.Error(fmt.Sprintf("[http ] error: %v", err))
return err
}
body := buf.Bytes()
g.logger.Info(fmt.Sprintf("[http ] http body:\n%s", hex.Dump(body[:])))
}
conn.Write([]byte("HTTP/1.1 200 OK\r\n\r\n"))
return nil
}