Skip to content

Commit e0daa39

Browse files
authored
caddyhttp: record num. bytes read when response writer is hijacked (#6173)
* record the number of bytes read when response writer is hijacked * record body size when not nil
1 parent 70953e8 commit e0daa39

File tree

2 files changed

+42
-0
lines changed

2 files changed

+42
-0
lines changed

modules/caddyhttp/responsewriter.go

Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,8 @@ type responseRecorder struct {
6666
size int
6767
wroteHeader bool
6868
stream bool
69+
70+
readSize *int
6971
}
7072

7173
// NewResponseRecorder returns a new ResponseRecorder that can be
@@ -240,6 +242,12 @@ func (rr *responseRecorder) FlushError() error {
240242
return nil
241243
}
242244

245+
// Private interface so it can only be used in this package
246+
// #TODO: maybe export it later
247+
func (rr *responseRecorder) setReadSize(size *int) {
248+
rr.readSize = size
249+
}
250+
243251
func (rr *responseRecorder) Hijack() (net.Conn, *bufio.ReadWriter, error) {
244252
//nolint:bodyclose
245253
conn, brw, err := http.NewResponseController(rr.ResponseWriterWrapper).Hijack()
@@ -249,6 +257,15 @@ func (rr *responseRecorder) Hijack() (net.Conn, *bufio.ReadWriter, error) {
249257
// Per http documentation, returned bufio.Writer is empty, but bufio.Read maybe not
250258
conn = &hijackedConn{conn, rr}
251259
brw.Writer.Reset(conn)
260+
261+
buffered := brw.Reader.Buffered()
262+
if buffered != 0 {
263+
conn.(*hijackedConn).updateReadSize(buffered)
264+
data, _ := brw.Peek(buffered)
265+
brw.Reader.Reset(io.MultiReader(bytes.NewReader(data), conn))
266+
} else {
267+
brw.Reader.Reset(conn)
268+
}
252269
return conn, brw, nil
253270
}
254271

@@ -258,6 +275,24 @@ type hijackedConn struct {
258275
rr *responseRecorder
259276
}
260277

278+
func (hc *hijackedConn) updateReadSize(n int) {
279+
if hc.rr.readSize != nil {
280+
*hc.rr.readSize += n
281+
}
282+
}
283+
284+
func (hc *hijackedConn) Read(p []byte) (int, error) {
285+
n, err := hc.Conn.Read(p)
286+
hc.updateReadSize(n)
287+
return n, err
288+
}
289+
290+
func (hc *hijackedConn) WriteTo(w io.Writer) (int64, error) {
291+
n, err := io.Copy(w, hc.Conn)
292+
hc.updateReadSize(int(n))
293+
return n, err
294+
}
295+
261296
func (hc *hijackedConn) Write(p []byte) (int, error) {
262297
n, err := hc.Conn.Write(p)
263298
hc.rr.size += n
@@ -298,4 +333,6 @@ var (
298333
_ io.ReaderFrom = (*ResponseWriterWrapper)(nil)
299334
_ io.ReaderFrom = (*responseRecorder)(nil)
300335
_ io.ReaderFrom = (*hijackedConn)(nil)
336+
337+
_ io.WriterTo = (*hijackedConn)(nil)
301338
)

modules/caddyhttp/server.go

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -326,6 +326,11 @@ func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
326326
if r.Body != nil {
327327
bodyReader = &lengthReader{Source: r.Body}
328328
r.Body = bodyReader
329+
330+
// should always be true, private interface can only be referenced in the same package
331+
if setReadSizer, ok := wrec.(interface{ setReadSize(*int) }); ok {
332+
setReadSizer.setReadSize(&bodyReader.Length)
333+
}
329334
}
330335

331336
// capture the original version of the request

0 commit comments

Comments
 (0)