2018-11-14 10:18:03 +01:00
|
|
|
package router
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2019-07-19 16:42:04 +02:00
|
|
|
"errors"
|
2022-12-06 18:28:05 +01:00
|
|
|
"fmt"
|
2018-11-14 10:18:03 +01:00
|
|
|
"net/http"
|
|
|
|
|
|
|
|
"github.com/containous/alice"
|
2022-11-21 18:36:05 +01:00
|
|
|
"github.com/rs/zerolog/log"
|
2023-02-03 15:24:05 +01:00
|
|
|
"github.com/traefik/traefik/v3/pkg/config/runtime"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/logs"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/metrics"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/middlewares/accesslog"
|
2023-11-29 12:20:57 +01:00
|
|
|
"github.com/traefik/traefik/v3/pkg/middlewares/denyrouterrecursion"
|
2023-02-03 15:24:05 +01:00
|
|
|
metricsMiddle "github.com/traefik/traefik/v3/pkg/middlewares/metrics"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/middlewares/recovery"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/middlewares/tracing"
|
|
|
|
httpmuxer "github.com/traefik/traefik/v3/pkg/muxer/http"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/server/middleware"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/server/provider"
|
|
|
|
"github.com/traefik/traefik/v3/pkg/tls"
|
2018-11-14 10:18:03 +01:00
|
|
|
)
|
|
|
|
|
2019-11-14 16:40:05 +01:00
|
|
|
type middlewareBuilder interface {
|
|
|
|
BuildChain(ctx context.Context, names []string) *alice.Chain
|
|
|
|
}
|
|
|
|
|
|
|
|
type serviceManager interface {
|
2020-09-01 18:16:04 +02:00
|
|
|
BuildHTTP(rootCtx context.Context, serviceName string) (http.Handler, error)
|
2022-11-16 11:38:07 +01:00
|
|
|
LaunchHealthCheck(ctx context.Context)
|
2019-11-14 16:40:05 +01:00
|
|
|
}
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// Manager A route/router manager.
|
2019-11-14 16:40:05 +01:00
|
|
|
type Manager struct {
|
|
|
|
routerHandlers map[string]http.Handler
|
|
|
|
serviceManager serviceManager
|
2021-04-30 10:22:04 +02:00
|
|
|
metricsRegistry metrics.Registry
|
2019-11-14 16:40:05 +01:00
|
|
|
middlewaresBuilder middlewareBuilder
|
|
|
|
chainBuilder *middleware.ChainBuilder
|
|
|
|
conf *runtime.Configuration
|
2022-12-06 18:28:05 +01:00
|
|
|
tlsManager *tls.Manager
|
2019-11-14 16:40:05 +01:00
|
|
|
}
|
|
|
|
|
2022-12-06 18:28:05 +01:00
|
|
|
// NewManager creates a new Manager.
|
|
|
|
func NewManager(conf *runtime.Configuration, serviceManager serviceManager, middlewaresBuilder middlewareBuilder, chainBuilder *middleware.ChainBuilder, metricsRegistry metrics.Registry, tlsManager *tls.Manager) *Manager {
|
2018-11-14 10:18:03 +01:00
|
|
|
return &Manager{
|
|
|
|
routerHandlers: make(map[string]http.Handler),
|
|
|
|
serviceManager: serviceManager,
|
2021-04-30 10:22:04 +02:00
|
|
|
metricsRegistry: metricsRegistry,
|
2018-11-14 10:18:03 +01:00
|
|
|
middlewaresBuilder: middlewaresBuilder,
|
2019-11-14 16:40:05 +01:00
|
|
|
chainBuilder: chainBuilder,
|
2019-06-17 18:14:08 +02:00
|
|
|
conf: conf,
|
2022-12-06 18:28:05 +01:00
|
|
|
tlsManager: tlsManager,
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-15 17:04:04 +02:00
|
|
|
func (m *Manager) getHTTPRouters(ctx context.Context, entryPoints []string, tls bool) map[string]map[string]*runtime.RouterInfo {
|
2019-06-17 18:14:08 +02:00
|
|
|
if m.conf != nil {
|
2019-07-15 17:04:04 +02:00
|
|
|
return m.conf.GetRoutersByEntryPoints(ctx, entryPoints, tls)
|
2019-06-17 18:14:08 +02:00
|
|
|
}
|
|
|
|
|
2019-07-15 17:04:04 +02:00
|
|
|
return make(map[string]map[string]*runtime.RouterInfo)
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2020-05-11 12:06:07 +02:00
|
|
|
// BuildHandlers Builds handler for all entry points.
|
2019-03-14 09:30:04 +01:00
|
|
|
func (m *Manager) BuildHandlers(rootCtx context.Context, entryPoints []string, tls bool) map[string]http.Handler {
|
2018-11-14 10:18:03 +01:00
|
|
|
entryPointHandlers := make(map[string]http.Handler)
|
2019-06-17 18:14:08 +02:00
|
|
|
|
|
|
|
for entryPointName, routers := range m.getHTTPRouters(rootCtx, entryPoints, tls) {
|
2019-01-30 16:24:07 +01:00
|
|
|
entryPointName := entryPointName
|
2022-11-21 18:36:05 +01:00
|
|
|
|
|
|
|
logger := log.Ctx(rootCtx).With().Str(logs.EntryPointName, entryPointName).Logger()
|
|
|
|
ctx := logger.WithContext(rootCtx)
|
2018-11-14 10:18:03 +01:00
|
|
|
|
|
|
|
handler, err := m.buildEntryPointHandler(ctx, routers)
|
|
|
|
if err != nil {
|
2022-11-21 18:36:05 +01:00
|
|
|
logger.Error().Err(err).Send()
|
2018-11-14 10:18:03 +01:00
|
|
|
continue
|
|
|
|
}
|
2019-01-18 15:18:04 +01:00
|
|
|
|
|
|
|
handlerWithAccessLog, err := alice.New(func(next http.Handler) (http.Handler, error) {
|
2023-10-11 16:20:26 +02:00
|
|
|
return accesslog.NewFieldHandler(next, logs.EntryPointName, entryPointName, accesslog.InitServiceFields), nil
|
2019-01-18 15:18:04 +01:00
|
|
|
}).Then(handler)
|
|
|
|
if err != nil {
|
2022-11-21 18:36:05 +01:00
|
|
|
logger.Error().Err(err).Send()
|
2019-01-18 15:18:04 +01:00
|
|
|
entryPointHandlers[entryPointName] = handler
|
|
|
|
} else {
|
|
|
|
entryPointHandlers[entryPointName] = handlerWithAccessLog
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2019-11-14 16:40:05 +01:00
|
|
|
for _, entryPointName := range entryPoints {
|
2022-11-21 18:36:05 +01:00
|
|
|
logger := log.Ctx(rootCtx).With().Str(logs.EntryPointName, entryPointName).Logger()
|
|
|
|
ctx := logger.WithContext(rootCtx)
|
2019-11-14 16:40:05 +01:00
|
|
|
|
|
|
|
handler, ok := entryPointHandlers[entryPointName]
|
|
|
|
if !ok || handler == nil {
|
|
|
|
handler = BuildDefaultHTTPRouter()
|
|
|
|
}
|
|
|
|
|
|
|
|
handlerWithMiddlewares, err := m.chainBuilder.Build(ctx, entryPointName).Then(handler)
|
|
|
|
if err != nil {
|
2022-11-21 18:36:05 +01:00
|
|
|
logger.Error().Err(err).Send()
|
2019-11-14 16:40:05 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
entryPointHandlers[entryPointName] = handlerWithMiddlewares
|
|
|
|
}
|
|
|
|
|
2018-11-14 10:18:03 +01:00
|
|
|
return entryPointHandlers
|
|
|
|
}
|
|
|
|
|
2019-07-15 17:04:04 +02:00
|
|
|
func (m *Manager) buildEntryPointHandler(ctx context.Context, configs map[string]*runtime.RouterInfo) (http.Handler, error) {
|
2022-03-17 11:02:08 -06:00
|
|
|
muxer, err := httpmuxer.NewMuxer()
|
2019-01-30 16:24:07 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-11-14 10:18:03 +01:00
|
|
|
|
|
|
|
for routerName, routerConfig := range configs {
|
2022-11-21 18:36:05 +01:00
|
|
|
logger := log.Ctx(ctx).With().Str(logs.RouterName, routerName).Logger()
|
|
|
|
ctxRouter := logger.WithContext(provider.AddInContext(ctx, routerName))
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2023-01-09 17:24:05 +01:00
|
|
|
if routerConfig.Priority == 0 {
|
|
|
|
routerConfig.Priority = httpmuxer.GetRulePriority(routerConfig.Rule)
|
|
|
|
}
|
|
|
|
|
2019-06-17 18:14:08 +02:00
|
|
|
handler, err := m.buildRouterHandler(ctxRouter, routerName, routerConfig)
|
2018-11-14 10:18:03 +01:00
|
|
|
if err != nil {
|
2019-07-15 17:04:04 +02:00
|
|
|
routerConfig.AddError(err, true)
|
2022-11-21 18:36:05 +01:00
|
|
|
logger.Error().Err(err).Send()
|
2018-11-14 10:18:03 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-01-09 17:24:05 +01:00
|
|
|
if err = muxer.AddRoute(routerConfig.Rule, routerConfig.Priority, handler); err != nil {
|
2019-07-15 17:04:04 +02:00
|
|
|
routerConfig.AddError(err, true)
|
2022-11-21 18:36:05 +01:00
|
|
|
logger.Error().Err(err).Send()
|
2018-11-14 10:18:03 +01:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
chain := alice.New()
|
|
|
|
chain = chain.Append(func(next http.Handler) (http.Handler, error) {
|
2021-01-21 10:04:04 +01:00
|
|
|
return recovery.New(ctx, next)
|
2018-11-14 10:18:03 +01:00
|
|
|
})
|
|
|
|
|
2022-03-17 11:02:08 -06:00
|
|
|
return chain.Then(muxer)
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
|
|
|
|
2019-07-15 17:04:04 +02:00
|
|
|
func (m *Manager) buildRouterHandler(ctx context.Context, routerName string, routerConfig *runtime.RouterInfo) (http.Handler, error) {
|
2018-11-14 10:18:03 +01:00
|
|
|
if handler, ok := m.routerHandlers[routerName]; ok {
|
|
|
|
return handler, nil
|
|
|
|
}
|
|
|
|
|
2022-12-06 18:28:05 +01:00
|
|
|
if routerConfig.TLS != nil {
|
|
|
|
// Don't build the router if the TLSOptions configuration is invalid.
|
|
|
|
tlsOptionsName := tls.DefaultTLSConfigName
|
|
|
|
if len(routerConfig.TLS.Options) > 0 && routerConfig.TLS.Options != tls.DefaultTLSConfigName {
|
|
|
|
tlsOptionsName = provider.GetQualifiedName(ctx, routerConfig.TLS.Options)
|
|
|
|
}
|
|
|
|
if _, err := m.tlsManager.Get(tls.DefaultTLSStoreName, tlsOptionsName); err != nil {
|
|
|
|
return nil, fmt.Errorf("building router handler: %w", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-17 18:14:08 +02:00
|
|
|
handler, err := m.buildHTTPHandler(ctx, routerConfig, routerName)
|
2018-11-14 10:18:03 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
handlerWithAccessLog, err := alice.New(func(next http.Handler) (http.Handler, error) {
|
|
|
|
return accesslog.NewFieldHandler(next, accesslog.RouterName, routerName, nil), nil
|
|
|
|
}).Then(handler)
|
|
|
|
if err != nil {
|
2022-11-21 18:36:05 +01:00
|
|
|
log.Ctx(ctx).Error().Err(err).Send()
|
2018-11-14 10:18:03 +01:00
|
|
|
m.routerHandlers[routerName] = handler
|
|
|
|
} else {
|
|
|
|
m.routerHandlers[routerName] = handlerWithAccessLog
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.routerHandlers[routerName], nil
|
|
|
|
}
|
|
|
|
|
2019-07-15 17:04:04 +02:00
|
|
|
func (m *Manager) buildHTTPHandler(ctx context.Context, router *runtime.RouterInfo, routerName string) (http.Handler, error) {
|
2019-09-25 18:20:04 +02:00
|
|
|
var qualifiedNames []string
|
|
|
|
for _, name := range router.Middlewares {
|
2020-01-27 10:40:05 +01:00
|
|
|
qualifiedNames = append(qualifiedNames, provider.GetQualifiedName(ctx, name))
|
2019-04-04 11:50:04 +02:00
|
|
|
}
|
2019-09-10 16:12:05 +02:00
|
|
|
router.Middlewares = qualifiedNames
|
2018-11-14 10:18:03 +01:00
|
|
|
|
2019-07-19 16:42:04 +02:00
|
|
|
if router.Service == "" {
|
|
|
|
return nil, errors.New("the service is missing on the router")
|
|
|
|
}
|
|
|
|
|
2020-09-01 18:16:04 +02:00
|
|
|
sHandler, err := m.serviceManager.BuildHTTP(ctx, router.Service)
|
2018-11-14 10:18:03 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-01-15 05:28:04 -08:00
|
|
|
mHandler := m.middlewaresBuilder.BuildChain(ctx, router.Middlewares)
|
2018-11-14 10:18:03 +01:00
|
|
|
|
|
|
|
tHandler := func(next http.Handler) (http.Handler, error) {
|
|
|
|
return tracing.NewForwarder(ctx, routerName, router.Service, next), nil
|
|
|
|
}
|
|
|
|
|
2021-04-30 10:22:04 +02:00
|
|
|
chain := alice.New()
|
|
|
|
|
|
|
|
if m.metricsRegistry != nil && m.metricsRegistry.IsRouterEnabled() {
|
2021-09-15 17:26:06 +02:00
|
|
|
chain = chain.Append(metricsMiddle.WrapRouterHandler(ctx, m.metricsRegistry, routerName, provider.GetQualifiedName(ctx, router.Service)))
|
2021-04-30 10:22:04 +02:00
|
|
|
}
|
|
|
|
|
2023-11-21 15:08:06 +01:00
|
|
|
if router.DefaultRule {
|
|
|
|
chain = chain.Append(denyrouterrecursion.WrapHandler(routerName))
|
|
|
|
}
|
|
|
|
|
2021-04-30 10:22:04 +02:00
|
|
|
return chain.Extend(*mHandler).Append(tHandler).Then(sHandler)
|
2018-11-14 10:18:03 +01:00
|
|
|
}
|
2019-11-14 16:40:05 +01:00
|
|
|
|
|
|
|
// BuildDefaultHTTPRouter creates a default HTTP router.
|
|
|
|
func BuildDefaultHTTPRouter() http.Handler {
|
|
|
|
return http.NotFoundHandler()
|
|
|
|
}
|