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.
 
 
 
 
 
 

335 lines
9.0 KiB

package kafka_test
import (
"net"
"testing"
"time"
"github.com/seaweedfs/seaweedfs/weed/mq/kafka/gateway"
)
// TestSeaweedMQIntegration_E2E tests the complete workflow with SeaweedMQ backend
// This test requires a real SeaweedMQ Agent running
func TestSeaweedMQIntegration_E2E(t *testing.T) {
// Skip by default - requires real SeaweedMQ setup
t.Skip("Integration test requires real SeaweedMQ setup - run manually")
// Test configuration
agentAddress := "localhost:17777" // Default SeaweedMQ Agent address
// Start the gateway with SeaweedMQ backend
gatewayServer := gateway.NewServer(gateway.Options{
Listen: ":0", // random port
AgentAddress: agentAddress,
UseSeaweedMQ: true,
})
err := gatewayServer.Start()
if err != nil {
t.Fatalf("Failed to start gateway with SeaweedMQ backend: %v", err)
}
defer gatewayServer.Close()
addr := gatewayServer.Addr()
t.Logf("Started Kafka Gateway with SeaweedMQ backend on %s", addr)
// Wait for startup
time.Sleep(200 * time.Millisecond)
// Test basic connectivity
t.Run("SeaweedMQ_BasicConnectivity", func(t *testing.T) {
testSeaweedMQConnectivity(t, addr)
})
// Test topic lifecycle with SeaweedMQ
t.Run("SeaweedMQ_TopicLifecycle", func(t *testing.T) {
testSeaweedMQTopicLifecycle(t, addr)
})
// Test produce/consume workflow
t.Run("SeaweedMQ_ProduceConsume", func(t *testing.T) {
testSeaweedMQProduceConsume(t, addr)
})
}
// testSeaweedMQConnectivity verifies gateway responds correctly
func testSeaweedMQConnectivity(t *testing.T, addr string) {
conn, err := net.DialTimeout("tcp", addr, 5*time.Second)
if err != nil {
t.Fatalf("Failed to connect to SeaweedMQ gateway: %v", err)
}
defer conn.Close()
// Send ApiVersions request
req := buildApiVersionsRequest()
_, err = conn.Write(req)
if err != nil {
t.Fatalf("Failed to send ApiVersions: %v", err)
}
// Read response
sizeBytes := make([]byte, 4)
conn.SetReadDeadline(time.Now().Add(5 * time.Second))
_, err = conn.Read(sizeBytes)
if err != nil {
t.Fatalf("Failed to read response size: %v", err)
}
responseSize := uint32(sizeBytes[0])<<24 | uint32(sizeBytes[1])<<16 | uint32(sizeBytes[2])<<8 | uint32(sizeBytes[3])
if responseSize == 0 || responseSize > 10000 {
t.Fatalf("Invalid response size: %d", responseSize)
}
responseBody := make([]byte, responseSize)
_, err = conn.Read(responseBody)
if err != nil {
t.Fatalf("Failed to read response body: %v", err)
}
// Verify API keys are advertised
if len(responseBody) < 20 {
t.Fatalf("Response too short")
}
apiKeyCount := uint32(responseBody[6])<<24 | uint32(responseBody[7])<<16 | uint32(responseBody[8])<<8 | uint32(responseBody[9])
if apiKeyCount < 6 {
t.Errorf("Expected at least 6 API keys, got %d", apiKeyCount)
}
t.Logf("SeaweedMQ gateway connectivity test passed, %d API keys advertised", apiKeyCount)
}
// testSeaweedMQTopicLifecycle tests creating and managing topics
func testSeaweedMQTopicLifecycle(t *testing.T, addr string) {
conn, err := net.DialTimeout("tcp", addr, 5*time.Second)
if err != nil {
t.Fatalf("Failed to connect: %v", err)
}
defer conn.Close()
// Test CreateTopics request
topicName := "seaweedmq-test-topic"
createReq := buildCreateTopicsRequestCustom(topicName)
_, err = conn.Write(createReq)
if err != nil {
t.Fatalf("Failed to send CreateTopics: %v", err)
}
// Read response
sizeBytes := make([]byte, 4)
conn.SetReadDeadline(time.Now().Add(5 * time.Second))
_, err = conn.Read(sizeBytes)
if err != nil {
t.Fatalf("Failed to read CreateTopics response size: %v", err)
}
responseSize := uint32(sizeBytes[0])<<24 | uint32(sizeBytes[1])<<16 | uint32(sizeBytes[2])<<8 | uint32(sizeBytes[3])
responseBody := make([]byte, responseSize)
_, err = conn.Read(responseBody)
if err != nil {
t.Fatalf("Failed to read CreateTopics response: %v", err)
}
// Parse response to check for success (basic validation)
if len(responseBody) < 10 {
t.Fatalf("CreateTopics response too short")
}
t.Logf("SeaweedMQ topic creation test completed: %d bytes response", len(responseBody))
}
// testSeaweedMQProduceConsume tests the produce/consume workflow
func testSeaweedMQProduceConsume(t *testing.T, addr string) {
// This would be a more comprehensive test in a full implementation
// For now, just test that Produce requests are handled
conn, err := net.DialTimeout("tcp", addr, 5*time.Second)
if err != nil {
t.Fatalf("Failed to connect: %v", err)
}
defer conn.Close()
// First create a topic
createReq := buildCreateTopicsRequestCustom("produce-test-topic")
_, err = conn.Write(createReq)
if err != nil {
t.Fatalf("Failed to send CreateTopics: %v", err)
}
// Read CreateTopics response
sizeBytes := make([]byte, 4)
conn.SetReadDeadline(time.Now().Add(5 * time.Second))
_, err = conn.Read(sizeBytes)
if err != nil {
t.Fatalf("Failed to read CreateTopics size: %v", err)
}
responseSize := uint32(sizeBytes[0])<<24 | uint32(sizeBytes[1])<<16 | uint32(sizeBytes[2])<<8 | uint32(sizeBytes[3])
responseBody := make([]byte, responseSize)
_, err = conn.Read(responseBody)
if err != nil {
t.Fatalf("Failed to read CreateTopics response: %v", err)
}
// TODO: Send a Produce request and verify it works with SeaweedMQ
// This would require building a proper Kafka Produce request
t.Logf("SeaweedMQ produce/consume test placeholder completed")
}
// buildCreateTopicsRequestCustom creates a CreateTopics request for a specific topic
func buildCreateTopicsRequestCustom(topicName string) []byte {
clientID := "seaweedmq-test-client"
// Approximate message size
messageSize := 2 + 2 + 4 + 2 + len(clientID) + 4 + 4 + 2 + len(topicName) + 4 + 2 + 4 + 4
request := make([]byte, 0, messageSize+4)
// Message size placeholder
sizePos := len(request)
request = append(request, 0, 0, 0, 0)
// API key (CreateTopics = 19)
request = append(request, 0, 19)
// API version
request = append(request, 0, 4)
// Correlation ID
request = append(request, 0, 0, 0x30, 0x42) // 12354
// Client ID
request = append(request, 0, byte(len(clientID)))
request = append(request, []byte(clientID)...)
// Timeout (5000ms)
request = append(request, 0, 0, 0x13, 0x88)
// Topics count (1)
request = append(request, 0, 0, 0, 1)
// Topic name
request = append(request, 0, byte(len(topicName)))
request = append(request, []byte(topicName)...)
// Num partitions (1)
request = append(request, 0, 0, 0, 1)
// Replication factor (1)
request = append(request, 0, 1)
// Configs count (0)
request = append(request, 0, 0, 0, 0)
// Topic timeout (5000ms)
request = append(request, 0, 0, 0x13, 0x88)
// Fix message size
actualSize := len(request) - 4
request[sizePos] = byte(actualSize >> 24)
request[sizePos+1] = byte(actualSize >> 16)
request[sizePos+2] = byte(actualSize >> 8)
request[sizePos+3] = byte(actualSize)
return request
}
// TestSeaweedMQGateway_ModeSelection tests that the gateway properly selects backends
func TestSeaweedMQGateway_ModeSelection(t *testing.T) {
// Test in-memory mode (should always work)
t.Run("InMemoryMode", func(t *testing.T) {
server := gateway.NewServer(gateway.Options{
Listen: ":0",
UseSeaweedMQ: false,
})
err := server.Start()
if err != nil {
t.Fatalf("In-memory mode should start: %v", err)
}
defer server.Close()
addr := server.Addr()
if addr == "" {
t.Errorf("Server should have listening address")
}
t.Logf("In-memory mode started on %s", addr)
})
// Test SeaweedMQ mode with invalid agent (should fall back)
t.Run("SeaweedMQModeFallback", func(t *testing.T) {
server := gateway.NewServer(gateway.Options{
Listen: ":0",
AgentAddress: "invalid:99999", // Invalid address
UseSeaweedMQ: true,
})
err := server.Start()
if err != nil {
t.Fatalf("Should start even with invalid agent (fallback to in-memory): %v", err)
}
defer server.Close()
addr := server.Addr()
if addr == "" {
t.Errorf("Server should have listening address")
}
t.Logf("SeaweedMQ mode with fallback started on %s", addr)
})
}
// TestSeaweedMQGateway_ConfigValidation tests configuration validation
func TestSeaweedMQGateway_ConfigValidation(t *testing.T) {
testCases := []struct {
name string
options gateway.Options
shouldWork bool
}{
{
name: "ValidInMemory",
options: gateway.Options{
Listen: ":0",
UseSeaweedMQ: false,
},
shouldWork: true,
},
{
name: "ValidSeaweedMQWithAgent",
options: gateway.Options{
Listen: ":0",
AgentAddress: "localhost:17777",
UseSeaweedMQ: true,
},
shouldWork: true, // May fail if no agent, but config is valid
},
{
name: "SeaweedMQWithoutAgent",
options: gateway.Options{
Listen: ":0",
UseSeaweedMQ: true,
// AgentAddress is empty
},
shouldWork: true, // Should fall back to in-memory
},
}
for _, tc := range testCases {
t.Run(tc.name, func(t *testing.T) {
server := gateway.NewServer(tc.options)
err := server.Start()
if tc.shouldWork && err != nil {
t.Errorf("Expected config to work, got error: %v", err)
}
if err == nil {
server.Close()
t.Logf("Config test passed for %s", tc.name)
}
})
}
}