2024-03-12 18:05:41 +01:00
package server
import (
"context"
"net"
"testing"
"time"
2024-06-13 01:20:46 +02:00
"github.com/stretchr/testify/require"
"go.opentelemetry.io/otel"
2024-04-23 14:42:53 +02:00
2024-09-08 12:06:14 +02:00
"github.com/netbirdio/management-integrations/integrations"
2024-03-12 18:05:41 +01:00
log "github.com/sirupsen/logrus"
"google.golang.org/grpc"
"google.golang.org/grpc/keepalive"
"github.com/netbirdio/netbird/client/internal"
"github.com/netbirdio/netbird/client/internal/peer"
mgmtProto "github.com/netbirdio/netbird/management/proto"
"github.com/netbirdio/netbird/management/server"
"github.com/netbirdio/netbird/management/server/activity"
2024-12-20 11:30:28 +01:00
"github.com/netbirdio/netbird/management/server/settings"
"github.com/netbirdio/netbird/management/server/store"
2024-08-07 10:52:31 +02:00
"github.com/netbirdio/netbird/management/server/telemetry"
2024-03-12 18:05:41 +01:00
"github.com/netbirdio/netbird/signal/proto"
signalServer "github.com/netbirdio/netbird/signal/server"
)
var (
kaep = keepalive . EnforcementPolicy {
MinTime : 15 * time . Second ,
PermitWithoutStream : true ,
}
kasp = keepalive . ServerParameters {
MaxConnectionIdle : 15 * time . Second ,
MaxConnectionAgeGrace : 5 * time . Second ,
Time : 5 * time . Second ,
Timeout : 2 * time . Second ,
}
)
// TestConnectWithRetryRuns checks that the connectWithRetry function runs and runs the retries according to the times specified via environment variables
// we will use a management server started via to simulate the server and capture the number of retries
func TestConnectWithRetryRuns ( t * testing . T ) {
// start the signal server
2024-06-13 01:20:46 +02:00
_ , signalAddr , err := startSignal ( t )
2024-03-12 18:05:41 +01:00
if err != nil {
t . Fatalf ( "failed to start signal server: %v" , err )
}
counter := 0
// start the management server
_ , mgmtAddr , err := startManagement ( t , signalAddr , & counter )
if err != nil {
t . Fatalf ( "failed to start management server: %v" , err )
}
ctx := internal . CtxInitState ( context . Background ( ) )
ctx , cancel := context . WithDeadline ( ctx , time . Now ( ) . Add ( 30 * time . Second ) )
defer cancel ( )
// create new server
s := New ( ctx , t . TempDir ( ) + "/config.json" , "debug" )
s . latestConfigInput . ManagementURL = "http://" + mgmtAddr
config , err := internal . UpdateOrCreateConfig ( s . latestConfigInput )
if err != nil {
t . Fatalf ( "failed to create config: %v" , err )
}
s . config = config
s . statusRecorder = peer . NewRecorder ( config . ManagementURL . String ( ) )
t . Setenv ( retryInitialIntervalVar , "1s" )
t . Setenv ( maxRetryIntervalVar , "2s" )
t . Setenv ( maxRetryTimeVar , "5s" )
t . Setenv ( retryMultiplierVar , "1" )
2024-09-02 19:19:14 +02:00
s . connectWithRetryRuns ( ctx , config , s . statusRecorder , nil )
2024-03-12 18:05:41 +01:00
if counter < 3 {
t . Fatalf ( "expected counter > 2, got %d" , counter )
}
}
type mockServer struct {
mgmtProto . ManagementServiceServer
counter * int
}
func ( m * mockServer ) Login ( ctx context . Context , req * mgmtProto . EncryptedMessage ) ( * mgmtProto . EncryptedMessage , error ) {
* m . counter ++
return m . ManagementServiceServer . Login ( ctx , req )
}
func startManagement ( t * testing . T , signalAddr string , counter * int ) ( * grpc . Server , string , error ) {
t . Helper ( )
dataDir := t . TempDir ( )
config := & server . Config {
Stuns : [ ] * server . Host { } ,
TURNConfig : & server . TURNConfig { } ,
Signal : & server . Host {
Proto : "http" ,
URI : signalAddr ,
} ,
Datadir : dataDir ,
HttpConfig : nil ,
}
lis , err := net . Listen ( "tcp" , "localhost:0" )
if err != nil {
return nil , "" , err
}
s := grpc . NewServer ( grpc . KeepaliveEnforcementPolicy ( kaep ) , grpc . KeepaliveParams ( kasp ) )
2024-12-20 11:30:28 +01:00
store , cleanUp , err := store . NewTestStoreFromSQL ( context . Background ( ) , "" , config . Datadir )
2024-03-12 18:05:41 +01:00
if err != nil {
return nil , "" , err
}
2024-05-16 18:28:37 +02:00
t . Cleanup ( cleanUp )
2024-03-12 18:05:41 +01:00
peersUpdateManager := server . NewPeersUpdateManager ( nil )
eventStore := & activity . InMemoryEventStore { }
if err != nil {
return nil , "" , err
}
2024-07-03 11:33:02 +02:00
ia , _ := integrations . NewIntegratedValidator ( context . Background ( ) , eventStore )
2024-08-07 10:52:31 +02:00
metrics , err := telemetry . NewDefaultAppMetrics ( context . Background ( ) )
require . NoError ( t , err )
accountManager , err := server . BuildManager ( context . Background ( ) , store , peersUpdateManager , nil , "" , "netbird.selfhosted" , eventStore , nil , false , ia , metrics )
2024-03-12 18:05:41 +01:00
if err != nil {
return nil , "" , err
}
2024-09-08 12:06:14 +02:00
secretsManager := server . NewTimeBasedAuthSecretsManager ( peersUpdateManager , config . TURNConfig , config . Relay )
2024-12-20 11:30:28 +01:00
mgmtServer , err := server . NewServer ( context . Background ( ) , config , accountManager , settings . NewManager ( store ) , peersUpdateManager , secretsManager , nil , nil )
2024-03-12 18:05:41 +01:00
if err != nil {
return nil , "" , err
}
mock := & mockServer {
ManagementServiceServer : mgmtServer ,
counter : counter ,
}
mgmtProto . RegisterManagementServiceServer ( s , mock )
go func ( ) {
if err = s . Serve ( lis ) ; err != nil {
log . Fatalf ( "failed to serve: %v" , err )
}
} ( )
return s , lis . Addr ( ) . String ( ) , nil
}
2024-06-13 01:20:46 +02:00
func startSignal ( t * testing . T ) ( * grpc . Server , string , error ) {
t . Helper ( )
2024-03-12 18:05:41 +01:00
s := grpc . NewServer ( grpc . KeepaliveEnforcementPolicy ( kaep ) , grpc . KeepaliveParams ( kasp ) )
lis , err := net . Listen ( "tcp" , "localhost:0" )
if err != nil {
log . Fatalf ( "failed to listen: %v" , err )
}
2024-09-29 00:22:47 +02:00
srv , err := signalServer . NewServer ( context . Background ( ) , otel . Meter ( "" ) )
2024-06-13 01:20:46 +02:00
require . NoError ( t , err )
proto . RegisterSignalExchangeServer ( s , srv )
2024-03-12 18:05:41 +01:00
go func ( ) {
if err = s . Serve ( lis ) ; err != nil {
log . Fatalf ( "failed to serve: %v" , err )
}
} ( )
return s , lis . Addr ( ) . String ( ) , nil
}