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.
		
		
		
		
		
			
		
			
				
					
					
						
							566 lines
						
					
					
						
							17 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							566 lines
						
					
					
						
							17 KiB
						
					
					
				
								package topic
							 | 
						|
								
							 | 
						|
								import (
							 | 
						|
									"fmt"
							 | 
						|
									"sync"
							 | 
						|
									"testing"
							 | 
						|
									"time"
							 | 
						|
								
							 | 
						|
									"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
							 | 
						|
									"github.com/seaweedfs/seaweedfs/weed/util/log_buffer"
							 | 
						|
								)
							 | 
						|
								
							 | 
						|
								// MockLogBuffer provides a controllable log buffer for testing
							 | 
						|
								type MockLogBuffer struct {
							 | 
						|
									// In-memory data
							 | 
						|
									memoryEntries     []*filer_pb.LogEntry
							 | 
						|
									memoryStartTime   time.Time
							 | 
						|
									memoryStopTime    time.Time
							 | 
						|
									memoryStartOffset int64
							 | 
						|
									memoryStopOffset  int64
							 | 
						|
								
							 | 
						|
									// Disk data
							 | 
						|
									diskEntries     []*filer_pb.LogEntry
							 | 
						|
									diskStartTime   time.Time
							 | 
						|
									diskStopTime    time.Time
							 | 
						|
									diskStartOffset int64
							 | 
						|
									diskStopOffset  int64
							 | 
						|
								
							 | 
						|
									// Behavior control
							 | 
						|
									diskReadDelay   time.Duration
							 | 
						|
									memoryReadDelay time.Duration
							 | 
						|
									diskReadError   error
							 | 
						|
									memoryReadError error
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// MockReadFromDiskFn simulates reading from disk
							 | 
						|
								func (m *MockLogBuffer) MockReadFromDiskFn(startPosition log_buffer.MessagePosition, stopTsNs int64, eachLogEntryFn log_buffer.EachLogEntryFuncType) (log_buffer.MessagePosition, bool, error) {
							 | 
						|
									if m.diskReadDelay > 0 {
							 | 
						|
										time.Sleep(m.diskReadDelay)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									if m.diskReadError != nil {
							 | 
						|
										return startPosition, false, m.diskReadError
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									isOffsetBased := startPosition.IsOffsetBased
							 | 
						|
									lastPosition := startPosition
							 | 
						|
									isDone := false
							 | 
						|
								
							 | 
						|
									for _, entry := range m.diskEntries {
							 | 
						|
										// Filter based on mode
							 | 
						|
										if isOffsetBased {
							 | 
						|
											if entry.Offset < startPosition.Offset {
							 | 
						|
												continue
							 | 
						|
											}
							 | 
						|
										} else {
							 | 
						|
											entryTime := time.Unix(0, entry.TsNs)
							 | 
						|
											if entryTime.Before(startPosition.Time) {
							 | 
						|
												continue
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Apply stopTsNs filter
							 | 
						|
										if stopTsNs > 0 && entry.TsNs > stopTsNs {
							 | 
						|
											isDone = true
							 | 
						|
											break
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Call handler
							 | 
						|
										done, err := eachLogEntryFn(entry)
							 | 
						|
										if err != nil {
							 | 
						|
											return lastPosition, false, err
							 | 
						|
										}
							 | 
						|
										if done {
							 | 
						|
											isDone = true
							 | 
						|
											break
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Update position
							 | 
						|
										if isOffsetBased {
							 | 
						|
											lastPosition = log_buffer.NewMessagePosition(entry.TsNs, entry.Offset+1)
							 | 
						|
										} else {
							 | 
						|
											lastPosition = log_buffer.NewMessagePosition(entry.TsNs, entry.Offset)
							 | 
						|
										}
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									return lastPosition, isDone, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// MockLoopProcessLogDataWithOffset simulates reading from memory with offset
							 | 
						|
								func (m *MockLogBuffer) MockLoopProcessLogDataWithOffset(readerName string, startPosition log_buffer.MessagePosition, stopTsNs int64, waitForDataFn func() bool, eachLogDataFn log_buffer.EachLogEntryWithOffsetFuncType) (log_buffer.MessagePosition, bool, error) {
							 | 
						|
									if m.memoryReadDelay > 0 {
							 | 
						|
										time.Sleep(m.memoryReadDelay)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									if m.memoryReadError != nil {
							 | 
						|
										return startPosition, false, m.memoryReadError
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									lastPosition := startPosition
							 | 
						|
									isDone := false
							 | 
						|
								
							 | 
						|
									// Check if requested offset is in memory
							 | 
						|
									if startPosition.Offset < m.memoryStartOffset {
							 | 
						|
										// Data is on disk
							 | 
						|
										return startPosition, false, log_buffer.ResumeFromDiskError
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									for _, entry := range m.memoryEntries {
							 | 
						|
										// Filter by offset
							 | 
						|
										if entry.Offset < startPosition.Offset {
							 | 
						|
											continue
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Apply stopTsNs filter
							 | 
						|
										if stopTsNs > 0 && entry.TsNs > stopTsNs {
							 | 
						|
											isDone = true
							 | 
						|
											break
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Call handler
							 | 
						|
										done, err := eachLogDataFn(entry, entry.Offset)
							 | 
						|
										if err != nil {
							 | 
						|
											return lastPosition, false, err
							 | 
						|
										}
							 | 
						|
										if done {
							 | 
						|
											isDone = true
							 | 
						|
											break
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Update position
							 | 
						|
										lastPosition = log_buffer.NewMessagePosition(entry.TsNs, entry.Offset+1)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									return lastPosition, isDone, nil
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// Helper to create test entries
							 | 
						|
								func createTestEntry(offset int64, timestamp time.Time, key, value string) *filer_pb.LogEntry {
							 | 
						|
									return &filer_pb.LogEntry{
							 | 
						|
										TsNs:   timestamp.UnixNano(),
							 | 
						|
										Offset: offset,
							 | 
						|
										Key:    []byte(key),
							 | 
						|
										Data:   []byte(value),
							 | 
						|
									}
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestOffsetBasedSubscribe_AllDataInMemory tests reading when all data is in memory
							 | 
						|
								func TestOffsetBasedSubscribe_AllDataInMemory(t *testing.T) {
							 | 
						|
									baseTime := time.Now()
							 | 
						|
								
							 | 
						|
									mock := &MockLogBuffer{
							 | 
						|
										memoryEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(0, baseTime, "key0", "value0"),
							 | 
						|
											createTestEntry(1, baseTime.Add(1*time.Second), "key1", "value1"),
							 | 
						|
											createTestEntry(2, baseTime.Add(2*time.Second), "key2", "value2"),
							 | 
						|
											createTestEntry(3, baseTime.Add(3*time.Second), "key3", "value3"),
							 | 
						|
										},
							 | 
						|
										memoryStartOffset: 0,
							 | 
						|
										memoryStopOffset:  3,
							 | 
						|
										diskEntries:       []*filer_pb.LogEntry{}, // No disk data
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Test reading from offset 0
							 | 
						|
									t.Run("ReadFromOffset0", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePositionFromOffset(0)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Simulate the Subscribe logic
							 | 
						|
										// 1. Try disk read first
							 | 
						|
										pos, done, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
										if done {
							 | 
						|
											t.Fatal("Should not be done after disk read")
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 2. Read from memory
							 | 
						|
										eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
											return eachLogFn(entry)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										_, _, err = mock.MockLoopProcessLogDataWithOffset("test", pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
										if err != nil && err != log_buffer.ResumeFromDiskError {
							 | 
						|
											t.Fatalf("Memory read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Verify we got all offsets in order
							 | 
						|
										expected := []int64{0, 1, 2, 3}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d", len(expected), len(receivedOffsets))
							 | 
						|
										}
							 | 
						|
										for i, offset := range receivedOffsets {
							 | 
						|
											if offset != expected[i] {
							 | 
						|
												t.Errorf("Offset[%d]: expected %d, got %d", i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test reading from offset 2
							 | 
						|
									t.Run("ReadFromOffset2", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePositionFromOffset(2)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
											return eachLogFn(entry)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Should skip disk and go straight to memory
							 | 
						|
										pos, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										_, _, err = mock.MockLoopProcessLogDataWithOffset("test", pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
										if err != nil && err != log_buffer.ResumeFromDiskError {
							 | 
						|
											t.Fatalf("Memory read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Verify we got offsets 2, 3
							 | 
						|
										expected := []int64{2, 3}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d", len(expected), len(receivedOffsets))
							 | 
						|
										}
							 | 
						|
										for i, offset := range receivedOffsets {
							 | 
						|
											if offset != expected[i] {
							 | 
						|
												t.Errorf("Offset[%d]: expected %d, got %d", i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestOffsetBasedSubscribe_DataOnDisk tests reading when data is on disk
							 | 
						|
								func TestOffsetBasedSubscribe_DataOnDisk(t *testing.T) {
							 | 
						|
									baseTime := time.Now()
							 | 
						|
								
							 | 
						|
									mock := &MockLogBuffer{
							 | 
						|
										// Offsets 0-9 on disk
							 | 
						|
										diskEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(0, baseTime, "key0", "value0"),
							 | 
						|
											createTestEntry(1, baseTime.Add(1*time.Second), "key1", "value1"),
							 | 
						|
											createTestEntry(2, baseTime.Add(2*time.Second), "key2", "value2"),
							 | 
						|
											createTestEntry(3, baseTime.Add(3*time.Second), "key3", "value3"),
							 | 
						|
											createTestEntry(4, baseTime.Add(4*time.Second), "key4", "value4"),
							 | 
						|
											createTestEntry(5, baseTime.Add(5*time.Second), "key5", "value5"),
							 | 
						|
											createTestEntry(6, baseTime.Add(6*time.Second), "key6", "value6"),
							 | 
						|
											createTestEntry(7, baseTime.Add(7*time.Second), "key7", "value7"),
							 | 
						|
											createTestEntry(8, baseTime.Add(8*time.Second), "key8", "value8"),
							 | 
						|
											createTestEntry(9, baseTime.Add(9*time.Second), "key9", "value9"),
							 | 
						|
										},
							 | 
						|
										diskStartOffset: 0,
							 | 
						|
										diskStopOffset:  9,
							 | 
						|
										// Offsets 10-12 in memory
							 | 
						|
										memoryEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(10, baseTime.Add(10*time.Second), "key10", "value10"),
							 | 
						|
											createTestEntry(11, baseTime.Add(11*time.Second), "key11", "value11"),
							 | 
						|
											createTestEntry(12, baseTime.Add(12*time.Second), "key12", "value12"),
							 | 
						|
										},
							 | 
						|
										memoryStartOffset: 10,
							 | 
						|
										memoryStopOffset:  12,
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Test reading from offset 0 (on disk)
							 | 
						|
									t.Run("ReadFromOffset0_OnDisk", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePositionFromOffset(0)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
											return eachLogFn(entry)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 1. Read from disk (should get 0-9)
							 | 
						|
										pos, done, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
										if done {
							 | 
						|
											t.Fatal("Should not be done after disk read")
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 2. Read from memory (should get 10-12)
							 | 
						|
										_, _, err = mock.MockLoopProcessLogDataWithOffset("test", pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
										if err != nil && err != log_buffer.ResumeFromDiskError {
							 | 
						|
											t.Fatalf("Memory read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Verify we got all offsets 0-12 in order
							 | 
						|
										expected := []int64{0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d: %v", len(expected), len(receivedOffsets), receivedOffsets)
							 | 
						|
										}
							 | 
						|
										for i, offset := range receivedOffsets {
							 | 
						|
											if i < len(expected) && offset != expected[i] {
							 | 
						|
												t.Errorf("Offset[%d]: expected %d, got %d", i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test reading from offset 5 (on disk, middle)
							 | 
						|
									t.Run("ReadFromOffset5_OnDisk", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePositionFromOffset(5)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
											return eachLogFn(entry)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 1. Read from disk (should get 5-9)
							 | 
						|
										pos, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 2. Read from memory (should get 10-12)
							 | 
						|
										_, _, err = mock.MockLoopProcessLogDataWithOffset("test", pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
										if err != nil && err != log_buffer.ResumeFromDiskError {
							 | 
						|
											t.Fatalf("Memory read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Verify we got offsets 5-12
							 | 
						|
										expected := []int64{5, 6, 7, 8, 9, 10, 11, 12}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d: %v", len(expected), len(receivedOffsets), receivedOffsets)
							 | 
						|
										}
							 | 
						|
										for i, offset := range receivedOffsets {
							 | 
						|
											if i < len(expected) && offset != expected[i] {
							 | 
						|
												t.Errorf("Offset[%d]: expected %d, got %d", i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test reading from offset 11 (in memory)
							 | 
						|
									t.Run("ReadFromOffset11_InMemory", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePositionFromOffset(11)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
											return eachLogFn(entry)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 1. Try disk read (should get nothing)
							 | 
						|
										pos, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// 2. Read from memory (should get 11-12)
							 | 
						|
										_, _, err = mock.MockLoopProcessLogDataWithOffset("test", pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
										if err != nil && err != log_buffer.ResumeFromDiskError {
							 | 
						|
											t.Fatalf("Memory read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Verify we got offsets 11-12
							 | 
						|
										expected := []int64{11, 12}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d: %v", len(expected), len(receivedOffsets), receivedOffsets)
							 | 
						|
										}
							 | 
						|
										for i, offset := range receivedOffsets {
							 | 
						|
											if i < len(expected) && offset != expected[i] {
							 | 
						|
												t.Errorf("Offset[%d]: expected %d, got %d", i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestTimestampBasedSubscribe tests timestamp-based reading
							 | 
						|
								func TestTimestampBasedSubscribe(t *testing.T) {
							 | 
						|
									baseTime := time.Now()
							 | 
						|
								
							 | 
						|
									mock := &MockLogBuffer{
							 | 
						|
										diskEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(0, baseTime, "key0", "value0"),
							 | 
						|
											createTestEntry(1, baseTime.Add(10*time.Second), "key1", "value1"),
							 | 
						|
											createTestEntry(2, baseTime.Add(20*time.Second), "key2", "value2"),
							 | 
						|
										},
							 | 
						|
										memoryEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(3, baseTime.Add(30*time.Second), "key3", "value3"),
							 | 
						|
											createTestEntry(4, baseTime.Add(40*time.Second), "key4", "value4"),
							 | 
						|
										},
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Test reading from beginning
							 | 
						|
									t.Run("ReadFromBeginning", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePosition(baseTime.UnixNano(), -1) // Timestamp-based
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Read from disk
							 | 
						|
										_, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// In real scenario, would then read from memory using LoopProcessLogData
							 | 
						|
										// For this test, just verify disk gave us 0-2
							 | 
						|
										expected := []int64{0, 1, 2}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d", len(expected), len(receivedOffsets))
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								
							 | 
						|
									// Test reading from middle timestamp
							 | 
						|
									t.Run("ReadFromMiddleTimestamp", func(t *testing.T) {
							 | 
						|
										var receivedOffsets []int64
							 | 
						|
										startPos := log_buffer.NewMessagePosition(baseTime.Add(15*time.Second).UnixNano(), -1)
							 | 
						|
								
							 | 
						|
										eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
											receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
											return false, nil
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Read from disk
							 | 
						|
										_, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
										if err != nil {
							 | 
						|
											t.Fatalf("Disk read failed: %v", err)
							 | 
						|
										}
							 | 
						|
								
							 | 
						|
										// Should get offset 2 only (timestamp at 20s >= 15s, offset 1 at 10s is excluded)
							 | 
						|
										expected := []int64{2}
							 | 
						|
										if len(receivedOffsets) != len(expected) {
							 | 
						|
											t.Errorf("Expected %d offsets, got %d: %v", len(expected), len(receivedOffsets), receivedOffsets)
							 | 
						|
										}
							 | 
						|
									})
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestConcurrentSubscribers tests multiple concurrent subscribers
							 | 
						|
								func TestConcurrentSubscribers(t *testing.T) {
							 | 
						|
									baseTime := time.Now()
							 | 
						|
								
							 | 
						|
									mock := &MockLogBuffer{
							 | 
						|
										diskEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(0, baseTime, "key0", "value0"),
							 | 
						|
											createTestEntry(1, baseTime.Add(1*time.Second), "key1", "value1"),
							 | 
						|
											createTestEntry(2, baseTime.Add(2*time.Second), "key2", "value2"),
							 | 
						|
										},
							 | 
						|
										memoryEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(3, baseTime.Add(3*time.Second), "key3", "value3"),
							 | 
						|
											createTestEntry(4, baseTime.Add(4*time.Second), "key4", "value4"),
							 | 
						|
										},
							 | 
						|
										memoryStartOffset: 3,
							 | 
						|
										memoryStopOffset:  4,
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									var wg sync.WaitGroup
							 | 
						|
									results := make(map[string][]int64)
							 | 
						|
									var mu sync.Mutex
							 | 
						|
								
							 | 
						|
									// Spawn 3 concurrent subscribers
							 | 
						|
									for i := 0; i < 3; i++ {
							 | 
						|
										wg.Add(1)
							 | 
						|
										subscriberName := fmt.Sprintf("subscriber-%d", i)
							 | 
						|
								
							 | 
						|
										go func(name string) {
							 | 
						|
											defer wg.Done()
							 | 
						|
								
							 | 
						|
											var receivedOffsets []int64
							 | 
						|
											startPos := log_buffer.NewMessagePositionFromOffset(0)
							 | 
						|
								
							 | 
						|
											eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
												receivedOffsets = append(receivedOffsets, entry.Offset)
							 | 
						|
												return false, nil
							 | 
						|
											}
							 | 
						|
								
							 | 
						|
											eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
												return eachLogFn(entry)
							 | 
						|
											}
							 | 
						|
								
							 | 
						|
											// Read from disk
							 | 
						|
											pos, _, _ := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
								
							 | 
						|
											// Read from memory
							 | 
						|
											mock.MockLoopProcessLogDataWithOffset(name, pos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
								
							 | 
						|
											mu.Lock()
							 | 
						|
											results[name] = receivedOffsets
							 | 
						|
											mu.Unlock()
							 | 
						|
										}(subscriberName)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									wg.Wait()
							 | 
						|
								
							 | 
						|
									// Verify all subscribers got the same data
							 | 
						|
									expected := []int64{0, 1, 2, 3, 4}
							 | 
						|
									for name, offsets := range results {
							 | 
						|
										if len(offsets) != len(expected) {
							 | 
						|
											t.Errorf("%s: Expected %d offsets, got %d", name, len(expected), len(offsets))
							 | 
						|
											continue
							 | 
						|
										}
							 | 
						|
										for i, offset := range offsets {
							 | 
						|
											if offset != expected[i] {
							 | 
						|
												t.Errorf("%s: Offset[%d]: expected %d, got %d", name, i, expected[i], offset)
							 | 
						|
											}
							 | 
						|
										}
							 | 
						|
									}
							 | 
						|
								}
							 | 
						|
								
							 | 
						|
								// TestResumeFromDiskError tests handling of ResumeFromDiskError
							 | 
						|
								func TestResumeFromDiskError(t *testing.T) {
							 | 
						|
									baseTime := time.Now()
							 | 
						|
								
							 | 
						|
									mock := &MockLogBuffer{
							 | 
						|
										diskEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(0, baseTime, "key0", "value0"),
							 | 
						|
											createTestEntry(1, baseTime.Add(1*time.Second), "key1", "value1"),
							 | 
						|
										},
							 | 
						|
										memoryEntries: []*filer_pb.LogEntry{
							 | 
						|
											createTestEntry(10, baseTime.Add(10*time.Second), "key10", "value10"),
							 | 
						|
										},
							 | 
						|
										memoryStartOffset: 10,
							 | 
						|
										memoryStopOffset:  10,
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Try to read offset 5, which is between disk (0-1) and memory (10)
							 | 
						|
									// This should trigger ResumeFromDiskError from memory read
							 | 
						|
									startPos := log_buffer.NewMessagePositionFromOffset(5)
							 | 
						|
								
							 | 
						|
									eachLogFn := func(entry *filer_pb.LogEntry) (bool, error) {
							 | 
						|
										return false, nil
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									eachLogWithOffsetFn := func(entry *filer_pb.LogEntry, offset int64) (bool, error) {
							 | 
						|
										return eachLogFn(entry)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Disk read should return no data (offset 5 > disk end)
							 | 
						|
									_, _, err := mock.MockReadFromDiskFn(startPos, 0, eachLogFn)
							 | 
						|
									if err != nil {
							 | 
						|
										t.Fatalf("Unexpected disk read error: %v", err)
							 | 
						|
									}
							 | 
						|
								
							 | 
						|
									// Memory read should return ResumeFromDiskError (offset 5 < memory start)
							 | 
						|
									_, _, err = mock.MockLoopProcessLogDataWithOffset("test", startPos, 0, func() bool { return true }, eachLogWithOffsetFn)
							 | 
						|
									if err != log_buffer.ResumeFromDiskError {
							 | 
						|
										t.Errorf("Expected ResumeFromDiskError, got: %v", err)
							 | 
						|
									}
							 | 
						|
								}
							 |