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.
340 lines
8.9 KiB
340 lines
8.9 KiB
package integration
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/mq_pb"
|
|
"google.golang.org/grpc/metadata"
|
|
)
|
|
|
|
// MockSubscribeStream implements mq_pb.SeaweedMessaging_SubscribeMessageClient for testing
|
|
type MockSubscribeStream struct {
|
|
sendCalls []interface{}
|
|
closed bool
|
|
}
|
|
|
|
func (m *MockSubscribeStream) Send(req *mq_pb.SubscribeMessageRequest) error {
|
|
m.sendCalls = append(m.sendCalls, req)
|
|
return nil
|
|
}
|
|
|
|
func (m *MockSubscribeStream) Recv() (*mq_pb.SubscribeMessageResponse, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func (m *MockSubscribeStream) CloseSend() error {
|
|
m.closed = true
|
|
return nil
|
|
}
|
|
|
|
func (m *MockSubscribeStream) Header() (metadata.MD, error) { return nil, nil }
|
|
func (m *MockSubscribeStream) Trailer() metadata.MD { return nil }
|
|
func (m *MockSubscribeStream) Context() context.Context { return context.Background() }
|
|
func (m *MockSubscribeStream) SendMsg(m2 interface{}) error { return nil }
|
|
func (m *MockSubscribeStream) RecvMsg(m2 interface{}) error { return nil }
|
|
|
|
// TestNeedsRestart tests the NeedsRestart logic
|
|
func TestNeedsRestart(t *testing.T) {
|
|
bc := &BrokerClient{}
|
|
|
|
tests := []struct {
|
|
name string
|
|
session *BrokerSubscriberSession
|
|
requestedOffset int64
|
|
want bool
|
|
reason string
|
|
}{
|
|
{
|
|
name: "Stream is nil - needs restart",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: nil,
|
|
},
|
|
requestedOffset: 100,
|
|
want: true,
|
|
reason: "Stream is nil",
|
|
},
|
|
{
|
|
name: "Offset in cache - no restart needed",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
consumedRecords: []*SeaweedRecord{
|
|
{Offset: 95},
|
|
{Offset: 96},
|
|
{Offset: 97},
|
|
{Offset: 98},
|
|
{Offset: 99},
|
|
},
|
|
},
|
|
requestedOffset: 97,
|
|
want: false,
|
|
reason: "Offset 97 is in cache [95-99]",
|
|
},
|
|
{
|
|
name: "Offset before current - needs restart",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
},
|
|
requestedOffset: 50,
|
|
want: true,
|
|
reason: "Requested offset 50 < current 100",
|
|
},
|
|
{
|
|
name: "Large gap ahead - needs restart",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
},
|
|
requestedOffset: 2000,
|
|
want: true,
|
|
reason: "Gap of 1900 is > 1000",
|
|
},
|
|
{
|
|
name: "Small gap ahead - no restart needed",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
},
|
|
requestedOffset: 150,
|
|
want: false,
|
|
reason: "Gap of 50 is < 1000",
|
|
},
|
|
{
|
|
name: "Exact match - no restart needed",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
},
|
|
requestedOffset: 100,
|
|
want: false,
|
|
reason: "Exact match with current offset",
|
|
},
|
|
{
|
|
name: "Context is nil - needs restart",
|
|
session: &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: nil,
|
|
},
|
|
requestedOffset: 100,
|
|
want: true,
|
|
reason: "Context is nil",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
got := bc.NeedsRestart(tt.session, tt.requestedOffset)
|
|
if got != tt.want {
|
|
t.Errorf("NeedsRestart() = %v, want %v (reason: %s)", got, tt.want, tt.reason)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestNeedsRestart_CacheLogic tests cache-based restart decisions
|
|
func TestNeedsRestart_CacheLogic(t *testing.T) {
|
|
bc := &BrokerClient{}
|
|
|
|
// Create session with cache containing offsets 100-109
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 110,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
consumedRecords: []*SeaweedRecord{
|
|
{Offset: 100}, {Offset: 101}, {Offset: 102}, {Offset: 103}, {Offset: 104},
|
|
{Offset: 105}, {Offset: 106}, {Offset: 107}, {Offset: 108}, {Offset: 109},
|
|
},
|
|
}
|
|
|
|
testCases := []struct {
|
|
offset int64
|
|
want bool
|
|
desc string
|
|
}{
|
|
{100, false, "First offset in cache"},
|
|
{105, false, "Middle offset in cache"},
|
|
{109, false, "Last offset in cache"},
|
|
{99, true, "Before cache start"},
|
|
{110, false, "Current position"},
|
|
{111, false, "One ahead"},
|
|
{1200, true, "Large gap > 1000"},
|
|
}
|
|
|
|
for _, tc := range testCases {
|
|
t.Run(tc.desc, func(t *testing.T) {
|
|
got := bc.NeedsRestart(session, tc.offset)
|
|
if got != tc.want {
|
|
t.Errorf("NeedsRestart(offset=%d) = %v, want %v (%s)", tc.offset, got, tc.want, tc.desc)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestNeedsRestart_EmptyCache tests behavior with empty cache
|
|
func TestNeedsRestart_EmptyCache(t *testing.T) {
|
|
bc := &BrokerClient{}
|
|
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
consumedRecords: nil, // Empty cache
|
|
}
|
|
|
|
tests := []struct {
|
|
offset int64
|
|
want bool
|
|
desc string
|
|
}{
|
|
{50, true, "Before current"},
|
|
{100, false, "At current"},
|
|
{150, false, "Small gap ahead"},
|
|
{1200, true, "Large gap ahead"},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.desc, func(t *testing.T) {
|
|
got := bc.NeedsRestart(session, tt.offset)
|
|
if got != tt.want {
|
|
t.Errorf("NeedsRestart(offset=%d) = %v, want %v (%s)", tt.offset, got, tt.want, tt.desc)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
// TestNeedsRestart_ThreadSafety tests concurrent access
|
|
func TestNeedsRestart_ThreadSafety(t *testing.T) {
|
|
bc := &BrokerClient{}
|
|
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
}
|
|
|
|
// Run many concurrent checks
|
|
done := make(chan bool)
|
|
for i := 0; i < 100; i++ {
|
|
go func(offset int64) {
|
|
bc.NeedsRestart(session, offset)
|
|
done <- true
|
|
}(int64(i))
|
|
}
|
|
|
|
// Wait for all to complete
|
|
for i := 0; i < 100; i++ {
|
|
<-done
|
|
}
|
|
|
|
// Test passes if no panic/race condition
|
|
}
|
|
|
|
// TestRestartSubscriber_StateManagement tests session state management
|
|
func TestRestartSubscriber_StateManagement(t *testing.T) {
|
|
oldStream := &MockSubscribeStream{}
|
|
oldCtx, oldCancel := context.WithCancel(context.Background())
|
|
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 100,
|
|
Stream: oldStream,
|
|
Ctx: oldCtx,
|
|
Cancel: oldCancel,
|
|
consumedRecords: []*SeaweedRecord{
|
|
{Offset: 100, Key: []byte("key100"), Value: []byte("value100")},
|
|
{Offset: 101, Key: []byte("key101"), Value: []byte("value101")},
|
|
{Offset: 102, Key: []byte("key102"), Value: []byte("value102")},
|
|
},
|
|
nextOffsetToRead: 103,
|
|
}
|
|
|
|
// Verify initial state
|
|
if len(session.consumedRecords) != 3 {
|
|
t.Errorf("Initial cache size = %d, want 3", len(session.consumedRecords))
|
|
}
|
|
if session.nextOffsetToRead != 103 {
|
|
t.Errorf("Initial nextOffsetToRead = %d, want 103", session.nextOffsetToRead)
|
|
}
|
|
if session.StartOffset != 100 {
|
|
t.Errorf("Initial StartOffset = %d, want 100", session.StartOffset)
|
|
}
|
|
|
|
// Note: Full RestartSubscriber testing requires gRPC mocking
|
|
// These tests verify the core state management and NeedsRestart logic
|
|
}
|
|
|
|
// BenchmarkNeedsRestart_CacheHit benchmarks cache hit performance
|
|
func BenchmarkNeedsRestart_CacheHit(b *testing.B) {
|
|
bc := &BrokerClient{}
|
|
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 1000,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
consumedRecords: make([]*SeaweedRecord, 100),
|
|
}
|
|
|
|
for i := 0; i < 100; i++ {
|
|
session.consumedRecords[i] = &SeaweedRecord{Offset: int64(i)}
|
|
}
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
bc.NeedsRestart(session, 50) // Hit cache
|
|
}
|
|
}
|
|
|
|
// BenchmarkNeedsRestart_CacheMiss benchmarks cache miss performance
|
|
func BenchmarkNeedsRestart_CacheMiss(b *testing.B) {
|
|
bc := &BrokerClient{}
|
|
|
|
session := &BrokerSubscriberSession{
|
|
Topic: "test-topic",
|
|
Partition: 0,
|
|
StartOffset: 1000,
|
|
Stream: &MockSubscribeStream{},
|
|
Ctx: context.Background(),
|
|
consumedRecords: make([]*SeaweedRecord, 100),
|
|
}
|
|
|
|
for i := 0; i < 100; i++ {
|
|
session.consumedRecords[i] = &SeaweedRecord{Offset: int64(i)}
|
|
}
|
|
|
|
b.ResetTimer()
|
|
for i := 0; i < b.N; i++ {
|
|
bc.NeedsRestart(session, 500) // Miss cache (within gap threshold)
|
|
}
|
|
}
|