2018-11-14 09:18:03 +00:00
package service
import (
"context"
2019-08-26 08:30:05 +00:00
"errors"
2018-11-14 09:18:03 +00:00
"fmt"
"net/http"
"net/http/httputil"
"net/url"
2019-08-26 17:00:04 +00:00
"reflect"
2018-11-14 09:18:03 +00:00
"time"
2019-01-18 14:18:04 +00:00
"github.com/containous/alice"
2019-08-03 01:58:23 +00:00
"github.com/containous/traefik/v2/pkg/config/dynamic"
"github.com/containous/traefik/v2/pkg/config/runtime"
"github.com/containous/traefik/v2/pkg/healthcheck"
"github.com/containous/traefik/v2/pkg/log"
"github.com/containous/traefik/v2/pkg/metrics"
"github.com/containous/traefik/v2/pkg/middlewares/accesslog"
"github.com/containous/traefik/v2/pkg/middlewares/emptybackendhandler"
metricsMiddle "github.com/containous/traefik/v2/pkg/middlewares/metrics"
"github.com/containous/traefik/v2/pkg/middlewares/pipelining"
2019-08-26 17:00:04 +00:00
"github.com/containous/traefik/v2/pkg/safe"
2019-08-03 01:58:23 +00:00
"github.com/containous/traefik/v2/pkg/server/cookie"
"github.com/containous/traefik/v2/pkg/server/internal"
2019-08-26 17:00:04 +00:00
"github.com/containous/traefik/v2/pkg/server/service/loadbalancer/mirror"
2019-08-26 08:30:05 +00:00
"github.com/containous/traefik/v2/pkg/server/service/loadbalancer/wrr"
2018-11-14 09:18:03 +00:00
"github.com/vulcand/oxy/roundrobin"
)
const (
defaultHealthCheckInterval = 30 * time . Second
defaultHealthCheckTimeout = 5 * time . Second
)
// NewManager creates a new Manager
2019-09-06 13:08:04 +00:00
func NewManager ( configs map [ string ] * runtime . ServiceInfo , defaultRoundTripper http . RoundTripper , metricsRegistry metrics . Registry , routinePool * safe . Pool , api http . Handler , rest http . Handler ) * Manager {
2018-11-14 09:18:03 +00:00
return & Manager {
2019-08-26 17:00:04 +00:00
routinePool : routinePool ,
2019-07-18 19:36:05 +00:00
metricsRegistry : metricsRegistry ,
2018-11-14 09:18:03 +00:00
bufferPool : newBufferPool ( ) ,
defaultRoundTripper : defaultRoundTripper ,
balancers : make ( map [ string ] [ ] healthcheck . BalancerHandler ) ,
configs : configs ,
2019-09-06 13:08:04 +00:00
api : api ,
rest : rest ,
2018-11-14 09:18:03 +00:00
}
}
// Manager The service manager
type Manager struct {
2019-08-26 17:00:04 +00:00
routinePool * safe . Pool
2019-07-18 19:36:05 +00:00
metricsRegistry metrics . Registry
2018-11-14 09:18:03 +00:00
bufferPool httputil . BufferPool
defaultRoundTripper http . RoundTripper
balancers map [ string ] [ ] healthcheck . BalancerHandler
2019-07-15 15:04:04 +00:00
configs map [ string ] * runtime . ServiceInfo
2019-09-06 13:08:04 +00:00
api http . Handler
rest http . Handler
2018-11-14 09:18:03 +00:00
}
2019-03-14 08:30:04 +00:00
// BuildHTTP Creates a http.Handler for a service configuration.
func ( m * Manager ) BuildHTTP ( rootCtx context . Context , serviceName string , responseModifier func ( * http . Response ) error ) ( http . Handler , error ) {
2019-09-06 13:08:04 +00:00
if serviceName == "api@internal" {
if m . api == nil {
return nil , errors . New ( "api is not enabled" )
}
return m . api , nil
}
if serviceName == "rest@internal" {
if m . rest == nil {
return nil , errors . New ( "rest is not enabled" )
}
return m . rest , nil
}
2018-11-14 09:18:03 +00:00
ctx := log . With ( rootCtx , log . Str ( log . ServiceName , serviceName ) )
2019-01-15 13:28:04 +00:00
serviceName = internal . GetQualifiedName ( ctx , serviceName )
ctx = internal . AddProviderInContext ( ctx , serviceName )
2019-05-16 08:58:06 +00:00
conf , ok := m . configs [ serviceName ]
if ! ok {
return nil , fmt . Errorf ( "the service %q does not exist" , serviceName )
}
2019-08-26 17:00:04 +00:00
value := reflect . ValueOf ( * conf . Service )
var count int
for i := 0 ; i < value . NumField ( ) ; i ++ {
if ! value . Field ( i ) . IsNil ( ) {
count ++
}
}
if count > 1 {
2019-09-13 18:00:06 +00:00
err := errors . New ( "cannot create service: multi-types service not supported, consider declaring two different pieces of service instead" )
conf . AddError ( err , true )
return nil , err
2019-08-26 08:30:05 +00:00
}
var lb http . Handler
switch {
case conf . LoadBalancer != nil :
var err error
lb , err = m . getLoadBalancerServiceHandler ( ctx , serviceName , conf . LoadBalancer , responseModifier )
if err != nil {
conf . AddError ( err , true )
return nil , err
}
case conf . Weighted != nil :
var err error
lb , err = m . getLoadBalancerWRRServiceHandler ( ctx , serviceName , conf . Weighted , responseModifier )
if err != nil {
conf . AddError ( err , true )
return nil , err
}
2019-08-26 17:00:04 +00:00
case conf . Mirroring != nil :
var err error
lb , err = m . getLoadBalancerMirrorServiceHandler ( ctx , serviceName , conf . Mirroring , responseModifier )
if err != nil {
conf . AddError ( err , true )
return nil , err
}
2019-08-26 08:30:05 +00:00
default :
sErr := fmt . Errorf ( "the service %q does not have any type defined" , serviceName )
2019-07-15 15:04:04 +00:00
conf . AddError ( sErr , true )
return nil , sErr
2018-11-14 09:18:03 +00:00
}
2019-05-16 08:58:06 +00:00
2019-08-26 08:30:05 +00:00
return lb , nil
}
2019-08-26 17:00:04 +00:00
func ( m * Manager ) getLoadBalancerMirrorServiceHandler ( ctx context . Context , serviceName string , config * dynamic . Mirroring , responseModifier func ( * http . Response ) error ) ( http . Handler , error ) {
serviceHandler , err := m . BuildHTTP ( ctx , config . Service , responseModifier )
if err != nil {
return nil , err
}
handler := mirror . New ( serviceHandler , m . routinePool )
for _ , mirrorConfig := range config . Mirrors {
mirrorHandler , err := m . BuildHTTP ( ctx , mirrorConfig . Name , responseModifier )
if err != nil {
return nil , err
}
err = handler . AddMirror ( mirrorHandler , mirrorConfig . Percent )
if err != nil {
return nil , err
}
}
return handler , nil
}
2019-08-26 08:30:05 +00:00
func ( m * Manager ) getLoadBalancerWRRServiceHandler ( ctx context . Context , serviceName string , config * dynamic . WeightedRoundRobin , responseModifier func ( * http . Response ) error ) ( http . Handler , error ) {
// TODO Handle accesslog and metrics with multiple service name
if config . Sticky != nil && config . Sticky . Cookie != nil {
config . Sticky . Cookie . Name = cookie . GetName ( config . Sticky . Cookie . Name , serviceName )
2019-05-16 08:58:06 +00:00
}
2019-08-26 08:30:05 +00:00
balancer := wrr . New ( config . Sticky )
for _ , service := range config . Services {
serviceHandler , err := m . BuildHTTP ( ctx , service . Name , responseModifier )
if err != nil {
return nil , err
}
balancer . AddService ( service . Name , serviceHandler , service . Weight )
}
return balancer , nil
2018-11-14 09:18:03 +00:00
}
func ( m * Manager ) getLoadBalancerServiceHandler (
ctx context . Context ,
serviceName string ,
2019-08-26 08:30:05 +00:00
service * dynamic . ServersLoadBalancer ,
2018-11-14 09:18:03 +00:00
responseModifier func ( * http . Response ) error ,
) ( http . Handler , error ) {
2019-09-30 16:12:04 +00:00
if service . PassHostHeader == nil {
defaultPassHostHeader := true
service . PassHostHeader = & defaultPassHostHeader
}
2019-03-14 08:30:04 +00:00
fwd , err := buildProxy ( service . PassHostHeader , service . ResponseForwarding , m . defaultRoundTripper , m . bufferPool , responseModifier )
2018-11-14 09:18:03 +00:00
if err != nil {
return nil , err
}
2019-01-18 14:18:04 +00:00
alHandler := func ( next http . Handler ) ( http . Handler , error ) {
return accesslog . NewFieldHandler ( next , accesslog . ServiceName , serviceName , accesslog . AddServiceFields ) , nil
}
2019-07-18 19:36:05 +00:00
chain := alice . New ( )
if m . metricsRegistry != nil && m . metricsRegistry . IsSvcEnabled ( ) {
chain = chain . Append ( metricsMiddle . WrapServiceHandler ( ctx , m . metricsRegistry , serviceName ) )
}
2018-11-14 09:18:03 +00:00
2019-07-18 19:36:05 +00:00
handler , err := chain . Append ( alHandler ) . Then ( pipelining . New ( ctx , fwd , "pipelining" ) )
2018-11-14 09:18:03 +00:00
if err != nil {
return nil , err
}
2019-01-18 14:18:04 +00:00
balancer , err := m . getLoadBalancer ( ctx , serviceName , service , handler )
2018-11-14 09:18:03 +00:00
if err != nil {
return nil , err
}
// TODO rename and checks
m . balancers [ serviceName ] = append ( m . balancers [ serviceName ] , balancer )
// Empty (backend with no servers)
return emptybackendhandler . New ( balancer ) , nil
}
// LaunchHealthCheck Launches the health checks.
func ( m * Manager ) LaunchHealthCheck ( ) {
backendConfigs := make ( map [ string ] * healthcheck . BackendConfig )
for serviceName , balancers := range m . balancers {
ctx := log . With ( context . Background ( ) , log . Str ( log . ServiceName , serviceName ) )
2019-03-18 10:30:07 +00:00
// TODO aggregate
2018-11-14 09:18:03 +00:00
balancer := balancers [ 0 ]
2019-03-18 10:30:07 +00:00
// TODO Should all the services handle healthcheck? Handle different types
2018-11-14 09:18:03 +00:00
service := m . configs [ serviceName ] . LoadBalancer
// Health Check
var backendHealthCheck * healthcheck . BackendConfig
if hcOpts := buildHealthCheckOptions ( ctx , balancer , serviceName , service . HealthCheck ) ; hcOpts != nil {
log . FromContext ( ctx ) . Debugf ( "Setting up healthcheck for service %s with %s" , serviceName , * hcOpts )
hcOpts . Transport = m . defaultRoundTripper
backendHealthCheck = healthcheck . NewBackendConfig ( * hcOpts , serviceName )
}
if backendHealthCheck != nil {
backendConfigs [ serviceName ] = backendHealthCheck
}
}
// FIXME metrics and context
2019-09-13 17:28:04 +00:00
healthcheck . GetHealthCheck ( ) . SetBackendsConfiguration ( context . Background ( ) , backendConfigs )
2018-11-14 09:18:03 +00:00
}
2019-07-10 07:26:04 +00:00
func buildHealthCheckOptions ( ctx context . Context , lb healthcheck . BalancerHandler , backend string , hc * dynamic . HealthCheck ) * healthcheck . Options {
2018-11-14 09:18:03 +00:00
if hc == nil || hc . Path == "" {
return nil
}
logger := log . FromContext ( ctx )
interval := defaultHealthCheckInterval
if hc . Interval != "" {
intervalOverride , err := time . ParseDuration ( hc . Interval )
2019-02-05 16:10:03 +00:00
switch {
case err != nil :
2018-11-14 09:18:03 +00:00
logger . Errorf ( "Illegal health check interval for '%s': %s" , backend , err )
2019-02-05 16:10:03 +00:00
case intervalOverride <= 0 :
2018-11-14 09:18:03 +00:00
logger . Errorf ( "Health check interval smaller than zero for service '%s'" , backend )
2019-02-05 16:10:03 +00:00
default :
2018-11-14 09:18:03 +00:00
interval = intervalOverride
}
}
timeout := defaultHealthCheckTimeout
if hc . Timeout != "" {
timeoutOverride , err := time . ParseDuration ( hc . Timeout )
2019-02-05 16:10:03 +00:00
switch {
case err != nil :
2018-11-14 09:18:03 +00:00
logger . Errorf ( "Illegal health check timeout for backend '%s': %s" , backend , err )
2019-02-05 16:10:03 +00:00
case timeoutOverride <= 0 :
2018-11-14 09:18:03 +00:00
logger . Errorf ( "Health check timeout smaller than zero for backend '%s', backend" , backend )
2019-02-05 16:10:03 +00:00
default :
2018-11-14 09:18:03 +00:00
timeout = timeoutOverride
}
}
if timeout >= interval {
2019-05-16 08:58:06 +00:00
logger . Warnf ( "Health check timeout for backend '%s' should be lower than the health check interval. Interval set to timeout + 1 second (%s)." , backend , interval )
2018-11-14 09:18:03 +00:00
}
return & healthcheck . Options {
Scheme : hc . Scheme ,
Path : hc . Path ,
Port : hc . Port ,
Interval : interval ,
Timeout : timeout ,
LB : lb ,
Hostname : hc . Hostname ,
Headers : hc . Headers ,
}
}
2019-08-26 08:30:05 +00:00
func ( m * Manager ) getLoadBalancer ( ctx context . Context , serviceName string , service * dynamic . ServersLoadBalancer , fwd http . Handler ) ( healthcheck . BalancerHandler , error ) {
2018-11-14 09:18:03 +00:00
logger := log . FromContext ( ctx )
2019-06-05 20:18:06 +00:00
logger . Debug ( "Creating load-balancer" )
var options [ ] roundrobin . LBOption
2018-11-14 09:18:03 +00:00
var cookieName string
2019-08-26 08:30:05 +00:00
if service . Sticky != nil && service . Sticky . Cookie != nil {
cookieName = cookie . GetName ( service . Sticky . Cookie . Name , serviceName )
opts := roundrobin . CookieOptions { HTTPOnly : service . Sticky . Cookie . HTTPOnly , Secure : service . Sticky . Cookie . Secure }
2019-06-12 22:42:06 +00:00
options = append ( options , roundrobin . EnableStickySession ( roundrobin . NewStickySessionWithOptions ( cookieName , opts ) ) )
2019-06-05 20:18:06 +00:00
logger . Debugf ( "Sticky session cookie name: %v" , cookieName )
2018-11-14 09:18:03 +00:00
}
2019-06-05 20:18:06 +00:00
lb , err := roundrobin . New ( fwd , options ... )
if err != nil {
return nil , err
2018-11-14 09:18:03 +00:00
}
2019-05-16 08:58:06 +00:00
lbsu := healthcheck . NewLBStatusUpdater ( lb , m . configs [ serviceName ] )
if err := m . upsertServers ( ctx , lbsu , service . Servers ) ; err != nil {
2018-11-14 09:18:03 +00:00
return nil , fmt . Errorf ( "error configuring load balancer for service %s: %v" , serviceName , err )
}
2019-09-18 15:56:05 +00:00
return lbsu , nil
2018-11-14 09:18:03 +00:00
}
2019-07-10 07:26:04 +00:00
func ( m * Manager ) upsertServers ( ctx context . Context , lb healthcheck . BalancerHandler , servers [ ] dynamic . Server ) error {
2018-11-14 09:18:03 +00:00
logger := log . FromContext ( ctx )
for name , srv := range servers {
u , err := url . Parse ( srv . URL )
if err != nil {
return fmt . Errorf ( "error parsing server URL %s: %v" , srv . URL , err )
}
2019-06-05 20:18:06 +00:00
logger . WithField ( log . ServerName , name ) . Debugf ( "Creating server %d %s" , name , u )
2018-11-14 09:18:03 +00:00
2019-06-05 20:18:06 +00:00
if err := lb . UpsertServer ( u , roundrobin . Weight ( 1 ) ) ; err != nil {
2018-11-14 09:18:03 +00:00
return fmt . Errorf ( "error adding server %s to load balancer: %v" , srv . URL , err )
}
// FIXME Handle Metrics
}
return nil
}