2017-05-03 10:20:33 +02:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
|
2018-07-06 10:32:03 +02:00
|
|
|
"github.com/containous/traefik/log"
|
2017-05-03 10:20:33 +02:00
|
|
|
"github.com/containous/traefik/middlewares"
|
|
|
|
)
|
|
|
|
|
|
|
|
// RecordingErrorHandler is an error handler, implementing the vulcand/oxy
|
|
|
|
// error handler interface, which is recording network errors by using the netErrorRecorder.
|
|
|
|
// In addition it sets a proper HTTP status code and body, depending on the type of error occurred.
|
|
|
|
type RecordingErrorHandler struct {
|
|
|
|
netErrorRecorder middlewares.NetErrorRecorder
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewRecordingErrorHandler creates and returns a new instance of RecordingErrorHandler.
|
|
|
|
func NewRecordingErrorHandler(recorder middlewares.NetErrorRecorder) *RecordingErrorHandler {
|
|
|
|
return &RecordingErrorHandler{recorder}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (eh *RecordingErrorHandler) ServeHTTP(w http.ResponseWriter, req *http.Request, err error) {
|
|
|
|
statusCode := http.StatusInternalServerError
|
|
|
|
|
|
|
|
if e, ok := err.(net.Error); ok {
|
|
|
|
eh.netErrorRecorder.Record(req.Context())
|
|
|
|
if e.Timeout() {
|
|
|
|
statusCode = http.StatusGatewayTimeout
|
|
|
|
} else {
|
|
|
|
statusCode = http.StatusBadGateway
|
|
|
|
}
|
|
|
|
} else if err == io.EOF {
|
|
|
|
eh.netErrorRecorder.Record(req.Context())
|
|
|
|
statusCode = http.StatusBadGateway
|
|
|
|
}
|
|
|
|
|
|
|
|
w.WriteHeader(statusCode)
|
|
|
|
w.Write([]byte(http.StatusText(statusCode)))
|
2018-07-06 10:32:03 +02:00
|
|
|
log.Debugf("'%d %s' caused by: %v", statusCode, http.StatusText(statusCode), err)
|
2017-05-03 10:20:33 +02:00
|
|
|
}
|