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
|
package http
import (
"context"
"encoding/json"
"net"
"net/http"
"time"
"github.com/mpolden/zdns/dns"
"github.com/mpolden/zdns/log"
)
// A Server defines paramaters for running an HTTP server. The HTTP server serves an API for inspecting cache contents
// and request log.
type Server struct {
server *http.Server
logger *log.Logger
}
type logEntry struct {
Time string `json:"time"`
RemoteAddr net.IP `json:"remote_addr"`
Qtype string `json:"type"`
Question string `json:"question"`
Answer string `json:"answer"`
}
type httpError struct {
err error
Status int `json:"status"`
Message string `json:"message"`
}
// NewServer creates a new HTTP server, serving logs from the given logger and listening on listenAddr.
func NewServer(logger *log.Logger, listenAddr string) *Server {
server := &http.Server{Addr: listenAddr}
s := &Server{logger: logger, server: server}
s.server.Handler = s.handler()
return s
}
func (s *Server) handler() http.Handler {
mux := http.NewServeMux()
mux.Handle("/log/v1/", appHandler(s.logHandler))
//mux.Handle("/cache/v1")
mux.Handle("/", appHandler(notFoundHandler))
return requestFilter(mux)
}
type appHandler func(http.ResponseWriter, *http.Request) (interface{}, *httpError)
func (fn appHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
data, e := fn(w, r)
if e != nil { // e is *Error, not os.Error.
if e.Message == "" {
e.Message = e.err.Error()
}
out, err := json.Marshal(e)
if err != nil {
panic(err)
}
w.WriteHeader(e.Status)
w.Write(out)
} else if data != nil {
out, err := json.Marshal(data)
if err != nil {
panic(err)
}
w.Write(out)
}
}
func requestFilter(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
w.Header().Set("Content-Type", "application/json")
next.ServeHTTP(w, r)
})
}
func notFoundHandler(w http.ResponseWriter, r *http.Request) (interface{}, *httpError) {
return nil, &httpError{
Status: http.StatusNotFound,
Message: "Resource not found",
}
}
func (s *Server) logHandler(w http.ResponseWriter, r *http.Request) (interface{}, *httpError) {
logEntries, err := s.logger.Get(100)
if err != nil {
return nil, &httpError{
err: err,
Status: http.StatusInternalServerError,
}
}
entries := make([]logEntry, 0, len(logEntries))
for _, entry := range logEntries {
dnsType := ""
switch entry.Qtype {
case dns.TypeA:
dnsType = "A"
case dns.TypeAAAA:
dnsType = "AAAA"
}
e := logEntry{
Time: entry.Time.UTC().Format(time.RFC3339),
RemoteAddr: entry.RemoteAddr,
Qtype: dnsType,
Question: entry.Question,
Answer: entry.Answer,
}
entries = append(entries, e)
}
return entries, nil
}
// Close shuts down the HTTP server.
func (s *Server) Close() error { return s.server.Shutdown(context.Background()) }
// ListenAndServe starts the HTTP server listening on the configured address.
func (s *Server) ListenAndServe() error {
s.logger.Printf("http server listening on http://%s", s.server.Addr)
err := s.server.ListenAndServe()
if err == http.ErrServerClosed {
return nil // Do not treat server closing as an error
}
return err
}
|