traefik/healthcheck/healthcheck.go

169 lines
4.3 KiB
Go
Raw Normal View History

2016-11-26 19:48:49 +01:00
package healthcheck
import (
"context"
"fmt"
"net"
2016-11-26 19:48:49 +01:00
"net/http"
"net/url"
"strconv"
2016-11-26 19:48:49 +01:00
"sync"
"time"
2017-01-31 22:55:02 +01:00
"github.com/containous/traefik/log"
"github.com/containous/traefik/safe"
"github.com/vulcand/oxy/roundrobin"
2016-11-26 19:48:49 +01:00
)
var singleton *HealthCheck
var once sync.Once
// GetHealthCheck returns the health check which is guaranteed to be a singleton.
2016-11-26 19:48:49 +01:00
func GetHealthCheck() *HealthCheck {
once.Do(func() {
singleton = newHealthCheck()
})
return singleton
}
// Options are the public health check options.
type Options struct {
2017-11-21 05:06:03 -05:00
Path string
Port int
Transport http.RoundTripper
Interval time.Duration
LB LoadBalancer
}
func (opt Options) String() string {
2017-09-18 15:50:03 +02:00
return fmt.Sprintf("[Path: %s Port: %d Interval: %s]", opt.Path, opt.Port, opt.Interval)
}
2016-11-26 19:48:49 +01:00
// BackendHealthCheck HealthCheck configuration for a backend
type BackendHealthCheck struct {
Options
disabledURLs []*url.URL
requestTimeout time.Duration
2016-11-26 19:48:49 +01:00
}
//HealthCheck struct
type HealthCheck struct {
2018-01-15 10:46:04 +00:00
mutex sync.Mutex
2016-11-26 19:48:49 +01:00
Backends map[string]*BackendHealthCheck
cancel context.CancelFunc
2016-11-26 19:48:49 +01:00
}
// LoadBalancer includes functionality for load-balancing management.
type LoadBalancer interface {
2016-11-26 19:48:49 +01:00
RemoveServer(u *url.URL) error
UpsertServer(u *url.URL, options ...roundrobin.ServerOption) error
Servers() []*url.URL
}
func newHealthCheck() *HealthCheck {
return &HealthCheck{
Backends: make(map[string]*BackendHealthCheck),
}
2016-11-26 19:48:49 +01:00
}
// NewBackendHealthCheck Instantiate a new BackendHealthCheck
func NewBackendHealthCheck(options Options) *BackendHealthCheck {
return &BackendHealthCheck{
Options: options,
requestTimeout: 5 * time.Second,
}
2016-11-26 19:48:49 +01:00
}
2016-11-29 19:30:51 +01:00
//SetBackendsConfiguration set backends configuration
2017-02-06 09:31:20 +01:00
func (hc *HealthCheck) SetBackendsConfiguration(parentCtx context.Context, backends map[string]*BackendHealthCheck) {
2018-01-15 10:46:04 +00:00
hc.mutex.Lock()
2016-11-26 19:48:49 +01:00
hc.Backends = backends
if hc.cancel != nil {
hc.cancel()
}
2017-01-31 22:55:02 +01:00
ctx, cancel := context.WithCancel(parentCtx)
hc.cancel = cancel
2018-01-15 10:46:04 +00:00
hc.mutex.Unlock()
2016-11-26 19:48:49 +01:00
2018-01-15 10:46:04 +00:00
for backendID, backend := range backends {
2017-01-31 22:55:02 +01:00
currentBackendID := backendID
currentBackend := backend
2017-01-31 22:55:02 +01:00
safe.Go(func() {
hc.execute(ctx, currentBackendID, currentBackend)
2017-01-31 22:55:02 +01:00
})
}
2016-11-26 19:48:49 +01:00
}
func (hc *HealthCheck) execute(ctx context.Context, backendID string, backend *BackendHealthCheck) {
log.Debugf("Initial healthcheck for currentBackend %s ", backendID)
checkBackend(backend)
ticker := time.NewTicker(backend.Interval)
defer ticker.Stop()
for {
select {
case <-ctx.Done():
log.Debug("Stopping all current Healthcheck goroutines")
return
case <-ticker.C:
log.Debugf("Refreshing healthcheck for currentBackend %s ", backendID)
checkBackend(backend)
}
}
}
func checkBackend(currentBackend *BackendHealthCheck) {
enabledURLs := currentBackend.LB.Servers()
var newDisabledURLs []*url.URL
for _, url := range currentBackend.disabledURLs {
if checkHealth(url, currentBackend) {
log.Debugf("HealthCheck is up [%s]: Upsert in server list", url.String())
currentBackend.LB.UpsertServer(url, roundrobin.Weight(1))
} else {
log.Warnf("HealthCheck is still failing [%s]", url.String())
newDisabledURLs = append(newDisabledURLs, url)
}
}
currentBackend.disabledURLs = newDisabledURLs
for _, url := range enabledURLs {
if !checkHealth(url, currentBackend) {
log.Warnf("HealthCheck has failed [%s]: Remove from server list", url.String())
currentBackend.LB.RemoveServer(url)
currentBackend.disabledURLs = append(currentBackend.disabledURLs, url)
}
}
}
func (backend *BackendHealthCheck) newRequest(serverURL *url.URL) (*http.Request, error) {
2017-09-18 15:50:03 +02:00
if backend.Port == 0 {
2017-11-20 09:40:03 +01:00
return http.NewRequest(http.MethodGet, serverURL.String()+backend.Path, nil)
}
// copy the url and add the port to the host
u := &url.URL{}
*u = *serverURL
2017-09-18 15:50:03 +02:00
u.Host = net.JoinHostPort(u.Hostname(), strconv.Itoa(backend.Port))
u.Path = u.Path + backend.Path
2017-11-20 09:40:03 +01:00
return http.NewRequest(http.MethodGet, u.String(), nil)
}
func checkHealth(serverURL *url.URL, backend *BackendHealthCheck) bool {
2016-11-30 22:48:09 +01:00
client := http.Client{
2017-11-21 05:06:03 -05:00
Timeout: backend.requestTimeout,
Transport: backend.Options.Transport,
2016-11-30 22:48:09 +01:00
}
req, err := backend.newRequest(serverURL)
if err != nil {
log.Errorf("Failed to create HTTP request [%s] for healthcheck: %s", serverURL, err)
return false
}
resp, err := client.Do(req)
if err == nil {
defer resp.Body.Close()
2016-11-26 19:48:49 +01:00
}
2017-11-20 09:40:03 +01:00
return err == nil && resp.StatusCode == http.StatusOK
2016-11-26 19:48:49 +01:00
}