Fix PR comments
Signed-off-by: glefloch <glfloch@gmail.com>
This commit is contained in:
parent
fdf4cbc87b
commit
c2c6fd2ab4
|
@ -59,6 +59,12 @@ var gzipPool = sync.Pool{
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var bufPool = sync.Pool{
|
||||||
|
New: func() interface{} {
|
||||||
|
return &bytes.Buffer{}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
// Handler returns an http.Handler for the prometheus.DefaultGatherer, using
|
// Handler returns an http.Handler for the prometheus.DefaultGatherer, using
|
||||||
// default HandlerOpts, i.e. it reports the first error as an HTTP error, it has
|
// default HandlerOpts, i.e. it reports the first error as an HTTP error, it has
|
||||||
// no error logging, and it applies compression if requested by the client.
|
// no error logging, and it applies compression if requested by the client.
|
||||||
|
@ -103,7 +109,6 @@ func HandlerFor(reg prometheus.Gatherer, opts HandlerOpts) http.Handler {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
mfs, err := reg.Gather()
|
mfs, err := reg.Gather()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if opts.ErrorLog != nil {
|
if opts.ErrorLog != nil {
|
||||||
|
@ -124,9 +129,12 @@ func HandlerFor(reg prometheus.Gatherer, opts HandlerOpts) http.Handler {
|
||||||
}
|
}
|
||||||
|
|
||||||
contentType := expfmt.Negotiate(req.Header)
|
contentType := expfmt.Negotiate(req.Header)
|
||||||
buf := &bytes.Buffer{}
|
buf := bufPool.Get().(*bytes.Buffer)
|
||||||
|
buf.Reset()
|
||||||
enc := expfmt.NewEncoder(buf, contentType)
|
enc := expfmt.NewEncoder(buf, contentType)
|
||||||
|
|
||||||
|
defer bufPool.Put(buf)
|
||||||
|
|
||||||
var lastErr error
|
var lastErr error
|
||||||
for _, mf := range mfs {
|
for _, mf := range mfs {
|
||||||
if err := enc.Encode(mf); err != nil {
|
if err := enc.Encode(mf); err != nil {
|
||||||
|
@ -150,45 +158,30 @@ func HandlerFor(reg prometheus.Gatherer, opts HandlerOpts) http.Handler {
|
||||||
http.Error(w, "No metrics encoded, last error:\n\n"+lastErr.Error(), http.StatusInternalServerError)
|
http.Error(w, "No metrics encoded, last error:\n\n"+lastErr.Error(), http.StatusInternalServerError)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
header := w.Header()
|
||||||
|
header.Set(contentTypeHeader, string(contentType))
|
||||||
|
header.Set(contentLengthHeader, fmt.Sprint(buf.Len()))
|
||||||
|
|
||||||
w.Header().Set(contentTypeHeader, string(contentType))
|
if !opts.DisableCompression && gzipAccepted(req.Header) {
|
||||||
|
header.Set(contentEncodingHeader, "gzip")
|
||||||
if _, err := w.Write(buf.Bytes()); err != nil && opts.ErrorLog != nil {
|
|
||||||
opts.ErrorLog.Println("error while sending encoded metrics:", err)
|
|
||||||
}
|
|
||||||
// TODO(beorn7): Consider streaming serving of metrics.
|
|
||||||
})
|
|
||||||
|
|
||||||
gzipHandler := http.HandlerFunc(func(w http.ResponseWriter, req *http.Request) {
|
|
||||||
if opts.DisableCompression {
|
|
||||||
h.ServeHTTP(w, req)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
header := req.Header.Get(acceptEncodingHeader)
|
|
||||||
parts := strings.Split(header, ",")
|
|
||||||
for _, part := range parts {
|
|
||||||
part = strings.TrimSpace(part)
|
|
||||||
if part == "gzip" || strings.HasPrefix(part, "gzip;") {
|
|
||||||
|
|
||||||
w.Header().Set(contentEncodingHeader, "gzip")
|
|
||||||
gz := gzipPool.Get().(*gzip.Writer)
|
gz := gzipPool.Get().(*gzip.Writer)
|
||||||
defer gzipPool.Put(gz)
|
defer gzipPool.Put(gz)
|
||||||
|
|
||||||
gz.Reset(w)
|
gz.Reset(w)
|
||||||
defer gz.Close()
|
defer gz.Close()
|
||||||
|
|
||||||
h.ServeHTTP(gzipResponseWriter{gz, w}, req)
|
zipWriter := gzipResponseWriter{gz, w}
|
||||||
|
writeResult(zipWriter, buf, opts)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
}
|
writeResult(w, buf, opts)
|
||||||
h.ServeHTTP(w, req)
|
// TODO(beorn7): Consider streaming serving of metrics.
|
||||||
return
|
|
||||||
})
|
})
|
||||||
|
|
||||||
if opts.Timeout <= 0 {
|
if opts.Timeout <= 0 {
|
||||||
return gzipHandler
|
return h
|
||||||
}
|
}
|
||||||
return http.TimeoutHandler(gzipHandler, opts.Timeout, fmt.Sprintf(
|
return http.TimeoutHandler(h, opts.Timeout, fmt.Sprintf(
|
||||||
"Exceeded configured timeout of %v.\n",
|
"Exceeded configured timeout of %v.\n",
|
||||||
opts.Timeout,
|
opts.Timeout,
|
||||||
))
|
))
|
||||||
|
@ -313,3 +306,26 @@ type gzipResponseWriter struct {
|
||||||
func (w gzipResponseWriter) Write(b []byte) (int, error) {
|
func (w gzipResponseWriter) Write(b []byte) (int, error) {
|
||||||
return w.Writer.Write(b)
|
return w.Writer.Write(b)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// writeResult to buf using http.ResponseWriter.
|
||||||
|
// If ErrorLog is enabled, err is logged in.
|
||||||
|
func writeResult(w http.ResponseWriter, buf *bytes.Buffer, opts HandlerOpts) {
|
||||||
|
if _, err := w.Write(buf.Bytes()); err != nil && opts.ErrorLog != nil {
|
||||||
|
opts.ErrorLog.Println("error while sending encoded metrics:", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// gzipHandler return a http.HandlerFunc in charge of compressing the content
|
||||||
|
// of the given http.HandlerFunc
|
||||||
|
func gzipAccepted(header http.Header) bool {
|
||||||
|
|
||||||
|
a := header.Get(acceptEncodingHeader)
|
||||||
|
parts := strings.Split(a, ",")
|
||||||
|
for _, part := range parts {
|
||||||
|
part = strings.TrimSpace(part)
|
||||||
|
if part == "gzip" || strings.HasPrefix(part, "gzip;") {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue