X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/a95f899d7ac84f29b3d019aa410d265bb40833e5..9992a8816837bd03a1beecd0c0e8082bd913319a:/sdk/go/httpserver/responsewriter.go?ds=sidebyside diff --git a/sdk/go/httpserver/responsewriter.go b/sdk/go/httpserver/responsewriter.go index b9f4c23a81..049a3f1aae 100644 --- a/sdk/go/httpserver/responsewriter.go +++ b/sdk/go/httpserver/responsewriter.go @@ -1,43 +1,88 @@ +// Copyright (C) The Arvados Authors. All rights reserved. +// +// SPDX-License-Identifier: Apache-2.0 + package httpserver import ( "net/http" ) -// ResponseWriter wraps http.ResponseWriter and exposes the status +const sniffBytes = 1024 + +type ResponseWriter interface { + http.ResponseWriter + WroteStatus() int + WroteBodyBytes() int + Sniffed() []byte +} + +// responseWriter wraps http.ResponseWriter and exposes the status // sent, the number of bytes sent to the client, and the last write // error. -type ResponseWriter struct { +type responseWriter struct { http.ResponseWriter - wroteStatus *int // Last status given to WriteHeader() - wroteBodyBytes *int // Bytes successfully written - err *error // Last error returned from Write() + wroteStatus int // First status given to WriteHeader() + wroteBodyBytes int // Bytes successfully written + err error // Last error returned from Write() + sniffed []byte } func WrapResponseWriter(orig http.ResponseWriter) ResponseWriter { - return ResponseWriter{orig, new(int), new(int), new(error)} + return &responseWriter{ResponseWriter: orig} } -func (w ResponseWriter) WriteHeader(s int) { - *w.wroteStatus = s +func (w *responseWriter) CloseNotify() <-chan bool { + if cn, ok := w.ResponseWriter.(http.CloseNotifier); ok { + return cn.CloseNotify() + } + return nil +} + +func (w *responseWriter) WriteHeader(s int) { + if w.wroteStatus == 0 { + w.wroteStatus = s + } + // ...else it's too late to change the status seen by the + // client -- but we call the wrapped WriteHeader() anyway so + // it can log a warning. w.ResponseWriter.WriteHeader(s) } -func (w ResponseWriter) Write(data []byte) (n int, err error) { +func (w *responseWriter) Write(data []byte) (n int, err error) { + if w.wroteStatus == 0 { + w.WriteHeader(http.StatusOK) + } else if w.wroteStatus >= 400 { + w.sniff(data) + } n, err = w.ResponseWriter.Write(data) - *w.wroteBodyBytes += n - *w.err = err + w.wroteBodyBytes += n + w.err = err return } -func (w ResponseWriter) WroteStatus() int { - return *w.wroteStatus +func (w *responseWriter) WroteStatus() int { + return w.wroteStatus +} + +func (w *responseWriter) WroteBodyBytes() int { + return w.wroteBodyBytes +} + +func (w *responseWriter) Err() error { + return w.err } -func (w ResponseWriter) WroteBodyBytes() int { - return *w.wroteBodyBytes +func (w *responseWriter) sniff(data []byte) { + max := sniffBytes - len(w.sniffed) + if max <= 0 { + return + } else if max < len(data) { + data = data[:max] + } + w.sniffed = append(w.sniffed, data...) } -func (w ResponseWriter) Err() error { - return *w.err +func (w *responseWriter) Sniffed() []byte { + return w.sniffed }