traefik/pkg/server/service/udp/service.go

105 lines
2.9 KiB
Go
Raw Normal View History

package udp
import (
"context"
"errors"
"fmt"
"math/rand"
"net"
"time"
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/server/provider"
"github.com/traefik/traefik/v3/pkg/udp"
)
// Manager handles UDP services creation.
type Manager struct {
configs map[string]*runtime.UDPServiceInfo
rand *rand.Rand // For the initial shuffling of load-balancers.
}
2020-05-11 12:06:07 +02:00
// NewManager creates a new manager.
func NewManager(conf *runtime.Configuration) *Manager {
return &Manager{
configs: conf.UDPServices,
rand: rand.New(rand.NewSource(time.Now().UnixNano())),
}
}
// BuildUDP creates the UDP handler for the given service name.
func (m *Manager) BuildUDP(rootCtx context.Context, serviceName string) (udp.Handler, error) {
serviceQualifiedName := provider.GetQualifiedName(rootCtx, serviceName)
2022-11-21 18:36:05 +01:00
logger := log.Ctx(rootCtx).With().Str(logs.ServiceName, serviceQualifiedName).Logger()
ctx := provider.AddInContext(rootCtx, serviceQualifiedName)
conf, ok := m.configs[serviceQualifiedName]
if !ok {
2022-11-21 18:36:05 +01:00
return nil, fmt.Errorf("the UDP service %q does not exist", serviceQualifiedName)
}
if conf.LoadBalancer != nil && conf.Weighted != nil {
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
}
switch {
case conf.LoadBalancer != nil:
loadBalancer := udp.NewWRRLoadBalancer()
2022-11-21 18:36:05 +01:00
for index, server := range shuffle(conf.LoadBalancer.Servers, m.rand) {
srvLogger := logger.With().
Int(logs.ServerIndex, index).
Str("serverAddress", server.Address).Logger()
if _, _, err := net.SplitHostPort(server.Address); err != nil {
2022-11-21 18:36:05 +01:00
srvLogger.Error().Err(err).Msg("Failed to split host port")
continue
}
handler, err := udp.NewProxy(server.Address)
if err != nil {
2022-11-21 18:36:05 +01:00
srvLogger.Error().Err(err).Msg("Failed to create server")
continue
}
loadBalancer.AddServer(handler)
2022-11-21 18:36:05 +01:00
srvLogger.Debug().Msg("Creating UDP server")
}
2022-11-21 18:36:05 +01:00
return loadBalancer, nil
2022-11-21 18:36:05 +01:00
case conf.Weighted != nil:
loadBalancer := udp.NewWRRLoadBalancer()
for _, service := range shuffle(conf.Weighted.Services, m.rand) {
2022-11-21 18:36:05 +01:00
handler, err := m.BuildUDP(ctx, service.Name)
if err != nil {
2022-11-21 18:36:05 +01:00
logger.Error().Err(err).Msg("Failed to build UDP handler")
return nil, err
}
2022-11-21 18:36:05 +01:00
loadBalancer.AddWeightedServer(handler, service.Weight)
}
2022-11-21 18:36:05 +01:00
return loadBalancer, nil
2022-11-21 18:36:05 +01:00
default:
2022-11-21 18:36:05 +01:00
err := fmt.Errorf("the UDP service %q does not have any type defined", serviceQualifiedName)
conf.AddError(err, true)
return nil, err
}
}
func shuffle[T any](values []T, r *rand.Rand) []T {
shuffled := make([]T, len(values))
copy(shuffled, values)
r.Shuffle(len(shuffled), func(i, j int) { shuffled[i], shuffled[j] = shuffled[j], shuffled[i] })
return shuffled
}