-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathresponsewriter.go
121 lines (103 loc) · 3.56 KB
/
responsewriter.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
113
114
115
116
117
118
119
120
121
package sigsci
import (
"bufio"
"fmt"
"io"
"net"
"net/http"
)
// ResponseWriter is a http.ResponseWriter allowing extraction of data needed for inspection
type ResponseWriter interface {
http.ResponseWriter
BaseResponseWriter() http.ResponseWriter
StatusCode() int
BytesWritten() int64
}
// ResponseWriterFlusher is a ResponseWriter with a http.Flusher interface
type ResponseWriterFlusher interface {
ResponseWriter
http.Flusher
}
// NewResponseWriter returns a ResponseWriter or ResponseWriterFlusher depending on the base http.ResponseWriter.
func NewResponseWriter(base http.ResponseWriter) ResponseWriter {
// NOTE: according to net/http docs, if WriteHeader is not called explicitly,
// the first call to Write will trigger an implicit WriteHeader(http.StatusOK).
// this is why the default code is 200 and it only changes if WriteHeader is called.
w := &responseRecorder{
base: base,
code: 200,
}
if _, ok := w.base.(http.Flusher); ok {
return &responseRecorderFlusher{w}
}
return w
}
// responseRecorder wraps a base http.ResponseWriter allowing extraction of additional inspection data
type responseRecorder struct {
base http.ResponseWriter
code int
size int64
}
// BaseResponseWriter returns the base http.ResponseWriter allowing access if needed
func (w *responseRecorder) BaseResponseWriter() http.ResponseWriter {
return w.base
}
// StatusCode returns the status code that was used
func (w *responseRecorder) StatusCode() int {
return w.code
}
// BytesWritten returns the number of bytes written
func (w *responseRecorder) BytesWritten() int64 {
return w.size
}
// Header returns the header object
func (w *responseRecorder) Header() http.Header {
return w.base.Header()
}
// WriteHeader writes the header, recording the status code for inspection
func (w *responseRecorder) WriteHeader(status int) {
w.code = status
w.base.WriteHeader(status)
}
// Write writes data, tracking the length written for inspection
func (w *responseRecorder) Write(b []byte) (int, error) {
w.size += int64(len(b))
return w.base.Write(b)
}
func (w *responseRecorder) ReadFrom(r io.Reader) (n int64, err error) {
if rf, ok := w.base.(io.ReaderFrom); ok {
return rf.ReadFrom(r)
}
return io.Copy(w.base, r)
}
// Hijack hijacks the connection from the HTTP handler so that it can be used directly (websockets, etc.)
// NOTE: This will fail if the wrapped http.responseRecorder is not a http.Hijacker.
func (w *responseRecorder) Hijack() (net.Conn, *bufio.ReadWriter, error) {
if h, ok := w.base.(http.Hijacker); ok {
return h.Hijack()
}
// Required for WebSockets to work
return nil, nil, fmt.Errorf("response writer (%T) does not implement http.Hijacker", w.base)
}
// CloseNotify wraps the underlying CloseNotify or returns a dummy channel if the CloseNotifier interface is not implemented
func (w *responseRecorder) CloseNotify() <-chan bool {
if cn, ok := w.base.(http.CloseNotifier); ok {
return cn.CloseNotify()
}
// Return a dummy channel that will never get used
return make(<-chan bool)
}
// responseRecorderFlusher wraps a base http.ResponseWriter/http.Flusher allowing extraction of additional inspection data
type responseRecorderFlusher struct {
*responseRecorder
}
// Flush flushes data if the underlying http.ResponseWriter is capable of flushing
func (w *responseRecorderFlusher) Flush() {
if f, ok := w.responseRecorder.base.(http.Flusher); ok {
f.Flush()
}
}
// ensure our writers satisfy the intended interfaces
var _ http.Hijacker = (*responseRecorder)(nil)
var _ io.ReaderFrom = (*responseRecorder)(nil)
var _ http.Flusher = (*responseRecorderFlusher)(nil)