mirror of
				https://github.com/redis/go-redis.git
				synced 2025-10-24 19:32:57 +03:00 
			
		
		
		
	* e2e wip * cleanup * remove unused fault injector mock * errChan in test * remove log messages tests * cleanup log messages * s/hitless/maintnotifications/ * fix moving when none * better logs * test with second client after action has started * Fixes Signed-off-by: Elena Kolevska <elena@kolevska.com> * Test fix Signed-off-by: Elena Kolevska <elena@kolevska.com> * feat(e2e-test): Extended e2e tests * imroved e2e test resiliency --------- Signed-off-by: Elena Kolevska <elena@kolevska.com> Co-authored-by: Elena Kolevska <elena@kolevska.com> Co-authored-by: Elena Kolevska <elena-kolevska@users.noreply.github.com> Co-authored-by: Hristo Temelski <hristo.temelski@redis.com>
		
			
				
	
	
		
			482 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			482 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package maintnotifications
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"net"
 | |
| 	"testing"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/redis/go-redis/v9/internal/util"
 | |
| )
 | |
| 
 | |
| func TestConfig(t *testing.T) {
 | |
| 	t.Run("DefaultConfig", func(t *testing.T) {
 | |
| 		config := DefaultConfig()
 | |
| 
 | |
| 		// MaxWorkers should be 0 in default config (auto-calculated)
 | |
| 		if config.MaxWorkers != 0 {
 | |
| 			t.Errorf("Expected MaxWorkers to be 0 (auto-calculated), got %d", config.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 		// HandoffQueueSize should be 0 in default config (auto-calculated)
 | |
| 		if config.HandoffQueueSize != 0 {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be 0 (auto-calculated), got %d", config.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		if config.RelaxedTimeout != 10*time.Second {
 | |
| 			t.Errorf("Expected RelaxedTimeout to be 10s, got %v", config.RelaxedTimeout)
 | |
| 		}
 | |
| 
 | |
| 		// Test configuration fields have proper defaults
 | |
| 		if config.MaxHandoffRetries != 3 {
 | |
| 			t.Errorf("Expected MaxHandoffRetries to be 3, got %d", config.MaxHandoffRetries)
 | |
| 		}
 | |
| 
 | |
| 		// Circuit breaker defaults
 | |
| 		if config.CircuitBreakerFailureThreshold != 5 {
 | |
| 			t.Errorf("Expected CircuitBreakerFailureThreshold=5, got %d", config.CircuitBreakerFailureThreshold)
 | |
| 		}
 | |
| 		if config.CircuitBreakerResetTimeout != 60*time.Second {
 | |
| 			t.Errorf("Expected CircuitBreakerResetTimeout=60s, got %v", config.CircuitBreakerResetTimeout)
 | |
| 		}
 | |
| 		if config.CircuitBreakerMaxRequests != 3 {
 | |
| 			t.Errorf("Expected CircuitBreakerMaxRequests=3, got %d", config.CircuitBreakerMaxRequests)
 | |
| 		}
 | |
| 
 | |
| 		if config.HandoffTimeout != 15*time.Second {
 | |
| 			t.Errorf("Expected HandoffTimeout to be 15s, got %v", config.HandoffTimeout)
 | |
| 		}
 | |
| 
 | |
| 		if config.PostHandoffRelaxedDuration != 0 {
 | |
| 			t.Errorf("Expected PostHandoffRelaxedDuration to be 0 (auto-calculated), got %v", config.PostHandoffRelaxedDuration)
 | |
| 		}
 | |
| 
 | |
| 		// Test that defaults are applied correctly
 | |
| 		configWithDefaults := config.ApplyDefaultsWithPoolSize(100)
 | |
| 		if configWithDefaults.PostHandoffRelaxedDuration != 20*time.Second {
 | |
| 			t.Errorf("Expected PostHandoffRelaxedDuration to be 20s (2x RelaxedTimeout) after applying defaults, got %v", configWithDefaults.PostHandoffRelaxedDuration)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ConfigValidation", func(t *testing.T) {
 | |
| 		// Valid config with applied defaults
 | |
| 		config := DefaultConfig().ApplyDefaults()
 | |
| 		if err := config.Validate(); err != nil {
 | |
| 			t.Errorf("Default config with applied defaults should be valid: %v", err)
 | |
| 		}
 | |
| 
 | |
| 		// Invalid worker configuration (negative MaxWorkers)
 | |
| 		config = &Config{
 | |
| 			RelaxedTimeout:             30 * time.Second,
 | |
| 			HandoffTimeout:             15 * time.Second,
 | |
| 			MaxWorkers:                 -1, // This should be invalid
 | |
| 			HandoffQueueSize:           100,
 | |
| 			PostHandoffRelaxedDuration: 10 * time.Second,
 | |
| 			MaxHandoffRetries:          3, // Add required field
 | |
| 		}
 | |
| 		if err := config.Validate(); err != ErrInvalidHandoffWorkers {
 | |
| 			t.Errorf("Expected ErrInvalidHandoffWorkers, got %v", err)
 | |
| 		}
 | |
| 
 | |
| 		// Invalid HandoffQueueSize
 | |
| 		config = DefaultConfig().ApplyDefaults()
 | |
| 		config.HandoffQueueSize = -1
 | |
| 		if err := config.Validate(); err != ErrInvalidHandoffQueueSize {
 | |
| 			t.Errorf("Expected ErrInvalidHandoffQueueSize, got %v", err)
 | |
| 		}
 | |
| 
 | |
| 		// Invalid PostHandoffRelaxedDuration
 | |
| 		config = DefaultConfig().ApplyDefaults()
 | |
| 		config.PostHandoffRelaxedDuration = -1 * time.Second
 | |
| 		if err := config.Validate(); err != ErrInvalidPostHandoffRelaxedDuration {
 | |
| 			t.Errorf("Expected ErrInvalidPostHandoffRelaxedDuration, got %v", err)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ConfigClone", func(t *testing.T) {
 | |
| 		original := DefaultConfig()
 | |
| 		original.MaxWorkers = 20
 | |
| 		original.HandoffQueueSize = 200
 | |
| 
 | |
| 		cloned := original.Clone()
 | |
| 
 | |
| 		if cloned.MaxWorkers != 20 {
 | |
| 			t.Errorf("Expected cloned MaxWorkers to be 20, got %d", cloned.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 		if cloned.HandoffQueueSize != 200 {
 | |
| 			t.Errorf("Expected cloned HandoffQueueSize to be 200, got %d", cloned.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		// Modify original to ensure clone is independent
 | |
| 		original.MaxWorkers = 2
 | |
| 		if cloned.MaxWorkers != 20 {
 | |
| 			t.Error("Clone should be independent of original")
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestApplyDefaults(t *testing.T) {
 | |
| 	t.Run("NilConfig", func(t *testing.T) {
 | |
| 		var config *Config
 | |
| 		result := config.ApplyDefaultsWithPoolSize(100) // Use explicit pool size for testing
 | |
| 
 | |
| 		// With nil config, should get default config with auto-calculated workers
 | |
| 		if result.MaxWorkers <= 0 {
 | |
| 			t.Errorf("Expected MaxWorkers to be > 0 after applying defaults, got %d", result.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 		// HandoffQueueSize should be auto-calculated with hybrid scaling
 | |
| 		workerBasedSize := result.MaxWorkers * 20
 | |
| 		poolSize := 100 // Default pool size used in ApplyDefaults
 | |
| 		poolBasedSize := poolSize
 | |
| 		expectedQueueSize := util.Max(workerBasedSize, poolBasedSize)
 | |
| 		expectedQueueSize = util.Min(expectedQueueSize, poolSize*5) // Cap by 5x pool size
 | |
| 		if result.HandoffQueueSize != expectedQueueSize {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be %d (max(20*MaxWorkers=%d, poolSize=%d) capped by 5*poolSize=%d), got %d",
 | |
| 				expectedQueueSize, workerBasedSize, poolBasedSize, poolSize*5, result.HandoffQueueSize)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("PartialConfig", func(t *testing.T) {
 | |
| 		config := &Config{
 | |
| 			MaxWorkers: 60, // Set this field explicitly (> poolSize/2 = 50)
 | |
| 			// Leave other fields as zero values
 | |
| 		}
 | |
| 
 | |
| 		result := config.ApplyDefaultsWithPoolSize(100) // Use explicit pool size for testing
 | |
| 
 | |
| 		// Should keep the explicitly set values when > poolSize/2
 | |
| 		if result.MaxWorkers != 60 {
 | |
| 			t.Errorf("Expected MaxWorkers to be 60 (explicitly set), got %d", result.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 		// Should apply default for unset fields (auto-calculated queue size with hybrid scaling)
 | |
| 		workerBasedSize := result.MaxWorkers * 20
 | |
| 		poolSize := 100 // Default pool size used in ApplyDefaults
 | |
| 		poolBasedSize := poolSize
 | |
| 		expectedQueueSize := util.Max(workerBasedSize, poolBasedSize)
 | |
| 		expectedQueueSize = util.Min(expectedQueueSize, poolSize*5) // Cap by 5x pool size
 | |
| 		if result.HandoffQueueSize != expectedQueueSize {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be %d (max(20*MaxWorkers=%d, poolSize=%d) capped by 5*poolSize=%d), got %d",
 | |
| 				expectedQueueSize, workerBasedSize, poolBasedSize, poolSize*5, result.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		// Test explicit queue size capping by 5x pool size
 | |
| 		configWithLargeQueue := &Config{
 | |
| 			MaxWorkers:       5,
 | |
| 			HandoffQueueSize: 1000, // Much larger than 5x pool size
 | |
| 		}
 | |
| 
 | |
| 		resultCapped := configWithLargeQueue.ApplyDefaultsWithPoolSize(20) // Small pool size
 | |
| 		expectedCap := 20 * 5                                              // 5x pool size = 100
 | |
| 		if resultCapped.HandoffQueueSize != expectedCap {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be capped by 5x pool size (%d), got %d", expectedCap, resultCapped.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		// Test explicit queue size minimum enforcement
 | |
| 		configWithSmallQueue := &Config{
 | |
| 			MaxWorkers:       5,
 | |
| 			HandoffQueueSize: 10, // Below minimum of 200
 | |
| 		}
 | |
| 
 | |
| 		resultMinimum := configWithSmallQueue.ApplyDefaultsWithPoolSize(100) // Large pool size
 | |
| 		if resultMinimum.HandoffQueueSize != 200 {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be enforced minimum (200), got %d", resultMinimum.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		// Test that large explicit values are capped by 5x pool size
 | |
| 		configWithVeryLargeQueue := &Config{
 | |
| 			MaxWorkers:       5,
 | |
| 			HandoffQueueSize: 1000, // Much larger than 5x pool size
 | |
| 		}
 | |
| 
 | |
| 		resultVeryLarge := configWithVeryLargeQueue.ApplyDefaultsWithPoolSize(100) // Pool size 100
 | |
| 		expectedVeryLargeCap := 100 * 5                                            // 5x pool size = 500
 | |
| 		if resultVeryLarge.HandoffQueueSize != expectedVeryLargeCap {
 | |
| 			t.Errorf("Expected very large HandoffQueueSize to be capped by 5x pool size (%d), got %d", expectedVeryLargeCap, resultVeryLarge.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		if result.RelaxedTimeout != 10*time.Second {
 | |
| 			t.Errorf("Expected RelaxedTimeout to be 10s (default), got %v", result.RelaxedTimeout)
 | |
| 		}
 | |
| 
 | |
| 		if result.HandoffTimeout != 15*time.Second {
 | |
| 			t.Errorf("Expected HandoffTimeout to be 15s (default), got %v", result.HandoffTimeout)
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ZeroValues", func(t *testing.T) {
 | |
| 		config := &Config{
 | |
| 			MaxWorkers:       0, // Zero value should get auto-calculated defaults
 | |
| 			HandoffQueueSize: 0, // Zero value should get default
 | |
| 			RelaxedTimeout:   0, // Zero value should get default
 | |
| 		}
 | |
| 
 | |
| 		result := config.ApplyDefaultsWithPoolSize(100) // Use explicit pool size for testing
 | |
| 
 | |
| 		// Zero values should get auto-calculated defaults
 | |
| 		if result.MaxWorkers <= 0 {
 | |
| 			t.Errorf("Expected MaxWorkers to be > 0 (auto-calculated), got %d", result.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 		// HandoffQueueSize should be auto-calculated with hybrid scaling
 | |
| 		workerBasedSize := result.MaxWorkers * 20
 | |
| 		poolSize := 100 // Default pool size used in ApplyDefaults
 | |
| 		poolBasedSize := poolSize
 | |
| 		expectedQueueSize := util.Max(workerBasedSize, poolBasedSize)
 | |
| 		expectedQueueSize = util.Min(expectedQueueSize, poolSize*5) // Cap by 5x pool size
 | |
| 		if result.HandoffQueueSize != expectedQueueSize {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be %d (max(20*MaxWorkers=%d, poolSize=%d) capped by 5*poolSize=%d), got %d",
 | |
| 				expectedQueueSize, workerBasedSize, poolBasedSize, poolSize*5, result.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		if result.RelaxedTimeout != 10*time.Second {
 | |
| 			t.Errorf("Expected RelaxedTimeout to be 10s (default), got %v", result.RelaxedTimeout)
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestProcessorWithConfig(t *testing.T) {
 | |
| 	t.Run("ProcessorUsesConfigValues", func(t *testing.T) {
 | |
| 		config := &Config{
 | |
| 			MaxWorkers:       5,
 | |
| 			HandoffQueueSize: 50,
 | |
| 			RelaxedTimeout:   10 * time.Second,
 | |
| 			HandoffTimeout:   5 * time.Second,
 | |
| 		}
 | |
| 
 | |
| 		baseDialer := func(ctx context.Context, network, addr string) (net.Conn, error) {
 | |
| 			return &mockNetConn{addr: addr}, nil
 | |
| 		}
 | |
| 
 | |
| 		processor := NewPoolHook(baseDialer, "tcp", config, nil)
 | |
| 		defer processor.Shutdown(context.Background())
 | |
| 
 | |
| 		// The processor should be created successfully with custom config
 | |
| 		if processor == nil {
 | |
| 			t.Error("Processor should be created with custom config")
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ProcessorWithPartialConfig", func(t *testing.T) {
 | |
| 		config := &Config{
 | |
| 			MaxWorkers: 7, // Only set worker field
 | |
| 			// Other fields will get defaults
 | |
| 		}
 | |
| 
 | |
| 		baseDialer := func(ctx context.Context, network, addr string) (net.Conn, error) {
 | |
| 			return &mockNetConn{addr: addr}, nil
 | |
| 		}
 | |
| 
 | |
| 		processor := NewPoolHook(baseDialer, "tcp", config, nil)
 | |
| 		defer processor.Shutdown(context.Background())
 | |
| 
 | |
| 		// Should work with partial config (defaults applied)
 | |
| 		if processor == nil {
 | |
| 			t.Error("Processor should be created with partial config")
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ProcessorWithNilConfig", func(t *testing.T) {
 | |
| 		baseDialer := func(ctx context.Context, network, addr string) (net.Conn, error) {
 | |
| 			return &mockNetConn{addr: addr}, nil
 | |
| 		}
 | |
| 
 | |
| 		processor := NewPoolHook(baseDialer, "tcp", nil, nil)
 | |
| 		defer processor.Shutdown(context.Background())
 | |
| 
 | |
| 		// Should use default config when nil is passed
 | |
| 		if processor == nil {
 | |
| 			t.Error("Processor should be created with nil config (using defaults)")
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestIntegrationWithApplyDefaults(t *testing.T) {
 | |
| 	t.Run("ProcessorWithPartialConfigAppliesDefaults", func(t *testing.T) {
 | |
| 		// Create a partial config with only some fields set
 | |
| 		partialConfig := &Config{
 | |
| 			MaxWorkers: 15, // Custom value (>= 10 to test preservation)
 | |
| 			// Other fields left as zero values - should get defaults
 | |
| 		}
 | |
| 
 | |
| 		baseDialer := func(ctx context.Context, network, addr string) (net.Conn, error) {
 | |
| 			return &mockNetConn{addr: addr}, nil
 | |
| 		}
 | |
| 
 | |
| 		// Create processor - should apply defaults to missing fields
 | |
| 		processor := NewPoolHook(baseDialer, "tcp", partialConfig, nil)
 | |
| 		defer processor.Shutdown(context.Background())
 | |
| 
 | |
| 		// Processor should be created successfully
 | |
| 		if processor == nil {
 | |
| 			t.Error("Processor should be created with partial config")
 | |
| 		}
 | |
| 
 | |
| 		// Test that the ApplyDefaults method worked correctly by creating the same config
 | |
| 		// and applying defaults manually
 | |
| 		expectedConfig := partialConfig.ApplyDefaultsWithPoolSize(100) // Use explicit pool size for testing
 | |
| 
 | |
| 		// Should preserve custom values (when >= poolSize/2)
 | |
| 		if expectedConfig.MaxWorkers != 50 { // max(poolSize/2, 15) = max(50, 15) = 50
 | |
| 			t.Errorf("Expected MaxWorkers to be 50, got %d", expectedConfig.MaxWorkers)
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 
 | |
| 		// Should apply defaults for missing fields (auto-calculated queue size with hybrid scaling)
 | |
| 		workerBasedSize := expectedConfig.MaxWorkers * 20
 | |
| 		poolSize := 100 // Default pool size used in ApplyDefaults
 | |
| 		poolBasedSize := poolSize
 | |
| 		expectedQueueSize := util.Max(workerBasedSize, poolBasedSize)
 | |
| 		expectedQueueSize = util.Min(expectedQueueSize, poolSize*5) // Cap by 5x pool size
 | |
| 		if expectedConfig.HandoffQueueSize != expectedQueueSize {
 | |
| 			t.Errorf("Expected HandoffQueueSize to be %d (max(20*MaxWorkers=%d, poolSize=%d) capped by 5*poolSize=%d), got %d",
 | |
| 				expectedQueueSize, workerBasedSize, poolBasedSize, poolSize*5, expectedConfig.HandoffQueueSize)
 | |
| 		}
 | |
| 
 | |
| 		// Test that queue size is always capped by 5x pool size
 | |
| 		if expectedConfig.HandoffQueueSize > poolSize*5 {
 | |
| 			t.Errorf("HandoffQueueSize (%d) should never exceed 5x pool size (%d)",
 | |
| 				expectedConfig.HandoffQueueSize, poolSize*2)
 | |
| 		}
 | |
| 
 | |
| 		if expectedConfig.RelaxedTimeout != 10*time.Second {
 | |
| 			t.Errorf("Expected RelaxedTimeout to be 10s (default), got %v", expectedConfig.RelaxedTimeout)
 | |
| 		}
 | |
| 
 | |
| 		if expectedConfig.HandoffTimeout != 15*time.Second {
 | |
| 			t.Errorf("Expected HandoffTimeout to be 15s (default), got %v", expectedConfig.HandoffTimeout)
 | |
| 		}
 | |
| 
 | |
| 		if expectedConfig.PostHandoffRelaxedDuration != 20*time.Second {
 | |
| 			t.Errorf("Expected PostHandoffRelaxedDuration to be 20s (2x RelaxedTimeout), got %v", expectedConfig.PostHandoffRelaxedDuration)
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestEnhancedConfigValidation(t *testing.T) {
 | |
| 	t.Run("ValidateFields", func(t *testing.T) {
 | |
| 		config := DefaultConfig()
 | |
| 		config.ApplyDefaultsWithPoolSize(100) // Apply defaults with pool size 100
 | |
| 
 | |
| 		// Should pass validation with default values
 | |
| 		if err := config.Validate(); err != nil {
 | |
| 			t.Errorf("Default config should be valid, got error: %v", err)
 | |
| 		}
 | |
| 
 | |
| 		// Test invalid MaxHandoffRetries
 | |
| 		config.MaxHandoffRetries = 0
 | |
| 		if err := config.Validate(); err == nil {
 | |
| 			t.Error("Expected validation error for MaxHandoffRetries = 0")
 | |
| 		}
 | |
| 		config.MaxHandoffRetries = 11
 | |
| 		if err := config.Validate(); err == nil {
 | |
| 			t.Error("Expected validation error for MaxHandoffRetries = 11")
 | |
| 		}
 | |
| 		config.MaxHandoffRetries = 3 // Reset to valid value
 | |
| 
 | |
| 		// Test circuit breaker validation
 | |
| 		config.CircuitBreakerFailureThreshold = 0
 | |
| 		if err := config.Validate(); err != ErrInvalidCircuitBreakerFailureThreshold {
 | |
| 			t.Errorf("Expected ErrInvalidCircuitBreakerFailureThreshold, got %v", err)
 | |
| 		}
 | |
| 		config.CircuitBreakerFailureThreshold = 5 // Reset to valid value
 | |
| 
 | |
| 		config.CircuitBreakerResetTimeout = -1 * time.Second
 | |
| 		if err := config.Validate(); err != ErrInvalidCircuitBreakerResetTimeout {
 | |
| 			t.Errorf("Expected ErrInvalidCircuitBreakerResetTimeout, got %v", err)
 | |
| 		}
 | |
| 		config.CircuitBreakerResetTimeout = 60 * time.Second // Reset to valid value
 | |
| 
 | |
| 		config.CircuitBreakerMaxRequests = 0
 | |
| 		if err := config.Validate(); err != ErrInvalidCircuitBreakerMaxRequests {
 | |
| 			t.Errorf("Expected ErrInvalidCircuitBreakerMaxRequests, got %v", err)
 | |
| 		}
 | |
| 		config.CircuitBreakerMaxRequests = 3 // Reset to valid value
 | |
| 
 | |
| 		// Should pass validation again
 | |
| 		if err := config.Validate(); err != nil {
 | |
| 			t.Errorf("Config should be valid after reset, got error: %v", err)
 | |
| 		}
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func TestConfigClone(t *testing.T) {
 | |
| 	original := DefaultConfig()
 | |
| 	original.MaxHandoffRetries = 7
 | |
| 	original.HandoffTimeout = 8 * time.Second
 | |
| 
 | |
| 	cloned := original.Clone()
 | |
| 
 | |
| 	// Test that values are copied
 | |
| 	if cloned.MaxHandoffRetries != 7 {
 | |
| 		t.Errorf("Expected cloned MaxHandoffRetries to be 7, got %d", cloned.MaxHandoffRetries)
 | |
| 	}
 | |
| 	if cloned.HandoffTimeout != 8*time.Second {
 | |
| 		t.Errorf("Expected cloned HandoffTimeout to be 8s, got %v", cloned.HandoffTimeout)
 | |
| 	}
 | |
| 
 | |
| 	// Test that modifying clone doesn't affect original
 | |
| 	cloned.MaxHandoffRetries = 10
 | |
| 	if original.MaxHandoffRetries != 7 {
 | |
| 		t.Errorf("Modifying clone should not affect original, original MaxHandoffRetries changed to %d", original.MaxHandoffRetries)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestMaxWorkersLogic(t *testing.T) {
 | |
| 	t.Run("AutoCalculatedMaxWorkers", func(t *testing.T) {
 | |
| 		testCases := []struct {
 | |
| 			poolSize        int
 | |
| 			expectedWorkers int
 | |
| 			description     string
 | |
| 		}{
 | |
| 			{6, 3, "Small pool: min(6/2, max(10, 6/3)) = min(3, max(10, 2)) = min(3, 10) = 3"},
 | |
| 			{15, 7, "Medium pool: min(15/2, max(10, 15/3)) = min(7, max(10, 5)) = min(7, 10) = 7"},
 | |
| 			{30, 10, "Large pool: min(30/2, max(10, 30/3)) = min(15, max(10, 10)) = min(15, 10) = 10"},
 | |
| 			{60, 20, "Very large pool: min(60/2, max(10, 60/3)) = min(30, max(10, 20)) = min(30, 20) = 20"},
 | |
| 			{120, 40, "Huge pool: min(120/2, max(10, 120/3)) = min(60, max(10, 40)) = min(60, 40) = 40"},
 | |
| 		}
 | |
| 
 | |
| 		for _, tc := range testCases {
 | |
| 			config := &Config{} // MaxWorkers = 0 (not set)
 | |
| 			result := config.ApplyDefaultsWithPoolSize(tc.poolSize)
 | |
| 
 | |
| 			if result.MaxWorkers != tc.expectedWorkers {
 | |
| 				t.Errorf("PoolSize=%d: expected MaxWorkers=%d, got %d (%s)",
 | |
| 					tc.poolSize, tc.expectedWorkers, result.MaxWorkers, tc.description)
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| 
 | |
| 	t.Run("ExplicitlySetMaxWorkers", func(t *testing.T) {
 | |
| 		testCases := []struct {
 | |
| 			setValue        int
 | |
| 			expectedWorkers int
 | |
| 			description     string
 | |
| 		}{
 | |
| 			{1, 50, "Set 1: max(poolSize/2, 1) = max(50, 1) = 50 (enforced minimum)"},
 | |
| 			{5, 50, "Set 5: max(poolSize/2, 5) = max(50, 5) = 50 (enforced minimum)"},
 | |
| 			{8, 50, "Set 8: max(poolSize/2, 8) = max(50, 8) = 50 (enforced minimum)"},
 | |
| 			{10, 50, "Set 10: max(poolSize/2, 10) = max(50, 10) = 50 (enforced minimum)"},
 | |
| 			{15, 50, "Set 15: max(poolSize/2, 15) = max(50, 15) = 50 (enforced minimum)"},
 | |
| 			{60, 60, "Set 60: max(poolSize/2, 60) = max(50, 60) = 60 (respects user choice)"},
 | |
| 		}
 | |
| 
 | |
| 		for _, tc := range testCases {
 | |
| 			config := &Config{
 | |
| 				MaxWorkers: tc.setValue, // Explicitly set
 | |
| 			}
 | |
| 			result := config.ApplyDefaultsWithPoolSize(100) // Pool size doesn't affect explicit values
 | |
| 
 | |
| 			if result.MaxWorkers != tc.expectedWorkers {
 | |
| 				t.Errorf("Set MaxWorkers=%d: expected %d, got %d (%s)",
 | |
| 					tc.setValue, tc.expectedWorkers, result.MaxWorkers, tc.description)
 | |
| 			}
 | |
| 		}
 | |
| 	})
 | |
| }
 |