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