2020-02-11 01:26:04 +01:00
|
|
|
package udp
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
2020-02-17 18:02:04 +01:00
|
|
|
"sort"
|
2020-02-11 01:26:04 +01:00
|
|
|
|
|
|
|
"github.com/containous/traefik/v2/pkg/config/runtime"
|
|
|
|
"github.com/containous/traefik/v2/pkg/log"
|
|
|
|
"github.com/containous/traefik/v2/pkg/server/provider"
|
|
|
|
udpservice "github.com/containous/traefik/v2/pkg/server/service/udp"
|
|
|
|
"github.com/containous/traefik/v2/pkg/udp"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewManager Creates a new Manager
|
|
|
|
func NewManager(conf *runtime.Configuration,
|
|
|
|
serviceManager *udpservice.Manager,
|
|
|
|
) *Manager {
|
|
|
|
return &Manager{
|
|
|
|
serviceManager: serviceManager,
|
|
|
|
conf: conf,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Manager is a route/router manager
|
|
|
|
type Manager struct {
|
|
|
|
serviceManager *udpservice.Manager
|
|
|
|
conf *runtime.Configuration
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Manager) getUDPRouters(ctx context.Context, entryPoints []string) map[string]map[string]*runtime.UDPRouterInfo {
|
|
|
|
if m.conf != nil {
|
|
|
|
return m.conf.GetUDPRoutersByEntryPoints(ctx, entryPoints)
|
|
|
|
}
|
|
|
|
|
|
|
|
return make(map[string]map[string]*runtime.UDPRouterInfo)
|
|
|
|
}
|
|
|
|
|
|
|
|
// BuildHandlers builds the handlers for the given entrypoints
|
|
|
|
func (m *Manager) BuildHandlers(rootCtx context.Context, entryPoints []string) map[string]udp.Handler {
|
|
|
|
entryPointsRouters := m.getUDPRouters(rootCtx, entryPoints)
|
|
|
|
|
|
|
|
entryPointHandlers := make(map[string]udp.Handler)
|
|
|
|
for _, entryPointName := range entryPoints {
|
|
|
|
entryPointName := entryPointName
|
|
|
|
|
|
|
|
routers := entryPointsRouters[entryPointName]
|
|
|
|
|
|
|
|
ctx := log.With(rootCtx, log.Str(log.EntryPointName, entryPointName))
|
|
|
|
|
2020-02-17 18:02:04 +01:00
|
|
|
if len(routers) > 1 {
|
|
|
|
log.FromContext(ctx).Warn("Config has more than one udp router for a given entrypoint.")
|
|
|
|
}
|
|
|
|
|
|
|
|
handlers, err := m.buildEntryPointHandlers(ctx, routers)
|
2020-02-11 01:26:04 +01:00
|
|
|
if err != nil {
|
|
|
|
log.FromContext(ctx).Error(err)
|
|
|
|
continue
|
|
|
|
}
|
2020-02-17 18:02:04 +01:00
|
|
|
|
|
|
|
if len(handlers) > 0 {
|
|
|
|
// As UDP support only one router per entrypoint, we only take the first one.
|
|
|
|
entryPointHandlers[entryPointName] = handlers[0]
|
|
|
|
}
|
2020-02-11 01:26:04 +01:00
|
|
|
}
|
|
|
|
return entryPointHandlers
|
|
|
|
}
|
|
|
|
|
2020-02-17 18:02:04 +01:00
|
|
|
func (m *Manager) buildEntryPointHandlers(ctx context.Context, configs map[string]*runtime.UDPRouterInfo) ([]udp.Handler, error) {
|
|
|
|
var rtNames []string
|
|
|
|
for routerName := range configs {
|
|
|
|
rtNames = append(rtNames, routerName)
|
2020-02-11 01:26:04 +01:00
|
|
|
}
|
2020-02-17 18:02:04 +01:00
|
|
|
|
|
|
|
sort.Slice(rtNames, func(i, j int) bool {
|
|
|
|
return rtNames[i] > rtNames[j]
|
|
|
|
})
|
|
|
|
|
|
|
|
var handlers []udp.Handler
|
|
|
|
|
|
|
|
for _, routerName := range rtNames {
|
|
|
|
routerConfig := configs[routerName]
|
|
|
|
|
2020-02-11 01:26:04 +01:00
|
|
|
ctxRouter := log.With(provider.AddInContext(ctx, routerName), log.Str(log.RouterName, routerName))
|
|
|
|
logger := log.FromContext(ctxRouter)
|
|
|
|
|
|
|
|
if routerConfig.Service == "" {
|
|
|
|
err := errors.New("the service is missing on the udp router")
|
|
|
|
routerConfig.AddError(err, true)
|
|
|
|
logger.Error(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
handler, err := m.serviceManager.BuildUDP(ctxRouter, routerConfig.Service)
|
|
|
|
if err != nil {
|
|
|
|
routerConfig.AddError(err, true)
|
|
|
|
logger.Error(err)
|
|
|
|
continue
|
|
|
|
}
|
2020-02-17 18:02:04 +01:00
|
|
|
|
|
|
|
handlers = append(handlers, handler)
|
2020-02-11 01:26:04 +01:00
|
|
|
}
|
|
|
|
|
2020-02-17 18:02:04 +01:00
|
|
|
return handlers, nil
|
2020-02-11 01:26:04 +01:00
|
|
|
}
|