2018-11-14 10:18:03 +01:00
package middleware
import (
"context"
2019-04-01 15:30:07 +02:00
"errors"
2018-11-14 10:18:03 +01:00
"fmt"
"net/http"
2019-01-15 05:28:04 -08:00
"strings"
2018-11-14 10:18:03 +01:00
"github.com/containous/alice"
2020-09-16 15:46:04 +02:00
"github.com/traefik/traefik/v2/pkg/config/runtime"
"github.com/traefik/traefik/v2/pkg/middlewares/addprefix"
"github.com/traefik/traefik/v2/pkg/middlewares/auth"
"github.com/traefik/traefik/v2/pkg/middlewares/buffering"
"github.com/traefik/traefik/v2/pkg/middlewares/chain"
"github.com/traefik/traefik/v2/pkg/middlewares/circuitbreaker"
"github.com/traefik/traefik/v2/pkg/middlewares/compress"
"github.com/traefik/traefik/v2/pkg/middlewares/customerrors"
"github.com/traefik/traefik/v2/pkg/middlewares/headers"
"github.com/traefik/traefik/v2/pkg/middlewares/inflightreq"
"github.com/traefik/traefik/v2/pkg/middlewares/ipwhitelist"
"github.com/traefik/traefik/v2/pkg/middlewares/passtlsclientcert"
"github.com/traefik/traefik/v2/pkg/middlewares/ratelimiter"
"github.com/traefik/traefik/v2/pkg/middlewares/redirect"
"github.com/traefik/traefik/v2/pkg/middlewares/replacepath"
"github.com/traefik/traefik/v2/pkg/middlewares/replacepathregex"
"github.com/traefik/traefik/v2/pkg/middlewares/retry"
"github.com/traefik/traefik/v2/pkg/middlewares/stripprefix"
"github.com/traefik/traefik/v2/pkg/middlewares/stripprefixregex"
"github.com/traefik/traefik/v2/pkg/middlewares/tracing"
"github.com/traefik/traefik/v2/pkg/server/provider"
2018-11-14 10:18:03 +01:00
)
2019-01-15 05:28:04 -08:00
type middlewareStackType int
const (
middlewareStackKey middlewareStackType = iota
)
2020-05-11 12:06:07 +02:00
// Builder the middleware builder.
2018-11-14 10:18:03 +01:00
type Builder struct {
2019-07-15 17:04:04 +02:00
configs map [ string ] * runtime . MiddlewareInfo
2020-09-07 13:58:03 +02:00
pluginBuilder PluginsBuilder
2018-11-14 10:18:03 +01:00
serviceBuilder serviceBuilder
}
type serviceBuilder interface {
2020-09-01 18:16:04 +02:00
BuildHTTP ( ctx context . Context , serviceName string ) ( http . Handler , error )
2018-11-14 10:18:03 +01:00
}
2020-05-11 12:06:07 +02:00
// NewBuilder creates a new Builder.
2020-09-07 13:58:03 +02:00
func NewBuilder ( configs map [ string ] * runtime . MiddlewareInfo , serviceBuilder serviceBuilder , pluginBuilder PluginsBuilder ) * Builder {
2020-04-20 18:36:34 +02:00
return & Builder { configs : configs , serviceBuilder : serviceBuilder , pluginBuilder : pluginBuilder }
2018-11-14 10:18:03 +01:00
}
2020-05-11 12:06:07 +02:00
// BuildChain creates a middleware chain.
2019-01-15 05:28:04 -08:00
func ( b * Builder ) BuildChain ( ctx context . Context , middlewares [ ] string ) * alice . Chain {
2018-11-14 10:18:03 +01:00
chain := alice . New ( )
2019-01-30 16:24:07 +01:00
for _ , name := range middlewares {
2020-01-27 10:40:05 +01:00
middlewareName := provider . GetQualifiedName ( ctx , name )
2019-01-18 15:18:04 +01:00
2019-01-15 05:28:04 -08:00
chain = chain . Append ( func ( next http . Handler ) ( http . Handler , error ) {
2020-01-27 10:40:05 +01:00
constructorContext := provider . AddInContext ( ctx , middlewareName )
2019-05-16 10:58:06 +02:00
if midInf , ok := b . configs [ middlewareName ] ; ! ok || midInf . Middleware == nil {
2019-01-18 15:18:04 +01:00
return nil , fmt . Errorf ( "middleware %q does not exist" , middlewareName )
}
2019-01-15 05:28:04 -08:00
var err error
2019-04-01 15:30:07 +02:00
if constructorContext , err = checkRecursion ( constructorContext , middlewareName ) ; err != nil {
2019-07-19 16:42:04 +02:00
b . configs [ middlewareName ] . AddError ( err , true )
2019-01-15 05:28:04 -08:00
return nil , err
}
2018-11-14 10:18:03 +01:00
2019-05-16 10:58:06 +02:00
constructor , err := b . buildConstructor ( constructorContext , middlewareName )
2019-01-15 05:28:04 -08:00
if err != nil {
2019-07-19 16:42:04 +02:00
b . configs [ middlewareName ] . AddError ( err , true )
2019-05-16 10:58:06 +02:00
return nil , err
}
handler , err := constructor ( next )
if err != nil {
2019-07-19 16:42:04 +02:00
b . configs [ middlewareName ] . AddError ( err , true )
2019-05-16 10:58:06 +02:00
return nil , err
2019-01-15 05:28:04 -08:00
}
2019-05-16 10:58:06 +02:00
return handler , nil
2019-01-15 05:28:04 -08:00
} )
}
return & chain
}
2019-04-01 15:30:07 +02:00
func checkRecursion ( ctx context . Context , middlewareName string ) ( context . Context , error ) {
2019-01-15 05:28:04 -08:00
currentStack , ok := ctx . Value ( middlewareStackKey ) . ( [ ] string )
if ! ok {
currentStack = [ ] string { }
}
if inSlice ( middlewareName , currentStack ) {
return ctx , fmt . Errorf ( "could not instantiate middleware %s: recursion detected in %s" , middlewareName , strings . Join ( append ( currentStack , middlewareName ) , "->" ) )
2018-11-14 10:18:03 +01:00
}
2019-01-15 05:28:04 -08:00
return context . WithValue ( ctx , middlewareStackKey , append ( currentStack , middlewareName ) ) , nil
2018-11-14 10:18:03 +01:00
}
2020-05-11 12:06:07 +02:00
// it is the responsibility of the caller to make sure that b.configs[middlewareName].Middleware exists.
2019-05-16 10:58:06 +02:00
func ( b * Builder ) buildConstructor ( ctx context . Context , middlewareName string ) ( alice . Constructor , error ) {
config := b . configs [ middlewareName ]
2019-09-03 15:22:05 +02:00
if config == nil || config . Middleware == nil {
return nil , fmt . Errorf ( "invalid middleware %q configuration" , middlewareName )
}
2018-11-14 10:18:03 +01:00
var middleware alice . Constructor
2019-04-01 15:30:07 +02:00
badConf := errors . New ( "cannot create middleware: multi-types middleware not supported, consider declaring two different pieces of middleware instead" )
2018-11-14 10:18:03 +01:00
// AddPrefix
if config . AddPrefix != nil {
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return addprefix . New ( ctx , next , * config . AddPrefix , middlewareName )
2018-11-14 10:18:03 +01:00
}
}
// BasicAuth
if config . BasicAuth != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return auth . NewBasic ( ctx , next , * config . BasicAuth , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// Buffering
2019-09-03 15:02:05 +02:00
if config . Buffering != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return buffering . New ( ctx , next , * config . Buffering , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// Chain
if config . Chain != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-09-10 16:12:05 +02:00
2019-09-25 18:20:04 +02:00
var qualifiedNames [ ] string
for _ , name := range config . Chain . Middlewares {
2020-01-27 10:40:05 +01:00
qualifiedNames = append ( qualifiedNames , provider . GetQualifiedName ( ctx , name ) )
2019-09-10 16:12:05 +02:00
}
config . Chain . Middlewares = qualifiedNames
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return chain . New ( ctx , next , * config . Chain , b , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// CircuitBreaker
if config . CircuitBreaker != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return circuitbreaker . New ( ctx , next , * config . CircuitBreaker , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// Compress
if config . Compress != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
2019-10-31 11:36:05 +01:00
return compress . New ( ctx , next , * config . Compress , middlewareName )
2019-04-01 15:30:07 +02:00
}
2018-11-14 10:18:03 +01:00
}
2020-01-21 18:06:03 +01:00
// ContentType
if config . ContentType != nil {
if middleware != nil {
return nil , badConf
}
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return http . HandlerFunc ( func ( rw http . ResponseWriter , req * http . Request ) {
if ! config . ContentType . AutoDetect {
rw . Header ( ) [ "Content-Type" ] = nil
}
next . ServeHTTP ( rw , req )
} ) , nil
}
}
2018-11-14 10:18:03 +01:00
// CustomErrors
if config . Errors != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return customerrors . New ( ctx , next , * config . Errors , b . serviceBuilder , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// DigestAuth
if config . DigestAuth != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return auth . NewDigest ( ctx , next , * config . DigestAuth , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// ForwardAuth
if config . ForwardAuth != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return auth . NewForward ( ctx , next , * config . ForwardAuth , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// Headers
if config . Headers != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return headers . New ( ctx , next , * config . Headers , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// IPWhiteList
if config . IPWhiteList != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return ipwhitelist . New ( ctx , next , * config . IPWhiteList , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
2019-08-26 12:20:06 +02:00
// InFlightReq
2019-09-03 15:02:05 +02:00
if config . InFlightReq != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
2019-08-26 12:20:06 +02:00
return inflightreq . New ( ctx , next , * config . InFlightReq , middlewareName )
2019-04-01 15:30:07 +02:00
}
2018-11-14 10:18:03 +01:00
}
// PassTLSClientCert
if config . PassTLSClientCert != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return passtlsclientcert . New ( ctx , next , * config . PassTLSClientCert , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// RateLimit
2019-08-26 12:20:06 +02:00
if config . RateLimit != nil {
if middleware != nil {
return nil , badConf
}
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return ratelimiter . New ( ctx , next , * config . RateLimit , middlewareName )
}
}
2018-11-14 10:18:03 +01:00
2019-01-21 23:30:04 -08:00
// RedirectRegex
if config . RedirectRegex != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2019-01-21 23:30:04 -08:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return redirect . NewRedirectRegex ( ctx , next , * config . RedirectRegex , middlewareName )
}
2019-01-21 23:30:04 -08:00
}
// RedirectScheme
if config . RedirectScheme != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return redirect . NewRedirectScheme ( ctx , next , * config . RedirectScheme , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// ReplacePath
if config . ReplacePath != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return replacepath . New ( ctx , next , * config . ReplacePath , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// ReplacePathRegex
if config . ReplacePathRegex != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return replacepathregex . New ( ctx , next , * config . ReplacePathRegex , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// Retry
if config . Retry != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
// FIXME missing metrics / accessLog
return retry . New ( ctx , next , * config . Retry , retry . Listeners { } , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// StripPrefix
if config . StripPrefix != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return stripprefix . New ( ctx , next , * config . StripPrefix , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
// StripPrefixRegex
if config . StripPrefixRegex != nil {
2019-04-01 15:30:07 +02:00
if middleware != nil {
2018-11-14 10:18:03 +01:00
return nil , badConf
}
2019-04-01 15:30:07 +02:00
middleware = func ( next http . Handler ) ( http . Handler , error ) {
return stripprefixregex . New ( ctx , next , * config . StripPrefixRegex , middlewareName )
}
2018-11-14 10:18:03 +01:00
}
2020-04-20 18:36:34 +02:00
// Plugin
if config . Plugin != nil {
if middleware != nil {
return nil , badConf
}
pluginType , rawPluginConfig , err := findPluginConfig ( config . Plugin )
if err != nil {
return nil , err
}
2020-09-07 13:58:03 +02:00
plug , err := b . pluginBuilder . Build ( pluginType , rawPluginConfig , middlewareName )
2020-04-20 18:36:34 +02:00
if err != nil {
return nil , err
}
middleware = func ( next http . Handler ) ( http . Handler , error ) {
2020-09-07 13:58:03 +02:00
return plug ( ctx , next )
2020-04-20 18:36:34 +02:00
}
}
2019-01-18 15:18:04 +01:00
if middleware == nil {
2019-09-26 10:04:04 +02:00
return nil , fmt . Errorf ( "invalid middleware %q configuration: invalid middleware type or middleware does not exist" , middlewareName )
2019-01-18 15:18:04 +01:00
}
2018-11-14 10:18:03 +01:00
return tracing . Wrap ( ctx , middleware ) , nil
}
2019-01-15 05:28:04 -08:00
func inSlice ( element string , stack [ ] string ) bool {
for _ , value := range stack {
if value == element {
return true
}
}
return false
}