Improve logging output for failing healthchecks

This commit is contained in:
Marco Jantke 2018-01-03 12:32:03 +01:00 committed by Traefiker
parent c84fb9895e
commit cbd54470ba
3 changed files with 40 additions and 32 deletions

View file

@ -42,6 +42,7 @@ func (opt Options) String() string {
// BackendHealthCheck HealthCheck configuration for a backend // BackendHealthCheck HealthCheck configuration for a backend
type BackendHealthCheck struct { type BackendHealthCheck struct {
Options Options
name string
disabledURLs []*url.URL disabledURLs []*url.URL
requestTimeout time.Duration requestTimeout time.Duration
} }
@ -66,9 +67,10 @@ func newHealthCheck() *HealthCheck {
} }
// NewBackendHealthCheck Instantiate a new BackendHealthCheck // NewBackendHealthCheck Instantiate a new BackendHealthCheck
func NewBackendHealthCheck(options Options) *BackendHealthCheck { func NewBackendHealthCheck(options Options, backendName string) *BackendHealthCheck {
return &BackendHealthCheck{ return &BackendHealthCheck{
Options: options, Options: options,
name: backendName,
requestTimeout: 5 * time.Second, requestTimeout: 5 * time.Second,
} }
} }
@ -82,51 +84,50 @@ func (hc *HealthCheck) SetBackendsConfiguration(parentCtx context.Context, backe
ctx, cancel := context.WithCancel(parentCtx) ctx, cancel := context.WithCancel(parentCtx)
hc.cancel = cancel hc.cancel = cancel
for backendID, backend := range hc.Backends { for _, backend := range hc.Backends {
currentBackendID := backendID
currentBackend := backend currentBackend := backend
safe.Go(func() { safe.Go(func() {
hc.execute(ctx, currentBackendID, currentBackend) hc.execute(ctx, currentBackend)
}) })
} }
} }
func (hc *HealthCheck) execute(ctx context.Context, backendID string, backend *BackendHealthCheck) { func (hc *HealthCheck) execute(ctx context.Context, backend *BackendHealthCheck) {
log.Debugf("Initial healthcheck for currentBackend %s ", backendID) log.Debugf("Initial health check for backend: %q", backend.name)
checkBackend(backend) hc.checkBackend(backend)
ticker := time.NewTicker(backend.Interval) ticker := time.NewTicker(backend.Interval)
defer ticker.Stop() defer ticker.Stop()
for { for {
select { select {
case <-ctx.Done(): case <-ctx.Done():
log.Debug("Stopping all current Healthcheck goroutines") log.Debug("Stopping current health check goroutines of backend: %s", backend.name)
return return
case <-ticker.C: case <-ticker.C:
log.Debugf("Refreshing healthcheck for currentBackend %s ", backendID) log.Debugf("Refreshing health check for backend: %s", backend.name)
checkBackend(backend) hc.checkBackend(backend)
} }
} }
} }
func checkBackend(currentBackend *BackendHealthCheck) { func (hc *HealthCheck) checkBackend(backend *BackendHealthCheck) {
enabledURLs := currentBackend.LB.Servers() enabledURLs := backend.LB.Servers()
var newDisabledURLs []*url.URL var newDisabledURLs []*url.URL
for _, url := range currentBackend.disabledURLs { for _, url := range backend.disabledURLs {
if checkHealth(url, currentBackend) { if err := checkHealth(url, backend); err == nil {
log.Debugf("HealthCheck is up [%s]: Upsert in server list", url.String()) log.Warnf("Health check up: Returning to server list. Backend: %q URL: %q", backend.name, url.String())
currentBackend.LB.UpsertServer(url, roundrobin.Weight(1)) backend.LB.UpsertServer(url, roundrobin.Weight(1))
} else { } else {
log.Warnf("HealthCheck is still failing [%s]", url.String()) log.Warnf("Health check still failing. Backend: %q URL: %q Reason: %s", backend.name, url.String(), err)
newDisabledURLs = append(newDisabledURLs, url) newDisabledURLs = append(newDisabledURLs, url)
} }
} }
currentBackend.disabledURLs = newDisabledURLs backend.disabledURLs = newDisabledURLs
for _, url := range enabledURLs { for _, url := range enabledURLs {
if !checkHealth(url, currentBackend) { if err := checkHealth(url, backend); err != nil {
log.Warnf("HealthCheck has failed [%s]: Remove from server list", url.String()) log.Warnf("Health check failed: Remove from server list. Backend: %q URL: %q Reason: %s", backend.name, url.String(), err)
currentBackend.LB.RemoveServer(url) backend.LB.RemoveServer(url)
currentBackend.disabledURLs = append(currentBackend.disabledURLs, url) backend.disabledURLs = append(backend.disabledURLs, url)
} }
} }
} }
@ -145,21 +146,28 @@ func (backend *BackendHealthCheck) newRequest(serverURL *url.URL) (*http.Request
return http.NewRequest(http.MethodGet, u.String(), nil) return http.NewRequest(http.MethodGet, u.String(), nil)
} }
func checkHealth(serverURL *url.URL, backend *BackendHealthCheck) bool { // checkHealth returns a nil error in case it was successful and otherwise
// a non-nil error with a meaningful description why the health check failed.
func checkHealth(serverURL *url.URL, backend *BackendHealthCheck) error {
client := http.Client{ client := http.Client{
Timeout: backend.requestTimeout, Timeout: backend.requestTimeout,
Transport: backend.Options.Transport, Transport: backend.Options.Transport,
} }
req, err := backend.newRequest(serverURL) req, err := backend.newRequest(serverURL)
if err != nil { if err != nil {
log.Errorf("Failed to create HTTP request [%s] for healthcheck: %s", serverURL, err) return fmt.Errorf("failed to create HTTP request: %s", err)
return false
} }
resp, err := client.Do(req) resp, err := client.Do(req)
if err == nil { if err == nil {
defer resp.Body.Close() defer resp.Body.Close()
} }
return err == nil && resp.StatusCode == http.StatusOK
switch {
case err != nil:
return fmt.Errorf("HTTP request failed: %s", err)
case resp.StatusCode != http.StatusOK:
return fmt.Errorf("received non-200 status code: %v", resp.StatusCode)
}
return nil
} }

