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")
|
|
}
|