You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							155 lines
						
					
					
						
							5.3 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							155 lines
						
					
					
						
							5.3 KiB
						
					
					
				
								package integration
							 | 
						|
								
							 | 
						|
								import (
							 | 
						|
									"testing"
							 | 
						|
									"time"
							 | 
						|
								)
							 | 
						|
								
							 | 
						|
								// TestAdaptiveFetchTimeout verifies that the adaptive timeout strategy
							 | 
						|
								// allows reading multiple records from disk within a reasonable time
							 | 
						|
								func TestAdaptiveFetchTimeout(t *testing.T) {
							 | 
						|
									t.Log("Testing adaptive fetch timeout strategy...")
							 | 
						|
								
							 | 
						|
									// Simulate the scenario where we need to read 4 records from disk
							 | 
						|
									// Each record takes 100-200ms to read (simulates disk I/O)
							 | 
						|
									recordReadTimes := []time.Duration{
							 | 
						|
										150 * time.Millisecond, // Record 1 (from disk)
							 | 
						|
										150 * time.Millisecond, // Record 2 (from disk)
							 | 
						|
										150 * time.Millisecond, // Record 3 (from disk)
							 | 
						|
										150 * time.Millisecond, // Record 4 (from disk)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Test 1: Old strategy (50ms timeout per record)
							 | 
						|
									t.Run("OldStrategy_50ms_Timeout", func(t *testing.T) {
							 | 
						|
										timeout := 50 * time.Millisecond
							 | 
						|
										recordsReceived := 0
							 | 
						|
								
							 | 
						|
										start := time.Now()
							 | 
						|
										for i, readTime := range recordReadTimes {
							 | 
						|
											if readTime <= timeout {
							 | 
						|
												recordsReceived++
							 | 
						|
											} else {
							 | 
						|
												t.Logf("Record %d timed out (readTime=%v > timeout=%v)", i+1, readTime, timeout)
							 | 
						|
												break
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
										duration := time.Since(start)
							 | 
						|
								
							 | 
						|
										t.Logf("Old strategy: received %d/%d records in %v", recordsReceived, len(recordReadTimes), duration)
							 | 
						|
								
							 | 
						|
										if recordsReceived >= len(recordReadTimes) {
							 | 
						|
											t.Error("Old strategy should NOT receive all records (timeout too short)")
							 | 
						|
										} else {
							 | 
						|
											t.Logf("✓ Bug reproduced: old strategy times out too quickly")
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test 2: New adaptive strategy (1 second timeout for first 5 records)
							 | 
						|
									t.Run("NewStrategy_1s_Timeout", func(t *testing.T) {
							 | 
						|
										timeout := 1 * time.Second // Generous timeout for first batch
							 | 
						|
										recordsReceived := 0
							 | 
						|
								
							 | 
						|
										start := time.Now()
							 | 
						|
										for i, readTime := range recordReadTimes {
							 | 
						|
											if readTime <= timeout {
							 | 
						|
												recordsReceived++
							 | 
						|
												t.Logf("Record %d received (readTime=%v)", i+1, readTime)
							 | 
						|
											} else {
							 | 
						|
												t.Logf("Record %d timed out (readTime=%v > timeout=%v)", i+1, readTime, timeout)
							 | 
						|
												break
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
										duration := time.Since(start)
							 | 
						|
								
							 | 
						|
										t.Logf("New strategy: received %d/%d records in %v", recordsReceived, len(recordReadTimes), duration)
							 | 
						|
								
							 | 
						|
										if recordsReceived < len(recordReadTimes) {
							 | 
						|
											t.Errorf("New strategy should receive all records (timeout=%v)", timeout)
							 | 
						|
										} else {
							 | 
						|
											t.Logf("✓ Fix verified: new strategy receives all records")
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test 3: Schema Registry catch-up scenario
							 | 
						|
									t.Run("SchemaRegistry_CatchUp_Scenario", func(t *testing.T) {
							 | 
						|
										// Schema Registry has 500ms total timeout to catch up from offset 3 to 6
							 | 
						|
										schemaRegistryTimeout := 500 * time.Millisecond
							 | 
						|
								
							 | 
						|
										// With old strategy (50ms per record after first):
							 | 
						|
										// - First record: 10s timeout ✓
							 | 
						|
										// - Records 2-4: 50ms each ✗ (times out after record 1)
							 | 
						|
										// Total time: > 500ms (only gets 1 record per fetch)
							 | 
						|
								
							 | 
						|
										// With new strategy (1s per record for first 5):
							 | 
						|
										// - Records 1-4: 1s each ✓
							 | 
						|
										// - All 4 records received in ~600ms
							 | 
						|
										// Total time: ~600ms (gets all 4 records in one fetch)
							 | 
						|
								
							 | 
						|
										recordsNeeded := 4
							 | 
						|
										perRecordReadTime := 150 * time.Millisecond
							 | 
						|
								
							 | 
						|
										// Old strategy simulation
							 | 
						|
										oldStrategyTime := time.Duration(recordsNeeded) * 50 * time.Millisecond // Times out, need multiple fetches
							 | 
						|
										oldStrategyRoundTrips := recordsNeeded                                  // One record per fetch
							 | 
						|
								
							 | 
						|
										// New strategy simulation
							 | 
						|
										newStrategyTime := time.Duration(recordsNeeded) * perRecordReadTime // All in one fetch
							 | 
						|
										newStrategyRoundTrips := 1
							 | 
						|
								
							 | 
						|
										t.Logf("Schema Registry catch-up simulation:")
							 | 
						|
										t.Logf("  Old strategy: %d round trips, ~%v total time", oldStrategyRoundTrips, oldStrategyTime*time.Duration(oldStrategyRoundTrips))
							 | 
						|
										t.Logf("  New strategy: %d round trip, ~%v total time", newStrategyRoundTrips, newStrategyTime)
							 | 
						|
										t.Logf("  Schema Registry timeout: %v", schemaRegistryTimeout)
							 | 
						|
								
							 | 
						|
										oldStrategyTotalTime := oldStrategyTime * time.Duration(oldStrategyRoundTrips)
							 | 
						|
										newStrategyTotalTime := newStrategyTime * time.Duration(newStrategyRoundTrips)
							 | 
						|
								
							 | 
						|
										if oldStrategyTotalTime > schemaRegistryTimeout {
							 | 
						|
											t.Logf("✓ Old strategy exceeds timeout: %v > %v", oldStrategyTotalTime, schemaRegistryTimeout)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										if newStrategyTotalTime <= schemaRegistryTimeout+200*time.Millisecond {
							 | 
						|
											t.Logf("✓ New strategy completes within timeout: %v <= %v", newStrategyTotalTime, schemaRegistryTimeout+200*time.Millisecond)
							 | 
						|
										} else {
							 | 
						|
											t.Errorf("New strategy too slow: %v > %v", newStrategyTotalTime, schemaRegistryTimeout)
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestFetchTimeoutProgression verifies the timeout progression logic
							 | 
						|
								func TestFetchTimeoutProgression(t *testing.T) {
							 | 
						|
									t.Log("Testing fetch timeout progression...")
							 | 
						|
								
							 | 
						|
									// Adaptive timeout logic:
							 | 
						|
									// - First 5 records: 1 second (catch-up from disk)
							 | 
						|
									// - After 5 records: 100ms (streaming from memory)
							 | 
						|
								
							 | 
						|
									getTimeout := func(recordNumber int) time.Duration {
							 | 
						|
										if recordNumber <= 5 {
							 | 
						|
											return 1 * time.Second
							 | 
						|
										}
							 | 
						|
										return 100 * time.Millisecond
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									t.Logf("Timeout progression:")
							 | 
						|
									for i := 1; i <= 10; i++ {
							 | 
						|
										timeout := getTimeout(i)
							 | 
						|
										t.Logf("  Record %2d: timeout = %v", i, timeout)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Verify the progression
							 | 
						|
									if getTimeout(1) != 1*time.Second {
							 | 
						|
										t.Error("First record should have 1s timeout")
							 | 
						|
									}
							 | 
						|
									if getTimeout(5) != 1*time.Second {
							 | 
						|
										t.Error("Fifth record should have 1s timeout")
							 | 
						|
									}
							 | 
						|
									if getTimeout(6) != 100*time.Millisecond {
							 | 
						|
										t.Error("Sixth record should have 100ms timeout (fast path)")
							 | 
						|
									}
							 | 
						|
									if getTimeout(10) != 100*time.Millisecond {
							 | 
						|
										t.Error("Tenth record should have 100ms timeout (fast path)")
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									t.Log("✓ Timeout progression is correct")
							 | 
						|
								}
							 |