View file

@ -81,7 +81,7 @@ func TestSetBackendsConfiguration(t *testing.T) {
Path: "/path", Path: "/path",
Interval: healthCheckInterval, Interval: healthCheckInterval,
LB: lb, LB: lb,
}) }, "backendName")
serverURL := testhelpers.MustParseURL(ts.URL) serverURL := testhelpers.MustParseURL(ts.URL)
if test.startHealthy { if test.startHealthy {
lb.servers = append(lb.servers, serverURL) lb.servers = append(lb.servers, serverURL)
@ -95,7 +95,7 @@ func TestSetBackendsConfiguration(t *testing.T) {
wg := sync.WaitGroup{} wg := sync.WaitGroup{}
wg.Add(1) wg.Add(1)
go func() { go func() {
check.execute(ctx, "id", backend) check.execute(ctx, backend)
wg.Done() wg.Done()
}() }()
@ -168,7 +168,7 @@ func TestNewRequest(t *testing.T) {
Options{ Options{
Path: test.path, Path: test.path,
Port: test.port, Port: test.port,
}) }, "backendName")
u := &url.URL{ u := &url.URL{
Scheme: "http", Scheme: "http",

View file

@ -1049,7 +1049,7 @@ func (s *Server) loadConfig(configurations types.Configurations, globalConfigura
if hcOpts != nil { if hcOpts != nil {
log.Debugf("Setting up backend health check %s", *hcOpts) log.Debugf("Setting up backend health check %s", *hcOpts)
hcOpts.Transport = s.defaultForwardingRoundTripper hcOpts.Transport = s.defaultForwardingRoundTripper
backendsHealthCheck[entryPointName+frontend.Backend] = healthcheck.NewBackendHealthCheck(*hcOpts) backendsHealthCheck[entryPointName+frontend.Backend] = healthcheck.NewBackendHealthCheck(*hcOpts, frontend.Backend)
} }
lb = middlewares.NewEmptyBackendHandler(rebalancer, lb) lb = middlewares.NewEmptyBackendHandler(rebalancer, lb)
case types.Wrr: case types.Wrr:
@ -1071,7 +1071,7 @@ func (s *Server) loadConfig(configurations types.Configurations, globalConfigura
if hcOpts != nil { if hcOpts != nil {
log.Debugf("Setting up backend health check %s", *hcOpts) log.Debugf("Setting up backend health check %s", *hcOpts)
hcOpts.Transport = s.defaultForwardingRoundTripper hcOpts.Transport = s.defaultForwardingRoundTripper
backendsHealthCheck[entryPointName+frontend.Backend] = healthcheck.NewBackendHealthCheck(*hcOpts) backendsHealthCheck[entryPointName+frontend.Backend] = healthcheck.NewBackendHealthCheck(*hcOpts, frontend.Backend)
} }
lb = middlewares.NewEmptyBackendHandler(rr, lb) lb = middlewares.NewEmptyBackendHandler(rr, lb)
} }