2017-04-17 23:47:53 +03:00
package server
2016-01-14 00:45:49 +03:00
import (
2016-08-16 18:26:10 +03:00
"context"
2016-01-14 00:45:49 +03:00
"crypto/tls"
2016-06-15 23:38:40 +03:00
"crypto/x509"
2016-01-29 22:34:17 +03:00
"encoding/json"
2016-01-14 00:45:49 +03:00
"errors"
2016-06-15 23:38:40 +03:00
"io/ioutil"
2017-08-18 16:34:04 +03:00
"net"
2016-02-26 17:29:53 +03:00
"net/http"
"net/url"
"os"
"os/signal"
"reflect"
"regexp"
"sort"
2017-04-04 12:36:23 +03:00
"sync"
2016-02-26 17:29:53 +03:00
"time"
2016-02-20 01:55:23 +03:00
2017-08-25 22:32:03 +03:00
"github.com/armon/go-proxyproto"
2016-06-03 18:58:33 +03:00
"github.com/containous/mux"
2016-08-18 14:03:10 +03:00
"github.com/containous/traefik/cluster"
2017-08-25 17:10:03 +03:00
"github.com/containous/traefik/configuration"
2016-11-26 21:48:49 +03:00
"github.com/containous/traefik/healthcheck"
2016-08-18 15:20:11 +03:00
"github.com/containous/traefik/log"
2017-08-23 21:46:03 +03:00
"github.com/containous/traefik/metrics"
2016-02-24 18:43:39 +03:00
"github.com/containous/traefik/middlewares"
2017-05-09 15:02:44 +03:00
"github.com/containous/traefik/middlewares/accesslog"
2017-09-18 18:48:07 +03:00
mauth "github.com/containous/traefik/middlewares/auth"
2016-02-24 18:43:39 +03:00
"github.com/containous/traefik/provider"
2016-03-31 19:57:08 +03:00
"github.com/containous/traefik/safe"
2016-02-24 18:43:39 +03:00
"github.com/containous/traefik/types"
2016-04-13 21:36:23 +03:00
"github.com/streamrail/concurrent-map"
2017-08-25 17:10:03 +03:00
thoas_stats "github.com/thoas/stats"
2017-07-19 13:02:51 +03:00
"github.com/urfave/negroni"
2016-06-15 20:07:33 +03:00
"github.com/vulcand/oxy/cbreaker"
"github.com/vulcand/oxy/connlimit"
"github.com/vulcand/oxy/forward"
"github.com/vulcand/oxy/roundrobin"
"github.com/vulcand/oxy/utils"
2017-08-18 16:34:04 +03:00
"golang.org/x/net/http2"
2016-01-14 00:45:49 +03:00
)
2017-08-25 17:10:03 +03:00
var (
oxyLogger = & OxyLogger { }
)
2016-01-14 00:46:44 +03:00
2016-01-14 00:45:49 +03:00
// Server is the reverse-proxy/load-balancer engine
type Server struct {
2017-08-18 16:34:04 +03:00
serverEntryPoints serverEntryPoints
configurationChan chan types . ConfigMessage
configurationValidatedChan chan types . ConfigMessage
signals chan os . Signal
stopChan chan bool
providers [ ] provider . Provider
currentConfigurations safe . Safe
2017-08-25 17:10:03 +03:00
globalConfiguration configuration . GlobalConfiguration
2017-08-18 16:34:04 +03:00
accessLoggerMiddleware * accesslog . LogHandler
routinesPool * safe . Pool
leadership * cluster . Leadership
defaultForwardingRoundTripper http . RoundTripper
2017-08-23 21:46:03 +03:00
metricsRegistry metrics . Registry
2016-01-14 00:45:49 +03:00
}
2016-02-25 20:30:13 +03:00
type serverEntryPoints map [ string ] * serverEntryPoint
2016-01-29 22:34:17 +03:00
type serverEntryPoint struct {
2017-03-10 01:27:09 +03:00
httpServer * http . Server
2017-08-25 22:32:03 +03:00
listener net . Listener
2016-03-04 13:32:23 +03:00
httpRouter * middlewares . HandlerSwitcher
2016-01-29 22:34:17 +03:00
}
2016-03-27 03:05:17 +03:00
type serverRoute struct {
2017-03-24 14:07:59 +03:00
route * mux . Route
stripPrefixes [ ] string
stripPrefixesRegex [ ] string
addPrefix string
replacePath string
2016-03-27 03:05:17 +03:00
}
2016-01-14 00:45:49 +03:00
// NewServer returns an initialized Server.
2017-08-25 17:10:03 +03:00
func NewServer ( globalConfiguration configuration . GlobalConfiguration ) * Server {
2016-01-14 00:45:49 +03:00
server := new ( Server )
2016-02-25 20:30:13 +03:00
server . serverEntryPoints = make ( map [ string ] * serverEntryPoint )
2016-05-19 21:09:01 +03:00
server . configurationChan = make ( chan types . ConfigMessage , 100 )
server . configurationValidatedChan = make ( chan types . ConfigMessage , 100 )
2016-01-29 22:34:17 +03:00
server . signals = make ( chan os . Signal , 1 )
2016-04-13 21:36:23 +03:00
server . stopChan = make ( chan bool , 1 )
2016-01-14 00:45:49 +03:00
server . providers = [ ] provider . Provider { }
2017-08-11 13:04:58 +03:00
server . configureSignals ( )
2017-08-25 17:10:03 +03:00
currentConfigurations := make ( types . Configurations )
2016-04-13 21:36:23 +03:00
server . currentConfigurations . Set ( currentConfigurations )
2016-01-14 00:45:49 +03:00
server . globalConfiguration = globalConfiguration
2016-08-18 14:03:10 +03:00
server . routinesPool = safe . NewPool ( context . Background ( ) )
2017-08-18 16:34:04 +03:00
server . defaultForwardingRoundTripper = createHTTPTransport ( globalConfiguration )
2017-08-23 21:46:03 +03:00
server . metricsRegistry = metrics . NewVoidRegistry ( )
if globalConfiguration . Web != nil && globalConfiguration . Web . Metrics != nil {
server . registerMetricClients ( globalConfiguration . Web . Metrics )
}
2016-08-18 14:03:10 +03:00
if globalConfiguration . Cluster != nil {
// leadership creation if cluster mode
server . leadership = cluster . NewLeadership ( server . routinesPool . Ctx ( ) , globalConfiguration . Cluster )
}
2016-01-14 00:45:49 +03:00
2017-05-25 14:25:53 +03:00
if globalConfiguration . AccessLogsFile != "" {
globalConfiguration . AccessLog = & types . AccessLog { FilePath : globalConfiguration . AccessLogsFile , Format : accesslog . CommonFormat }
}
if globalConfiguration . AccessLog != nil {
var err error
server . accessLoggerMiddleware , err = accesslog . NewLogHandler ( globalConfiguration . AccessLog )
if err != nil {
log . Warnf ( "Unable to create log handler: %s" , err )
}
2017-05-22 22:39:29 +03:00
}
2016-01-14 00:45:49 +03:00
return server
}
2017-08-18 16:34:04 +03:00
// createHTTPTransport creates an http.Transport configured with the GlobalConfiguration settings.
// For the settings that can't be configured in Traefik it uses the default http.Transport settings.
// An exception to this is the MaxIdleConns setting as we only provide the option MaxIdleConnsPerHost
// in Traefik at this point in time. Setting this value to the default of 100 could lead to confusing
// behaviour and backwards compatibility issues.
2017-08-25 17:10:03 +03:00
func createHTTPTransport ( globalConfiguration configuration . GlobalConfiguration ) * http . Transport {
2017-08-18 16:34:04 +03:00
dialer := & net . Dialer {
Timeout : 30 * time . Second ,
KeepAlive : 30 * time . Second ,
DualStack : true ,
}
if globalConfiguration . ForwardingTimeouts != nil {
dialer . Timeout = time . Duration ( globalConfiguration . ForwardingTimeouts . DialTimeout )
}
transport := & http . Transport {
Proxy : http . ProxyFromEnvironment ,
DialContext : dialer . DialContext ,
MaxIdleConnsPerHost : globalConfiguration . MaxIdleConnsPerHost ,
IdleConnTimeout : 90 * time . Second ,
TLSHandshakeTimeout : 10 * time . Second ,
ExpectContinueTimeout : 1 * time . Second ,
}
if globalConfiguration . ForwardingTimeouts != nil {
transport . ResponseHeaderTimeout = time . Duration ( globalConfiguration . ForwardingTimeouts . ResponseHeaderTimeout )
}
if globalConfiguration . InsecureSkipVerify {
transport . TLSClientConfig = & tls . Config { InsecureSkipVerify : true }
}
if len ( globalConfiguration . RootCAs ) > 0 {
transport . TLSClientConfig = & tls . Config {
RootCAs : createRootCACertPool ( globalConfiguration . RootCAs ) ,
}
http2 . ConfigureTransport ( transport )
}
return transport
}
2017-08-25 17:10:03 +03:00
func createRootCACertPool ( rootCAs configuration . RootCAs ) * x509 . CertPool {
2017-08-18 16:34:04 +03:00
roots := x509 . NewCertPool ( )
for _ , cert := range rootCAs {
certContent , err := cert . Read ( )
if err != nil {
log . Error ( "Error while read RootCAs" , err )
continue
}
roots . AppendCertsFromPEM ( certContent )
}
return roots
}
2016-10-25 18:59:39 +03:00
// Start starts the server.
2016-01-14 00:45:49 +03:00
func ( server * Server ) Start ( ) {
2016-02-25 20:30:13 +03:00
server . startHTTPServers ( )
2016-08-18 15:20:11 +03:00
server . startLeadership ( )
2016-04-13 21:36:23 +03:00
server . routinesPool . Go ( func ( stop chan bool ) {
server . listenProviders ( stop )
2016-03-31 19:57:08 +03:00
} )
2016-04-13 21:36:23 +03:00
server . routinesPool . Go ( func ( stop chan bool ) {
server . listenConfigurations ( stop )
2016-03-31 19:57:08 +03:00
} )
2016-01-14 00:45:49 +03:00
server . configureProviders ( )
server . startProviders ( )
go server . listenSignals ( )
2016-10-25 18:59:39 +03:00
}
// Wait blocks until server is shutted down.
func ( server * Server ) Wait ( ) {
2016-01-14 00:45:49 +03:00
<- server . stopChan
}
// Stop stops the server
func ( server * Server ) Stop ( ) {
2017-03-10 01:27:09 +03:00
defer log . Info ( "Server stopped" )
var wg sync . WaitGroup
for sepn , sep := range server . serverEntryPoints {
wg . Add ( 1 )
go func ( serverEntryPointName string , serverEntryPoint * serverEntryPoint ) {
defer wg . Done ( )
2017-03-27 12:51:53 +03:00
graceTimeOut := time . Duration ( server . globalConfiguration . GraceTimeOut )
ctx , cancel := context . WithTimeout ( context . Background ( ) , graceTimeOut )
log . Debugf ( "Waiting %s seconds before killing connections on entrypoint %s..." , graceTimeOut , serverEntryPointName )
2017-03-10 01:27:09 +03:00
if err := serverEntryPoint . httpServer . Shutdown ( ctx ) ; err != nil {
log . Debugf ( "Wait is over due to: %s" , err )
serverEntryPoint . httpServer . Close ( )
}
2016-07-13 18:50:57 +03:00
cancel ( )
2017-03-10 01:27:09 +03:00
log . Debugf ( "Entrypoint %s closed" , serverEntryPointName )
} ( sepn , sep )
2016-01-29 22:34:17 +03:00
}
2017-03-10 01:27:09 +03:00
wg . Wait ( )
2016-01-14 00:45:49 +03:00
server . stopChan <- true
}
// Close destroys the server
func ( server * Server ) Close ( ) {
2017-03-27 12:51:53 +03:00
ctx , cancel := context . WithTimeout ( context . Background ( ) , time . Duration ( server . globalConfiguration . GraceTimeOut ) )
2016-07-13 18:50:57 +03:00
go func ( ctx context . Context ) {
<- ctx . Done ( )
if ctx . Err ( ) == context . Canceled {
return
} else if ctx . Err ( ) == context . DeadlineExceeded {
2017-05-26 18:03:14 +03:00
log . Warn ( "Timeout while stopping traefik, killing instance ✝" )
2016-07-13 18:50:57 +03:00
os . Exit ( 1 )
}
} ( ctx )
2017-08-23 21:46:03 +03:00
stopMetricsClients ( )
2016-08-18 14:03:10 +03:00
server . stopLeadership ( )
2016-11-15 22:14:11 +03:00
server . routinesPool . Cleanup ( )
2016-01-14 00:46:44 +03:00
close ( server . configurationChan )
2016-01-29 22:34:17 +03:00
close ( server . configurationValidatedChan )
2016-07-14 11:22:18 +03:00
signal . Stop ( server . signals )
2016-01-29 22:34:17 +03:00
close ( server . signals )
2016-01-14 00:46:44 +03:00
close ( server . stopChan )
2017-05-22 22:39:29 +03:00
if server . accessLoggerMiddleware != nil {
if err := server . accessLoggerMiddleware . Close ( ) ; err != nil {
log . Errorf ( "Error closing access log file: %s" , err )
}
}
2016-07-13 18:50:57 +03:00
cancel ( )
2016-01-14 00:45:49 +03:00
}
2016-08-18 14:03:10 +03:00
func ( server * Server ) startLeadership ( ) {
if server . leadership != nil {
server . leadership . Participate ( server . routinesPool )
}
}
func ( server * Server ) stopLeadership ( ) {
if server . leadership != nil {
2016-08-18 15:20:11 +03:00
server . leadership . Stop ( )
2016-08-18 14:03:10 +03:00
}
}
2016-02-25 20:30:13 +03:00
func ( server * Server ) startHTTPServers ( ) {
server . serverEntryPoints = server . buildEntryPoints ( server . globalConfiguration )
2017-01-12 16:34:54 +03:00
2016-02-25 20:30:13 +03:00
for newServerEntryPointName , newServerEntryPoint := range server . serverEntryPoints {
2017-07-08 13:21:14 +03:00
serverEntryPoint := server . setupServerEntryPoint ( newServerEntryPointName , newServerEntryPoint )
2017-08-25 22:32:03 +03:00
go server . startServer ( serverEntryPoint , server . globalConfiguration )
2017-07-08 13:21:14 +03:00
}
}
func ( server * Server ) setupServerEntryPoint ( newServerEntryPointName string , newServerEntryPoint * serverEntryPoint ) * serverEntryPoint {
2017-08-25 17:10:03 +03:00
serverMiddlewares := [ ] negroni . Handler { middlewares . NegroniRecoverHandler ( ) }
2017-07-08 13:21:14 +03:00
if server . accessLoggerMiddleware != nil {
serverMiddlewares = append ( serverMiddlewares , server . accessLoggerMiddleware )
}
2017-08-23 21:46:03 +03:00
if server . metricsRegistry . IsEnabled ( ) {
serverMiddlewares = append ( serverMiddlewares , middlewares . NewMetricsWrapper ( server . metricsRegistry , newServerEntryPointName ) )
2017-07-08 13:21:14 +03:00
}
2017-08-25 17:10:03 +03:00
if server . globalConfiguration . Web != nil {
server . globalConfiguration . Web . Stats = thoas_stats . New ( )
serverMiddlewares = append ( serverMiddlewares , server . globalConfiguration . Web . Stats )
if server . globalConfiguration . Web . Statistics != nil {
server . globalConfiguration . Web . StatsRecorder = middlewares . NewStatsRecorder ( server . globalConfiguration . Web . Statistics . RecentErrors )
serverMiddlewares = append ( serverMiddlewares , server . globalConfiguration . Web . StatsRecorder )
}
2017-07-08 13:21:14 +03:00
}
if server . globalConfiguration . EntryPoints [ newServerEntryPointName ] . Auth != nil {
2017-09-18 18:48:07 +03:00
authMiddleware , err := mauth . NewAuthenticator ( server . globalConfiguration . EntryPoints [ newServerEntryPointName ] . Auth )
2017-07-08 13:21:14 +03:00
if err != nil {
log . Fatal ( "Error starting server: " , err )
2016-09-29 00:07:06 +03:00
}
2017-07-08 13:21:14 +03:00
serverMiddlewares = append ( serverMiddlewares , authMiddleware )
}
if server . globalConfiguration . EntryPoints [ newServerEntryPointName ] . Compress {
serverMiddlewares = append ( serverMiddlewares , & middlewares . Compress { } )
}
if len ( server . globalConfiguration . EntryPoints [ newServerEntryPointName ] . WhitelistSourceRange ) > 0 {
ipWhitelistMiddleware , err := middlewares . NewIPWhitelister ( server . globalConfiguration . EntryPoints [ newServerEntryPointName ] . WhitelistSourceRange )
2016-02-25 20:30:13 +03:00
if err != nil {
2017-07-08 13:21:14 +03:00
log . Fatal ( "Error starting server: " , err )
2016-02-25 20:30:13 +03:00
}
2017-07-08 13:21:14 +03:00
serverMiddlewares = append ( serverMiddlewares , ipWhitelistMiddleware )
2016-02-25 20:30:13 +03:00
}
2017-08-25 22:32:03 +03:00
newSrv , listener , err := server . prepareServer ( newServerEntryPointName , server . globalConfiguration . EntryPoints [ newServerEntryPointName ] , newServerEntryPoint . httpRouter , serverMiddlewares ... )
2017-07-08 13:21:14 +03:00
if err != nil {
log . Fatal ( "Error preparing server: " , err )
}
serverEntryPoint := server . serverEntryPoints [ newServerEntryPointName ]
2017-08-25 17:10:03 +03:00
serverEntryPoint . httpServer = newSrv
2017-08-25 22:32:03 +03:00
serverEntryPoint . listener = listener
2017-07-08 13:21:14 +03:00
return serverEntryPoint
2016-02-25 20:30:13 +03:00
}
2016-04-13 21:36:23 +03:00
func ( server * Server ) listenProviders ( stop chan bool ) {
lastReceivedConfiguration := safe . New ( time . Unix ( 0 , 0 ) )
lastConfigs := cmap . New ( )
2016-01-14 00:45:49 +03:00
for {
2016-04-13 21:36:23 +03:00
select {
case <- stop :
return
case configMsg , ok := <- server . configurationChan :
if ! ok {
return
}
2016-06-22 19:31:14 +03:00
server . defaultConfigurationValues ( configMsg . Configuration )
2017-08-25 17:10:03 +03:00
currentConfigurations := server . currentConfigurations . Get ( ) . ( types . Configurations )
2016-04-13 21:36:23 +03:00
jsonConf , _ := json . Marshal ( configMsg . Configuration )
log . Debugf ( "Configuration received from provider %s: %s" , configMsg . ProviderName , string ( jsonConf ) )
2016-06-22 19:31:14 +03:00
if configMsg . Configuration == nil || configMsg . Configuration . Backends == nil && configMsg . Configuration . Frontends == nil {
log . Infof ( "Skipping empty Configuration for provider %s" , configMsg . ProviderName )
} else if reflect . DeepEqual ( currentConfigurations [ configMsg . ProviderName ] , configMsg . Configuration ) {
log . Infof ( "Skipping same configuration for provider %s" , configMsg . ProviderName )
2016-04-13 21:36:23 +03:00
} else {
2016-06-22 19:31:14 +03:00
lastConfigs . Set ( configMsg . ProviderName , & configMsg )
lastReceivedConfigurationValue := lastReceivedConfiguration . Get ( ) . ( time . Time )
2017-03-27 12:51:53 +03:00
providersThrottleDuration := time . Duration ( server . globalConfiguration . ProvidersThrottleDuration )
if time . Now ( ) . After ( lastReceivedConfigurationValue . Add ( providersThrottleDuration ) ) {
2016-10-21 17:02:18 +03:00
log . Debugf ( "Last %s config received more than %s, OK" , configMsg . ProviderName , server . globalConfiguration . ProvidersThrottleDuration . String ( ) )
2016-06-22 19:31:14 +03:00
// last config received more than n s ago
server . configurationValidatedChan <- configMsg
} else {
2016-10-21 17:02:18 +03:00
log . Debugf ( "Last %s config received less than %s, waiting..." , configMsg . ProviderName , server . globalConfiguration . ProvidersThrottleDuration . String ( ) )
2016-06-22 19:31:14 +03:00
safe . Go ( func ( ) {
2017-03-27 12:51:53 +03:00
<- time . After ( providersThrottleDuration )
2016-06-22 19:31:14 +03:00
lastReceivedConfigurationValue := lastReceivedConfiguration . Get ( ) . ( time . Time )
2017-03-27 12:51:53 +03:00
if time . Now ( ) . After ( lastReceivedConfigurationValue . Add ( time . Duration ( providersThrottleDuration ) ) ) {
2016-06-22 19:31:14 +03:00
log . Debugf ( "Waited for %s config, OK" , configMsg . ProviderName )
if lastConfig , ok := lastConfigs . Get ( configMsg . ProviderName ) ; ok {
server . configurationValidatedChan <- * lastConfig . ( * types . ConfigMessage )
}
2016-04-13 21:36:23 +03:00
}
2016-06-22 19:31:14 +03:00
} )
}
lastReceivedConfiguration . Set ( time . Now ( ) )
2016-04-13 21:36:23 +03:00
}
2016-06-22 19:31:14 +03:00
}
}
}
func ( server * Server ) defaultConfigurationValues ( configuration * types . Configuration ) {
if configuration == nil || configuration . Frontends == nil {
return
}
2017-05-11 01:34:47 +03:00
server . configureFrontends ( configuration . Frontends )
2017-05-16 00:53:35 +03:00
server . configureBackends ( configuration . Backends )
2016-01-14 00:45:49 +03:00
}
2016-04-13 21:36:23 +03:00
func ( server * Server ) listenConfigurations ( stop chan bool ) {
2016-01-14 00:45:49 +03:00
for {
2016-04-13 21:36:23 +03:00
select {
case <- stop :
return
case configMsg , ok := <- server . configurationValidatedChan :
if ! ok {
return
2016-01-14 00:45:49 +03:00
}
2017-08-25 17:10:03 +03:00
currentConfigurations := server . currentConfigurations . Get ( ) . ( types . Configurations )
2016-04-13 21:36:23 +03:00
2016-06-22 19:31:14 +03:00
// Copy configurations to new map so we don't change current if LoadConfig fails
2017-08-25 17:10:03 +03:00
newConfigurations := make ( types . Configurations )
2016-06-22 19:31:14 +03:00
for k , v := range currentConfigurations {
newConfigurations [ k ] = v
}
newConfigurations [ configMsg . ProviderName ] = configMsg . Configuration
newServerEntryPoints , err := server . loadConfig ( newConfigurations , server . globalConfiguration )
if err == nil {
for newServerEntryPointName , newServerEntryPoint := range newServerEntryPoints {
server . serverEntryPoints [ newServerEntryPointName ] . httpRouter . UpdateHandler ( newServerEntryPoint . httpRouter . GetHandler ( ) )
log . Infof ( "Server configuration reloaded on %s" , server . serverEntryPoints [ newServerEntryPointName ] . httpServer . Addr )
2016-04-13 21:36:23 +03:00
}
2016-06-22 19:31:14 +03:00
server . currentConfigurations . Set ( newConfigurations )
2016-08-05 21:42:45 +03:00
server . postLoadConfig ( )
2016-06-22 19:31:14 +03:00
} else {
log . Error ( "Error loading new configuration, aborted " , err )
2016-01-14 00:45:49 +03:00
}
}
}
}
2016-08-05 21:42:45 +03:00
func ( server * Server ) postLoadConfig ( ) {
2016-08-18 15:20:11 +03:00
if server . globalConfiguration . ACME == nil {
return
}
if server . leadership != nil && ! server . leadership . IsLeader ( ) {
return
}
if server . globalConfiguration . ACME . OnHostRule {
2017-08-25 17:10:03 +03:00
currentConfigurations := server . currentConfigurations . Get ( ) . ( types . Configurations )
for _ , config := range currentConfigurations {
for _ , frontend := range config . Frontends {
2017-01-05 14:32:56 +03:00
// check if one of the frontend entrypoints is configured with TLS
2017-04-07 16:48:58 +03:00
// and is configured with ACME
ACMEEnabled := false
2017-01-05 14:32:56 +03:00
for _ , entrypoint := range frontend . EntryPoints {
2017-04-07 16:48:58 +03:00
if server . globalConfiguration . ACME . EntryPoint == entrypoint && server . globalConfiguration . EntryPoints [ entrypoint ] . TLS != nil {
ACMEEnabled = true
2017-01-05 14:32:56 +03:00
break
2016-08-05 21:42:45 +03:00
}
}
2017-04-07 16:48:58 +03:00
if ACMEEnabled {
2017-01-05 14:32:56 +03:00
for _ , route := range frontend . Routes {
rules := Rules { }
domains , err := rules . ParseDomains ( route . Rule )
if err != nil {
log . Errorf ( "Error parsing domains: %v" , err )
} else {
server . globalConfiguration . ACME . LoadCertificateForDomains ( domains )
}
}
}
2016-08-05 21:42:45 +03:00
}
}
}
}
2016-01-14 00:45:49 +03:00
func ( server * Server ) configureProviders ( ) {
// configure providers
if server . globalConfiguration . Docker != nil {
server . providers = append ( server . providers , server . globalConfiguration . Docker )
}
if server . globalConfiguration . Marathon != nil {
server . providers = append ( server . providers , server . globalConfiguration . Marathon )
}
if server . globalConfiguration . File != nil {
server . providers = append ( server . providers , server . globalConfiguration . File )
}
if server . globalConfiguration . Web != nil {
2017-08-25 17:10:03 +03:00
server . globalConfiguration . Web . CurrentConfigurations = & server . currentConfigurations
server . globalConfiguration . Web . Debug = server . globalConfiguration . Debug
2016-01-14 00:45:49 +03:00
server . providers = append ( server . providers , server . globalConfiguration . Web )
}
if server . globalConfiguration . Consul != nil {
server . providers = append ( server . providers , server . globalConfiguration . Consul )
}
2016-02-02 20:03:40 +03:00
if server . globalConfiguration . ConsulCatalog != nil {
server . providers = append ( server . providers , server . globalConfiguration . ConsulCatalog )
}
2016-01-14 00:45:49 +03:00
if server . globalConfiguration . Etcd != nil {
server . providers = append ( server . providers , server . globalConfiguration . Etcd )
}
if server . globalConfiguration . Zookeeper != nil {
server . providers = append ( server . providers , server . globalConfiguration . Zookeeper )
}
if server . globalConfiguration . Boltdb != nil {
server . providers = append ( server . providers , server . globalConfiguration . Boltdb )
}
2016-02-08 23:57:32 +03:00
if server . globalConfiguration . Kubernetes != nil {
server . providers = append ( server . providers , server . globalConfiguration . Kubernetes )
}
2016-07-20 12:56:14 +03:00
if server . globalConfiguration . Mesos != nil {
server . providers = append ( server . providers , server . globalConfiguration . Mesos )
}
2016-08-31 23:43:05 +03:00
if server . globalConfiguration . Eureka != nil {
server . providers = append ( server . providers , server . globalConfiguration . Eureka )
}
2017-01-05 17:24:17 +03:00
if server . globalConfiguration . ECS != nil {
server . providers = append ( server . providers , server . globalConfiguration . ECS )
}
2017-01-29 02:01:56 +03:00
if server . globalConfiguration . Rancher != nil {
server . providers = append ( server . providers , server . globalConfiguration . Rancher )
2017-03-09 04:53:34 +03:00
}
if server . globalConfiguration . DynamoDB != nil {
server . providers = append ( server . providers , server . globalConfiguration . DynamoDB )
2017-01-29 02:01:56 +03:00
}
2016-01-14 00:45:49 +03:00
}
func ( server * Server ) startProviders ( ) {
// start providers
2017-08-25 17:10:03 +03:00
for _ , p := range server . providers {
providerType := reflect . TypeOf ( p )
jsonConf , _ := json . Marshal ( p )
2017-03-07 15:09:11 +03:00
log . Infof ( "Starting provider %v %s" , providerType , jsonConf )
2017-08-25 17:10:03 +03:00
currentProvider := p
2016-03-31 19:57:08 +03:00
safe . Go ( func ( ) {
2016-08-18 14:03:10 +03:00
err := currentProvider . Provide ( server . configurationChan , server . routinesPool , server . globalConfiguration . Constraints )
2016-01-14 00:45:49 +03:00
if err != nil {
2017-03-07 15:09:11 +03:00
log . Errorf ( "Error starting provider %v: %s" , providerType , err )
2016-01-14 00:45:49 +03:00
}
2016-03-31 19:57:08 +03:00
} )
2016-01-14 00:45:49 +03:00
}
}
2017-08-25 17:10:03 +03:00
func createClientTLSConfig ( tlsOption * configuration . TLS ) ( * tls . Config , error ) {
2017-04-07 01:10:02 +03:00
if tlsOption == nil {
return nil , errors . New ( "no TLS provided" )
}
config , err := tlsOption . Certificates . CreateTLSConfig ( )
if err != nil {
return nil , err
}
if len ( tlsOption . ClientCAFiles ) > 0 {
pool := x509 . NewCertPool ( )
for _ , caFile := range tlsOption . ClientCAFiles {
data , err := ioutil . ReadFile ( caFile )
if err != nil {
return nil , err
}
if ! pool . AppendCertsFromPEM ( data ) {
return nil , errors . New ( "invalid certificate(s) in " + caFile )
}
}
config . RootCAs = pool
}
config . BuildNameToCertificate ( )
return config , nil
}
2016-01-14 00:45:49 +03:00
// creates a TLS config that allows terminating HTTPS for multiple domains using SNI
2017-08-25 17:10:03 +03:00
func ( server * Server ) createTLSConfig ( entryPointName string , tlsOption * configuration . TLS , router * middlewares . HandlerSwitcher ) ( * tls . Config , error ) {
2016-01-29 22:34:17 +03:00
if tlsOption == nil {
return nil , nil
}
2016-01-14 00:45:49 +03:00
2016-06-27 13:19:14 +03:00
config , err := tlsOption . Certificates . CreateTLSConfig ( )
if err != nil {
return nil , err
2016-03-21 13:10:18 +03:00
}
2016-11-09 19:56:41 +03:00
// ensure http2 enabled
config . NextProtos = [ ] string { "h2" , "http/1.1" }
2016-06-15 23:38:40 +03:00
if len ( tlsOption . ClientCAFiles ) > 0 {
pool := x509 . NewCertPool ( )
for _ , caFile := range tlsOption . ClientCAFiles {
data , err := ioutil . ReadFile ( caFile )
if err != nil {
return nil , err
}
ok := pool . AppendCertsFromPEM ( data )
if ! ok {
return nil , errors . New ( "invalid certificate(s) in " + caFile )
}
}
config . ClientCAs = pool
config . ClientAuth = tls . RequireAndVerifyClientCert
}
2016-03-21 13:10:18 +03:00
if server . globalConfiguration . ACME != nil {
if _ , ok := server . serverEntryPoints [ server . globalConfiguration . ACME . EntryPoint ] ; ok {
if entryPointName == server . globalConfiguration . ACME . EntryPoint {
checkOnDemandDomain := func ( domain string ) bool {
2016-07-11 17:43:23 +03:00
routeMatch := & mux . RouteMatch { }
router := router . GetHandler ( )
match := router . Match ( & http . Request { URL : & url . URL { } , Host : domain } , routeMatch )
if match && routeMatch . Route != nil {
2016-03-21 13:10:18 +03:00
return true
}
return false
}
2016-08-18 15:20:11 +03:00
if server . leadership == nil {
err := server . globalConfiguration . ACME . CreateLocalConfig ( config , checkOnDemandDomain )
if err != nil {
return nil , err
}
} else {
err := server . globalConfiguration . ACME . CreateClusterConfig ( server . leadership , config , checkOnDemandDomain )
if err != nil {
return nil , err
}
2016-03-21 13:10:18 +03:00
}
}
} else {
return nil , errors . New ( "Unknown entrypoint " + server . globalConfiguration . ACME . EntryPoint + " for ACME configuration" )
}
}
if len ( config . Certificates ) == 0 {
return nil , errors . New ( "No certificates found for TLS entrypoint " + entryPointName )
2016-01-14 00:45:49 +03:00
}
// BuildNameToCertificate parses the CommonName and SubjectAlternateName fields
// in each certificate and populates the config.NameToCertificate map.
config . BuildNameToCertificate ( )
2016-09-20 09:06:06 +03:00
//Set the minimum TLS version if set in the config TOML
2017-08-25 17:10:03 +03:00
if minConst , exists := configuration . MinVersion [ server . globalConfiguration . EntryPoints [ entryPointName ] . TLS . MinVersion ] ; exists {
2016-09-20 09:06:06 +03:00
config . PreferServerCipherSuites = true
config . MinVersion = minConst
}
//Set the list of CipherSuites if set in the config TOML
if server . globalConfiguration . EntryPoints [ entryPointName ] . TLS . CipherSuites != nil {
//if our list of CipherSuites is defined in the entrypoint config, we can re-initilize the suites list as empty
config . CipherSuites = make ( [ ] uint16 , 0 )
for _ , cipher := range server . globalConfiguration . EntryPoints [ entryPointName ] . TLS . CipherSuites {
2017-08-25 17:10:03 +03:00
if cipherConst , exists := configuration . CipherSuites [ cipher ] ; exists {
2016-09-20 09:06:06 +03:00
config . CipherSuites = append ( config . CipherSuites , cipherConst )
} else {
//CipherSuite listed in the toml does not exist in our listed
return nil , errors . New ( "Invalid CipherSuite: " + cipher )
}
}
}
2017-04-07 01:10:02 +03:00
2016-01-14 00:45:49 +03:00
return config , nil
}
2017-08-25 22:32:03 +03:00
func ( server * Server ) startServer ( serverEntryPoint * serverEntryPoint , globalConfiguration configuration . GlobalConfiguration ) {
log . Infof ( "Starting server on %s" , serverEntryPoint . httpServer . Addr )
2017-03-10 01:27:09 +03:00
var err error
2017-08-25 22:32:03 +03:00
if serverEntryPoint . httpServer . TLSConfig != nil {
err = serverEntryPoint . httpServer . ServeTLS ( serverEntryPoint . listener , "" , "" )
2016-01-14 00:45:49 +03:00
} else {
2017-08-25 22:32:03 +03:00
err = serverEntryPoint . httpServer . Serve ( serverEntryPoint . listener )
2017-03-10 01:27:09 +03:00
}
if err != nil {
log . Error ( "Error creating server: " , err )
2016-01-14 00:45:49 +03:00
}
}
2017-08-25 22:32:03 +03:00
func ( server * Server ) prepareServer ( entryPointName string , entryPoint * configuration . EntryPoint , router * middlewares . HandlerSwitcher , middlewares ... negroni . Handler ) ( * http . Server , net . Listener , error ) {
2017-08-18 16:34:04 +03:00
readTimeout , writeTimeout , idleTimeout := buildServerTimeouts ( server . globalConfiguration )
log . Infof ( "Preparing server %s %+v with readTimeout=%s writeTimeout=%s idleTimeout=%s" , entryPointName , entryPoint , readTimeout , writeTimeout , idleTimeout )
2016-01-14 00:45:49 +03:00
// middlewares
2017-08-18 16:34:04 +03:00
n := negroni . New ( )
2016-01-14 00:45:49 +03:00
for _ , middleware := range middlewares {
2017-08-18 16:34:04 +03:00
n . Use ( middleware )
2016-01-14 00:45:49 +03:00
}
2017-08-18 16:34:04 +03:00
n . UseHandler ( router )
2016-03-21 13:10:18 +03:00
tlsConfig , err := server . createTLSConfig ( entryPointName , entryPoint . TLS , router )
2016-01-14 00:45:49 +03:00
if err != nil {
2017-01-12 13:04:11 +03:00
log . Errorf ( "Error creating TLS config: %s" , err )
2017-08-25 22:32:03 +03:00
return nil , nil , err
}
listener , err := net . Listen ( "tcp" , entryPoint . Address )
if err != nil {
log . Error ( "Error opening listener " , err )
2017-09-07 21:14:03 +03:00
return nil , nil , err
2017-08-25 22:32:03 +03:00
}
if entryPoint . ProxyProtocol {
listener = & proxyproto . Listener { Listener : listener }
2016-01-14 00:45:49 +03:00
}
2017-03-10 01:27:09 +03:00
return & http . Server {
2017-08-25 22:32:03 +03:00
Addr : entryPoint . Address ,
Handler : n ,
TLSConfig : tlsConfig ,
ReadTimeout : readTimeout ,
WriteTimeout : writeTimeout ,
IdleTimeout : idleTimeout ,
} ,
listener ,
nil
2016-01-14 00:45:49 +03:00
}
2017-08-25 17:10:03 +03:00
func buildServerTimeouts ( globalConfig configuration . GlobalConfiguration ) ( readTimeout , writeTimeout , idleTimeout time . Duration ) {
2017-08-18 16:34:04 +03:00
readTimeout = time . Duration ( 0 )
writeTimeout = time . Duration ( 0 )
if globalConfig . RespondingTimeouts != nil {
readTimeout = time . Duration ( globalConfig . RespondingTimeouts . ReadTimeout )
writeTimeout = time . Duration ( globalConfig . RespondingTimeouts . WriteTimeout )
}
// When RespondingTimeouts.IdleTimout is configured, always use that setting
if globalConfig . RespondingTimeouts != nil {
idleTimeout = time . Duration ( globalConfig . RespondingTimeouts . IdleTimeout )
} else if globalConfig . IdleTimeout != 0 {
// Backwards compatibility for deprecated IdleTimeout
idleTimeout = time . Duration ( globalConfig . IdleTimeout )
} else {
// Default value if neither the deprecated IdleTimeout nor the new RespondingTimeouts.IdleTimout are configured
2017-08-25 17:10:03 +03:00
idleTimeout = time . Duration ( configuration . DefaultIdleTimeout )
2017-08-18 16:34:04 +03:00
}
return readTimeout , writeTimeout , idleTimeout
}
2017-08-25 17:10:03 +03:00
func ( server * Server ) buildEntryPoints ( globalConfiguration configuration . GlobalConfiguration ) map [ string ] * serverEntryPoint {
2016-02-25 20:30:13 +03:00
serverEntryPoints := make ( map [ string ] * serverEntryPoint )
2016-01-29 22:34:17 +03:00
for entryPointName := range globalConfiguration . EntryPoints {
router := server . buildDefaultHTTPRouter ( )
2016-02-25 20:30:13 +03:00
serverEntryPoints [ entryPointName ] = & serverEntryPoint {
2016-03-04 13:32:23 +03:00
httpRouter : middlewares . NewHandlerSwitcher ( router ) ,
2016-01-29 22:34:17 +03:00
}
}
return serverEntryPoints
}
2017-08-18 16:34:04 +03:00
// getRoundTripper will either use server.defaultForwardingRoundTripper or create a new one
// given a custom TLS configuration is passed and the passTLSCert option is set to true.
2017-08-25 17:10:03 +03:00
func ( server * Server ) getRoundTripper ( globalConfiguration configuration . GlobalConfiguration , passTLSCert bool , tls * configuration . TLS ) ( http . RoundTripper , error ) {
2017-08-18 16:34:04 +03:00
if passTLSCert {
tlsConfig , err := createClientTLSConfig ( tls )
if err != nil {
log . Errorf ( "Failed to create TLSClientConfig: %s" , err )
return nil , err
}
transport := createHTTPTransport ( globalConfiguration )
transport . TLSClientConfig = tlsConfig
return transport , nil
2017-04-07 01:10:02 +03:00
}
2017-08-18 16:34:04 +03:00
return server . defaultForwardingRoundTripper , nil
2017-04-07 01:10:02 +03:00
}
2016-01-14 00:45:49 +03:00
// LoadConfig returns a new gorilla.mux Route from the specified global configuration and the dynamic
// provider configurations.
2017-08-25 17:10:03 +03:00
func ( server * Server ) loadConfig ( configurations types . Configurations , globalConfiguration configuration . GlobalConfiguration ) ( map [ string ] * serverEntryPoint , error ) {
2016-01-29 22:34:17 +03:00
serverEntryPoints := server . buildEntryPoints ( globalConfiguration )
2017-03-10 21:43:20 +03:00
redirectHandlers := make ( map [ string ] negroni . Handler )
2016-01-14 00:45:49 +03:00
backends := map [ string ] http . Handler { }
2017-08-25 17:10:03 +03:00
backendsHealthCheck := map [ string ] * healthcheck . BackendHealthCheck { }
2017-05-03 11:20:33 +03:00
errorHandler := NewRecordingErrorHandler ( middlewares . DefaultNetErrorRecorder { } )
2017-04-13 17:37:07 +03:00
2017-08-25 17:10:03 +03:00
for _ , config := range configurations {
frontendNames := sortedFrontendNamesForConfig ( config )
2016-06-20 16:19:52 +03:00
frontend :
2016-02-20 01:55:23 +03:00
for _ , frontendName := range frontendNames {
2017-08-25 17:10:03 +03:00
frontend := config . Frontends [ frontendName ]
2016-02-20 01:55:23 +03:00
2016-01-29 22:34:17 +03:00
log . Debugf ( "Creating frontend %s" , frontendName )
2016-07-04 20:30:32 +03:00
2016-03-22 03:32:02 +03:00
if len ( frontend . EntryPoints ) == 0 {
2016-06-20 16:19:52 +03:00
log . Errorf ( "No entrypoint defined for frontend %s, defaultEntryPoints:%s" , frontendName , globalConfiguration . DefaultEntryPoints )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2016-03-22 03:32:02 +03:00
}
2017-04-19 12:14:05 +03:00
2016-01-29 22:34:17 +03:00
for _ , entryPointName := range frontend . EntryPoints {
log . Debugf ( "Wiring frontend %s to entryPoint %s" , frontendName , entryPointName )
if _ , ok := serverEntryPoints [ entryPointName ] ; ! ok {
2016-06-20 16:19:52 +03:00
log . Errorf ( "Undefined entrypoint '%s' for frontend %s" , entryPointName , frontendName )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2016-01-14 00:45:49 +03:00
}
2017-04-19 12:14:05 +03:00
2016-03-27 03:05:17 +03:00
newServerRoute := & serverRoute { route : serverEntryPoints [ entryPointName ] . httpRouter . GetHandler ( ) . NewRoute ( ) . Name ( frontendName ) }
2016-01-29 22:34:17 +03:00
for routeName , route := range frontend . Routes {
2016-03-30 20:05:43 +03:00
err := getRoute ( newServerRoute , & route )
2016-01-29 22:34:17 +03:00
if err != nil {
2016-06-20 16:19:52 +03:00
log . Errorf ( "Error creating route for frontend %s: %v" , frontendName , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2016-01-29 22:34:17 +03:00
}
2016-03-30 20:05:43 +03:00
log . Debugf ( "Creating route %s %s" , routeName , route . Rule )
2016-01-14 00:45:49 +03:00
}
2017-04-19 12:14:05 +03:00
2016-01-29 22:34:17 +03:00
entryPoint := globalConfiguration . EntryPoints [ entryPointName ]
2017-08-25 17:10:03 +03:00
n := negroni . New ( )
2016-01-29 22:34:17 +03:00
if entryPoint . Redirect != nil {
if redirectHandlers [ entryPointName ] != nil {
2017-08-25 17:10:03 +03:00
n . Use ( redirectHandlers [ entryPointName ] )
2016-01-29 22:34:17 +03:00
} else if handler , err := server . loadEntryPointConfig ( entryPointName , entryPoint ) ; err != nil {
2016-06-20 16:19:52 +03:00
log . Errorf ( "Error loading entrypoint configuration for frontend %s: %v" , frontendName , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2016-01-29 22:34:17 +03:00
} else {
2017-05-22 22:39:29 +03:00
if server . accessLoggerMiddleware != nil {
saveFrontend := accesslog . NewSaveNegroniFrontend ( handler , frontendName )
2017-08-25 17:10:03 +03:00
n . Use ( saveFrontend )
2017-05-22 22:39:29 +03:00
redirectHandlers [ entryPointName ] = saveFrontend
} else {
2017-08-25 17:10:03 +03:00
n . Use ( handler )
2017-05-22 22:39:29 +03:00
redirectHandlers [ entryPointName ] = handler
}
2016-01-14 00:45:49 +03:00
}
2017-03-10 21:43:20 +03:00
}
2017-04-13 17:37:07 +03:00
if backends [ entryPointName + frontend . Backend ] == nil {
2017-03-10 21:43:20 +03:00
log . Debugf ( "Creating backend %s" , frontend . Backend )
2017-04-07 01:10:02 +03:00
2017-08-18 16:34:04 +03:00
roundTripper , err := server . getRoundTripper ( globalConfiguration , frontend . PassTLSCert , entryPoint . TLS )
if err != nil {
log . Errorf ( "Failed to create RoundTripper for frontend %s: %v" , frontendName , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2017-04-07 01:10:02 +03:00
}
2017-05-03 11:20:33 +03:00
fwd , err := forward . New (
forward . Logger ( oxyLogger ) ,
forward . PassHostHeader ( frontend . PassHostHeader ) ,
2017-08-18 16:34:04 +03:00
forward . RoundTripper ( roundTripper ) ,
2017-05-03 11:20:33 +03:00
forward . ErrorHandler ( errorHandler ) ,
)
2017-08-18 16:34:04 +03:00
2017-04-07 01:10:02 +03:00
if err != nil {
log . Errorf ( "Error creating forwarder for frontend %s: %v" , frontendName , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
2017-05-22 22:39:29 +03:00
var rr * roundrobin . RoundRobin
var saveFrontend http . Handler
if server . accessLoggerMiddleware != nil {
saveBackend := accesslog . NewSaveBackend ( fwd , frontend . Backend )
saveFrontend = accesslog . NewSaveFrontend ( saveBackend , frontendName )
rr , _ = roundrobin . New ( saveFrontend )
} else {
rr , _ = roundrobin . New ( fwd )
}
2017-04-07 01:10:02 +03:00
2017-08-25 17:10:03 +03:00
if config . Backends [ frontend . Backend ] == nil {
2017-03-10 21:43:20 +03:00
log . Errorf ( "Undefined backend '%s' for frontend %s" , frontend . Backend , frontendName )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
2016-05-13 17:22:11 +03:00
2017-08-25 17:10:03 +03:00
lbMethod , err := types . NewLoadBalancerMethod ( config . Backends [ frontend . Backend ] . LoadBalancer )
2017-03-10 21:43:20 +03:00
if err != nil {
2017-08-25 17:10:03 +03:00
log . Errorf ( "Error loading load balancer method '%+v' for frontend %s: %v" , config . Backends [ frontend . Backend ] . LoadBalancer , frontendName , err )
2017-03-10 21:43:20 +03:00
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
2016-05-13 17:22:11 +03:00
2017-08-25 17:10:03 +03:00
stickySession := config . Backends [ frontend . Backend ] . LoadBalancer . Sticky
cookieName := "_TRAEFIK_BACKEND_" + frontend . Backend
2017-03-10 21:43:20 +03:00
var sticky * roundrobin . StickySession
2016-05-26 20:38:55 +03:00
2017-08-25 17:10:03 +03:00
if stickySession {
sticky = roundrobin . NewStickySession ( cookieName )
2017-03-10 21:43:20 +03:00
}
2016-05-13 17:22:11 +03:00
2017-08-18 16:34:04 +03:00
var lb http . Handler
2017-04-13 17:37:07 +03:00
switch lbMethod {
case types . Drr :
log . Debugf ( "Creating load-balancer drr" )
rebalancer , _ := roundrobin . NewRebalancer ( rr , roundrobin . RebalancerLogger ( oxyLogger ) )
2017-08-25 17:10:03 +03:00
if stickySession {
log . Debugf ( "Sticky session with cookie %v" , cookieName )
2017-04-13 17:37:07 +03:00
rebalancer , _ = roundrobin . NewRebalancer ( rr , roundrobin . RebalancerLogger ( oxyLogger ) , roundrobin . RebalancerStickySession ( sticky ) )
}
lb = rebalancer
2017-08-25 17:10:03 +03:00
if err := configureLBServers ( rebalancer , config , frontend ) ; err != nil {
2017-07-08 11:33:17 +03:00
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
2017-08-25 17:10:03 +03:00
hcOpts := parseHealthCheckOptions ( rebalancer , frontend . Backend , config . Backends [ frontend . Backend ] . HealthCheck , globalConfiguration . HealthCheck )
2017-07-08 11:33:17 +03:00
if hcOpts != nil {
log . Debugf ( "Setting up backend health check %s" , * hcOpts )
2017-08-25 17:10:03 +03:00
backendsHealthCheck [ entryPointName + frontend . Backend ] = healthcheck . NewBackendHealthCheck ( * hcOpts )
2016-01-29 22:34:17 +03:00
}
2017-07-10 13:11:44 +03:00
lb = middlewares . NewEmptyBackendHandler ( rebalancer , lb )
2017-04-13 17:37:07 +03:00
case types . Wrr :
log . Debugf ( "Creating load-balancer wrr" )
2017-08-25 17:10:03 +03:00
if stickySession {
log . Debugf ( "Sticky session with cookie %v" , cookieName )
2017-05-22 22:39:29 +03:00
if server . accessLoggerMiddleware != nil {
rr , _ = roundrobin . New ( saveFrontend , roundrobin . EnableStickySession ( sticky ) )
} else {
rr , _ = roundrobin . New ( fwd , roundrobin . EnableStickySession ( sticky ) )
}
2017-04-13 17:37:07 +03:00
}
lb = rr
2017-08-25 17:10:03 +03:00
if err := configureLBServers ( rr , config , frontend ) ; err != nil {
2017-07-08 11:33:17 +03:00
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
2016-04-13 11:11:36 +03:00
}
2017-08-25 17:10:03 +03:00
hcOpts := parseHealthCheckOptions ( rr , frontend . Backend , config . Backends [ frontend . Backend ] . HealthCheck , globalConfiguration . HealthCheck )
2017-04-13 17:37:07 +03:00
if hcOpts != nil {
log . Debugf ( "Setting up backend health check %s" , * hcOpts )
2017-08-25 17:10:03 +03:00
backendsHealthCheck [ entryPointName + frontend . Backend ] = healthcheck . NewBackendHealthCheck ( * hcOpts )
2016-03-29 23:25:32 +03:00
}
2017-07-10 13:11:44 +03:00
lb = middlewares . NewEmptyBackendHandler ( rr , lb )
2017-04-13 17:37:07 +03:00
}
2017-07-01 02:04:18 +03:00
if len ( frontend . Errors ) > 0 {
for _ , errorPage := range frontend . Errors {
2017-08-25 17:10:03 +03:00
if config . Backends [ errorPage . Backend ] != nil && config . Backends [ errorPage . Backend ] . Servers [ "error" ] . URL != "" {
errorPageHandler , err := middlewares . NewErrorPagesHandler ( errorPage , config . Backends [ errorPage . Backend ] . Servers [ "error" ] . URL )
2017-07-01 02:04:18 +03:00
if err != nil {
log . Errorf ( "Error creating custom error page middleware, %v" , err )
} else {
2017-08-25 17:10:03 +03:00
n . Use ( errorPageHandler )
2017-07-01 02:04:18 +03:00
}
} else {
log . Errorf ( "Error Page is configured for Frontend %s, but either Backend %s is not set or Backend URL is missing" , frontendName , errorPage . Backend )
}
}
}
2017-08-25 17:10:03 +03:00
maxConns := config . Backends [ frontend . Backend ] . MaxConn
2017-04-13 17:37:07 +03:00
if maxConns != nil && maxConns . Amount != 0 {
extractFunc , err := utils . NewExtractor ( maxConns . ExtractorFunc )
if err != nil {
log . Errorf ( "Error creating connlimit: %v" , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
log . Debugf ( "Creating load-balancer connlimit" )
lb , err = connlimit . New ( lb , extractFunc , maxConns . Amount , connlimit . Logger ( oxyLogger ) )
if err != nil {
log . Errorf ( "Error creating connlimit: %v" , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
}
2017-04-18 09:22:06 +03:00
2017-04-13 17:37:07 +03:00
if globalConfiguration . Retry != nil {
2017-08-28 13:50:02 +03:00
countServers := len ( config . Backends [ frontend . Backend ] . Servers )
lb = server . buildRetryMiddleware ( lb , globalConfiguration , countServers , frontend . Backend )
2017-04-13 17:37:07 +03:00
}
2017-08-28 13:50:02 +03:00
2017-08-23 21:46:03 +03:00
if server . metricsRegistry . IsEnabled ( ) {
2017-08-25 17:10:03 +03:00
n . Use ( middlewares . NewMetricsWrapper ( server . metricsRegistry , frontend . Backend ) )
2017-04-13 17:37:07 +03:00
}
2017-05-26 18:03:14 +03:00
2017-04-30 12:22:07 +03:00
ipWhitelistMiddleware , err := configureIPWhitelistMiddleware ( frontend . WhitelistSourceRange )
if err != nil {
log . Fatalf ( "Error creating IP Whitelister: %s" , err )
} else if ipWhitelistMiddleware != nil {
2017-08-25 17:10:03 +03:00
n . Use ( ipWhitelistMiddleware )
2017-04-30 12:22:07 +03:00
log . Infof ( "Configured IP Whitelists: %s" , frontend . WhitelistSourceRange )
}
2017-04-13 17:37:07 +03:00
if len ( frontend . BasicAuth ) > 0 {
users := types . Users { }
for _ , user := range frontend . BasicAuth {
users = append ( users , user )
2017-01-12 16:34:54 +03:00
}
2017-04-19 12:14:05 +03:00
2017-04-13 17:37:07 +03:00
auth := & types . Auth { }
auth . Basic = & types . Basic {
Users : users ,
2016-01-14 00:45:49 +03:00
}
2017-09-18 18:48:07 +03:00
authMiddleware , err := mauth . NewAuthenticator ( auth )
2017-04-13 17:37:07 +03:00
if err != nil {
2017-05-15 10:02:32 +03:00
log . Errorf ( "Error creating Auth: %s" , err )
} else {
2017-08-25 17:10:03 +03:00
n . Use ( authMiddleware )
2017-04-13 17:37:07 +03:00
}
2016-01-14 00:45:49 +03:00
}
2017-05-26 18:03:14 +03:00
2017-06-13 03:48:21 +03:00
if frontend . Headers . HasCustomHeadersDefined ( ) {
headerMiddleware := middlewares . NewHeaderFromStruct ( frontend . Headers )
log . Debugf ( "Adding header middleware for frontend %s" , frontendName )
2017-08-25 17:10:03 +03:00
n . Use ( headerMiddleware )
2017-06-13 03:48:21 +03:00
}
if frontend . Headers . HasSecureHeadersDefined ( ) {
secureMiddleware := middlewares . NewSecure ( frontend . Headers )
log . Debugf ( "Adding secure middleware for frontend %s" , frontendName )
2017-08-25 17:10:03 +03:00
n . UseFunc ( secureMiddleware . HandlerFuncWithNext )
2017-06-13 03:48:21 +03:00
}
2017-08-25 17:10:03 +03:00
if config . Backends [ frontend . Backend ] . CircuitBreaker != nil {
log . Debugf ( "Creating circuit breaker %s" , config . Backends [ frontend . Backend ] . CircuitBreaker . Expression )
circuitBreaker , err := middlewares . NewCircuitBreaker ( lb , config . Backends [ frontend . Backend ] . CircuitBreaker . Expression , cbreaker . Logger ( oxyLogger ) )
2017-04-13 17:37:07 +03:00
if err != nil {
log . Errorf ( "Error creating circuit breaker: %v" , err )
log . Errorf ( "Skipping frontend %s..." , frontendName )
continue frontend
}
2017-08-25 17:10:03 +03:00
n . Use ( circuitBreaker )
2017-04-13 17:37:07 +03:00
} else {
2017-08-25 17:10:03 +03:00
n . UseHandler ( lb )
2016-06-03 18:58:33 +03:00
}
2017-08-25 17:10:03 +03:00
backends [ entryPointName + frontend . Backend ] = n
2017-04-13 17:37:07 +03:00
} else {
log . Debugf ( "Reusing backend %s" , frontend . Backend )
}
if frontend . Priority > 0 {
newServerRoute . route . Priority ( frontend . Priority )
}
server . wireFrontendBackend ( newServerRoute , backends [ entryPointName + frontend . Backend ] )
2017-03-10 21:43:20 +03:00
2016-03-27 03:05:17 +03:00
err := newServerRoute . route . GetError ( )
2016-01-29 22:34:17 +03:00
if err != nil {
log . Errorf ( "Error building route: %s" , err )
2016-01-14 00:45:49 +03:00
}
}
2016-01-29 22:34:17 +03:00
}
}
2017-08-25 17:10:03 +03:00
healthcheck . GetHealthCheck ( ) . SetBackendsConfiguration ( server . routinesPool . Ctx ( ) , backendsHealthCheck )
2016-06-03 18:58:33 +03:00
//sort routes
for _ , serverEntryPoint := range serverEntryPoints {
serverEntryPoint . httpRouter . GetHandler ( ) . SortRoutes ( )
}
2016-01-29 22:34:17 +03:00
return serverEntryPoints , nil
}
2016-01-14 00:45:49 +03:00
2017-07-08 11:33:17 +03:00
func configureLBServers ( lb healthcheck . LoadBalancer , config * types . Configuration , frontend * types . Frontend ) error {
for serverName , server := range config . Backends [ frontend . Backend ] . Servers {
u , err := url . Parse ( server . URL )
if err != nil {
log . Errorf ( "Error parsing server URL %s: %v" , server . URL , err )
return err
}
log . Debugf ( "Creating server %s at %s with weight %d" , serverName , u , server . Weight )
if err := lb . UpsertServer ( u , roundrobin . Weight ( server . Weight ) ) ; err != nil {
log . Errorf ( "Error adding server %s to load balancer: %v" , server . URL , err )
return err
}
}
return nil
}
2017-04-30 12:22:07 +03:00
func configureIPWhitelistMiddleware ( whitelistSourceRanges [ ] string ) ( negroni . Handler , error ) {
if len ( whitelistSourceRanges ) > 0 {
ipSourceRanges := whitelistSourceRanges
ipWhitelistMiddleware , err := middlewares . NewIPWhitelister ( ipSourceRanges )
if err != nil {
return nil , err
}
return ipWhitelistMiddleware , nil
}
return nil , nil
}
2016-03-27 03:05:17 +03:00
func ( server * Server ) wireFrontendBackend ( serverRoute * serverRoute , handler http . Handler ) {
2017-05-09 23:31:16 +03:00
// path replace - This needs to always be the very last on the handler chain (first in the order in this function)
// -- Replacing Path should happen at the very end of the Modifier chain, after all the Matcher+Modifiers ran
if len ( serverRoute . replacePath ) > 0 {
handler = & middlewares . ReplacePath {
Path : serverRoute . replacePath ,
Handler : handler ,
}
}
// add prefix - This needs to always be right before ReplacePath on the chain (second in order in this function)
// -- Adding Path Prefix should happen after all *Strip Matcher+Modifiers ran, but before Replace (in case it's configured)
2016-12-02 15:40:18 +03:00
if len ( serverRoute . addPrefix ) > 0 {
handler = & middlewares . AddPrefix {
Prefix : serverRoute . addPrefix ,
Handler : handler ,
}
}
2016-02-26 17:29:53 +03:00
// strip prefix
2016-04-06 14:06:31 +03:00
if len ( serverRoute . stripPrefixes ) > 0 {
2016-12-02 15:40:18 +03:00
handler = & middlewares . StripPrefix {
2016-04-06 14:06:31 +03:00
Prefixes : serverRoute . stripPrefixes ,
Handler : handler ,
2016-12-02 15:40:18 +03:00
}
2017-04-25 21:13:39 +03:00
}
2017-03-24 14:07:59 +03:00
// strip prefix with regex
if len ( serverRoute . stripPrefixesRegex ) > 0 {
handler = middlewares . NewStripPrefixRegex ( handler , serverRoute . stripPrefixesRegex )
}
2016-12-02 15:40:18 +03:00
serverRoute . route . Handler ( handler )
2016-02-26 17:29:53 +03:00
}
2017-08-25 17:10:03 +03:00
func ( server * Server ) loadEntryPointConfig ( entryPointName string , entryPoint * configuration . EntryPoint ) ( negroni . Handler , error ) {
2016-01-29 22:34:17 +03:00
regex := entryPoint . Redirect . Regex
replacement := entryPoint . Redirect . Replacement
if len ( entryPoint . Redirect . EntryPoint ) > 0 {
2017-08-18 03:18:02 +03:00
regex = ` ^(?:https?:\/\/)?([\w\._-]+)(?::\d+)?(.*)$ `
2016-01-29 22:34:17 +03:00
if server . globalConfiguration . EntryPoints [ entryPoint . Redirect . EntryPoint ] == nil {
2016-03-03 23:29:52 +03:00
return nil , errors . New ( "Unknown entrypoint " + entryPoint . Redirect . EntryPoint )
2016-01-29 22:34:17 +03:00
}
protocol := "http"
if server . globalConfiguration . EntryPoints [ entryPoint . Redirect . EntryPoint ] . TLS != nil {
protocol = "https"
2016-01-14 00:45:49 +03:00
}
2017-08-18 03:18:02 +03:00
r , _ := regexp . Compile ( ` (:\d+) ` )
2016-01-29 22:34:17 +03:00
match := r . FindStringSubmatch ( server . globalConfiguration . EntryPoints [ entryPoint . Redirect . EntryPoint ] . Address )
if len ( match ) == 0 {
return nil , errors . New ( "Bad Address format: " + server . globalConfiguration . EntryPoints [ entryPoint . Redirect . EntryPoint ] . Address )
}
replacement = protocol + "://$1" + match [ 0 ] + "$2"
2016-01-14 00:45:49 +03:00
}
2016-01-29 22:34:17 +03:00
rewrite , err := middlewares . NewRewrite ( regex , replacement , true )
if err != nil {
return nil , err
}
log . Debugf ( "Creating entryPoint redirect %s -> %s : %s -> %s" , entryPointName , entryPoint . Redirect . EntryPoint , regex , replacement )
2017-03-10 21:43:20 +03:00
return rewrite , nil
2016-01-29 22:34:17 +03:00
}
func ( server * Server ) buildDefaultHTTPRouter ( ) * mux . Router {
router := mux . NewRouter ( )
router . NotFoundHandler = http . HandlerFunc ( notFoundHandler )
2016-02-26 17:29:53 +03:00
router . StrictSlash ( true )
2016-09-20 17:43:09 +03:00
router . SkipClean ( true )
2016-01-29 22:34:17 +03:00
return router
2016-01-14 00:45:49 +03:00
}
2016-02-20 01:55:23 +03:00
2017-08-25 17:10:03 +03:00
func parseHealthCheckOptions ( lb healthcheck . LoadBalancer , backend string , hc * types . HealthCheck , hcConfig * configuration . HealthCheckConfig ) * healthcheck . Options {
2017-05-15 10:02:32 +03:00
if hc == nil || hc . Path == "" || hcConfig == nil {
2017-03-15 21:16:06 +03:00
return nil
}
2017-03-24 11:36:33 +03:00
interval := time . Duration ( hcConfig . Interval )
2017-03-15 21:16:06 +03:00
if hc . Interval != "" {
intervalOverride , err := time . ParseDuration ( hc . Interval )
switch {
case err != nil :
log . Errorf ( "Illegal healthcheck interval for backend '%s': %s" , backend , err )
case intervalOverride <= 0 :
log . Errorf ( "Healthcheck interval smaller than zero for backend '%s', backend" , backend )
default :
interval = intervalOverride
}
}
return & healthcheck . Options {
Path : hc . Path ,
2017-09-18 16:50:03 +03:00
Port : hc . Port ,
2017-03-15 21:16:06 +03:00
Interval : interval ,
LB : lb ,
}
}
2016-03-30 20:05:43 +03:00
func getRoute ( serverRoute * serverRoute , route * types . Route ) error {
2016-03-27 03:05:17 +03:00
rules := Rules { route : serverRoute }
newRoute , err := rules . Parse ( route . Rule )
if err != nil {
return err
2016-02-26 17:29:53 +03:00
}
2016-06-03 18:58:33 +03:00
newRoute . Priority ( serverRoute . route . GetPriority ( ) + len ( route . Rule ) )
2016-03-27 03:05:17 +03:00
serverRoute . route = newRoute
return nil
2016-02-26 17:29:53 +03:00
}
2016-02-20 01:55:23 +03:00
func sortedFrontendNamesForConfig ( configuration * types . Configuration ) [ ] string {
keys := [ ] string { }
for key := range configuration . Frontends {
keys = append ( keys , key )
}
sort . Strings ( keys )
return keys
}
2017-05-11 01:34:47 +03:00
func ( server * Server ) configureFrontends ( frontends map [ string ] * types . Frontend ) {
for _ , frontend := range frontends {
// default endpoints if not defined in frontends
if len ( frontend . EntryPoints ) == 0 {
frontend . EntryPoints = server . globalConfiguration . DefaultEntryPoints
}
}
}
2017-05-16 00:53:35 +03:00
func ( * Server ) configureBackends ( backends map [ string ] * types . Backend ) {
2017-05-11 01:34:47 +03:00
for backendName , backend := range backends {
_ , err := types . NewLoadBalancerMethod ( backend . LoadBalancer )
if err != nil {
log . Debugf ( "Validation of load balancer method for backend %s failed: %s. Using default method wrr." , backendName , err )
var sticky bool
if backend . LoadBalancer != nil {
sticky = backend . LoadBalancer . Sticky
}
backend . LoadBalancer = & types . LoadBalancer {
Method : "wrr" ,
Sticky : sticky ,
}
}
}
}
2017-04-18 09:22:06 +03:00
2017-08-23 21:46:03 +03:00
func ( server * Server ) registerMetricClients ( metricsConfig * types . Metrics ) {
registries := [ ] metrics . Registry { }
2017-07-21 01:26:43 +03:00
2017-08-23 21:46:03 +03:00
if metricsConfig . Prometheus != nil {
registries = append ( registries , metrics . RegisterPrometheus ( metricsConfig . Prometheus ) )
log . Debug ( "Configured Prometheus metrics" )
}
if metricsConfig . Datadog != nil {
registries = append ( registries , metrics . RegisterDatadog ( metricsConfig . Datadog ) )
log . Debugf ( "Configured DataDog metrics pushing to %s once every %s" , metricsConfig . Datadog . Address , metricsConfig . Datadog . PushInterval )
}
if metricsConfig . StatsD != nil {
registries = append ( registries , metrics . RegisterStatsd ( metricsConfig . StatsD ) )
log . Debugf ( "Configured StatsD metrics pushing to %s once every %s" , metricsConfig . StatsD . Address , metricsConfig . StatsD . PushInterval )
2017-04-18 09:22:06 +03:00
}
2017-08-23 21:46:03 +03:00
if len ( registries ) > 0 {
server . metricsRegistry = metrics . NewMultiRegistry ( registries )
2017-07-21 01:26:43 +03:00
}
}
2017-08-23 21:46:03 +03:00
func stopMetricsClients ( ) {
metrics . StopDatadog ( )
metrics . StopStatsd ( )
2017-07-21 01:26:43 +03:00
}
2017-08-28 13:50:02 +03:00
func ( server * Server ) buildRetryMiddleware ( handler http . Handler , globalConfig configuration . GlobalConfiguration , countServers int , backendName string ) http . Handler {
retryListeners := middlewares . RetryListeners { }
if server . metricsRegistry . IsEnabled ( ) {
retryListeners = append ( retryListeners , middlewares . NewMetricsRetryListener ( server . metricsRegistry , backendName ) )
}
if server . accessLoggerMiddleware != nil {
retryListeners = append ( retryListeners , & accesslog . SaveRetries { } )
}
retryAttempts := countServers
2017-04-18 09:22:06 +03:00
if globalConfig . Retry . Attempts > 0 {
2017-08-28 13:50:02 +03:00
retryAttempts = globalConfig . Retry . Attempts
2017-04-18 09:22:06 +03:00
}
2017-08-28 13:50:02 +03:00
log . Debugf ( "Creating retries max attempts %d" , retryAttempts )
2017-04-18 09:22:06 +03:00
2017-08-28 13:50:02 +03:00
return middlewares . NewRetry ( retryAttempts , handler , retryListeners )
2017-04-18 09:22:06 +03:00
}