2019-03-14 09:30:04 +01:00
package tcp
import (
"context"
2019-09-13 20:00:06 +02:00
"errors"
2019-03-14 09:30:04 +01:00
"fmt"
2022-09-14 20:42:08 +08:00
"math/rand"
2019-03-14 09:30:04 +01:00
"net"
2019-09-13 17:46:04 +02:00
"time"
2019-03-14 09:30:04 +01:00
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/tcp"
2024-01-29 17:32:05 +01:00
"golang.org/x/net/proxy"
2019-03-14 09:30:04 +01:00
)
2020-05-11 12:06:07 +02:00
// Manager is the TCPHandlers factory.
2019-03-14 09:30:04 +01:00
type Manager struct {
2022-12-09 09:58:05 +01:00
dialerManager * tcp . DialerManager
configs map [ string ] * runtime . TCPServiceInfo
rand * rand . Rand // For the initial shuffling of load-balancers.
2019-03-14 09:30:04 +01:00
}
2020-05-11 12:06:07 +02:00
// NewManager creates a new manager.
2022-12-09 09:58:05 +01:00
func NewManager ( conf * runtime . Configuration , dialerManager * tcp . DialerManager ) * Manager {
2019-03-14 09:30:04 +01:00
return & Manager {
2022-12-09 09:58:05 +01:00
dialerManager : dialerManager ,
configs : conf . TCPServices ,
rand : rand . New ( rand . NewSource ( time . Now ( ) . UnixNano ( ) ) ) ,
2019-03-14 09:30:04 +01:00
}
}
// BuildTCP Creates a tcp.Handler for a service configuration.
func ( m * Manager ) BuildTCP ( rootCtx context . Context , serviceName string ) ( tcp . Handler , error ) {
2020-01-27 10:40:05 +01:00
serviceQualifiedName := provider . GetQualifiedName ( rootCtx , serviceName )
2022-11-21 18:36:05 +01:00
logger := log . Ctx ( rootCtx ) . With ( ) . Str ( logs . ServiceName , serviceQualifiedName ) . Logger ( )
2020-01-27 10:40:05 +01:00
ctx := provider . AddInContext ( rootCtx , serviceQualifiedName )
2019-05-09 14:30:06 +02:00
conf , ok := m . configs [ serviceQualifiedName ]
if ! ok {
2019-05-16 10:58:06 +02:00
return nil , fmt . Errorf ( "the service %q does not exist" , serviceQualifiedName )
2019-05-09 14:30:06 +02:00
}
2019-09-13 20:00:06 +02:00
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" )
2019-07-19 16:42:04 +02:00
conf . AddError ( err , true )
return nil , err
2019-05-09 14:30:06 +02:00
}
2019-09-13 20:00:06 +02:00
switch {
case conf . LoadBalancer != nil :
loadBalancer := tcp . NewWRRLoadBalancer ( )
2019-05-09 14:30:06 +02:00
2024-01-29 17:32:05 +01:00
if conf . LoadBalancer . TerminationDelay != nil {
log . Ctx ( ctx ) . Warn ( ) . Msgf ( "Service %q load balancer uses `TerminationDelay`, but this option is deprecated, please use ServersTransport configuration instead." , serviceName )
}
2022-12-09 09:58:05 +01:00
if len ( conf . LoadBalancer . ServersTransport ) > 0 {
conf . LoadBalancer . ServersTransport = provider . GetQualifiedName ( ctx , conf . LoadBalancer . ServersTransport )
2019-09-13 20:00:06 +02:00
}
2019-05-09 14:30:06 +02:00
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 ( )
2019-09-13 20:00:06 +02:00
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" )
2019-09-13 20:00:06 +02:00
continue
}
2019-09-13 17:46:04 +02:00
2022-12-09 09:58:05 +01:00
dialer , err := m . dialerManager . Get ( conf . LoadBalancer . ServersTransport , server . TLS )
if err != nil {
return nil , err
}
2024-01-29 17:32:05 +01:00
// Handle TerminationDelay deprecated option.
if conf . LoadBalancer . ServersTransport == "" && conf . LoadBalancer . TerminationDelay != nil {
dialer = & dialerWrapper {
Dialer : dialer ,
terminationDelay : time . Duration ( * conf . LoadBalancer . TerminationDelay ) ,
}
}
2022-12-09 09:58:05 +01:00
handler , err := tcp . NewProxy ( server . Address , conf . LoadBalancer . ProxyProtocol , dialer )
2019-09-13 20:00:06 +02:00
if err != nil {
2022-11-21 18:36:05 +01:00
srvLogger . Error ( ) . Err ( err ) . Msg ( "Failed to create server" )
2019-09-13 20:00:06 +02:00
continue
}
2019-05-09 14:30:06 +02:00
2019-09-13 20:00:06 +02:00
loadBalancer . AddServer ( handler )
2022-11-21 18:36:05 +01:00
logger . Debug ( ) . Msg ( "Creating TCP server" )
2019-03-14 09:30:04 +01:00
}
2022-11-21 18:36:05 +01:00
2019-09-13 20:00:06 +02:00
return loadBalancer , nil
2022-11-21 18:36:05 +01:00
2019-09-13 20:00:06 +02:00
case conf . Weighted != nil :
loadBalancer := tcp . NewWRRLoadBalancer ( )
2022-09-14 20:42:08 +08:00
for _ , service := range shuffle ( conf . Weighted . Services , m . rand ) {
2022-11-21 18:36:05 +01:00
handler , err := m . BuildTCP ( ctx , service . Name )
2019-09-13 20:00:06 +02:00
if err != nil {
2022-11-21 18:36:05 +01:00
logger . Error ( ) . Err ( err ) . Msg ( "Failed to build TCP handler" )
2019-09-13 20:00:06 +02:00
return nil , err
}
2022-11-21 18:36:05 +01:00
2019-09-13 20:00:06 +02:00
loadBalancer . AddWeightServer ( handler , service . Weight )
}
2022-11-21 18:36:05 +01:00
2019-09-13 20:00:06 +02:00
return loadBalancer , nil
2022-11-21 18:36:05 +01:00
2019-09-13 20:00:06 +02:00
default :
2020-02-11 01:26:04 +01:00
err := fmt . Errorf ( "the service %q does not have any type defined" , serviceQualifiedName )
2019-09-13 20:00:06 +02:00
conf . AddError ( err , true )
return nil , err
2019-03-14 09:30:04 +01:00
}
}
2022-09-14 20:42:08 +08:00
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
}
2024-01-29 17:32:05 +01:00
// dialerWrapper is only used to handle TerminationDelay deprecated option on TCPServersLoadBalancer.
type dialerWrapper struct {
proxy . Dialer
terminationDelay time . Duration
}
func ( d dialerWrapper ) TerminationDelay ( ) time . Duration {
return d . terminationDelay
}