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