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.
		
		
		
		
		
			
		
			
				
					
					
						
							143 lines
						
					
					
						
							5.8 KiB
						
					
					
				
			
		
		
		
			
			
			
		
		
	
	
							143 lines
						
					
					
						
							5.8 KiB
						
					
					
				| package command | |
| 
 | |
| import ( | |
| 	"fmt" | |
| 	"net/http" | |
| 	_ "net/http/pprof" | |
| 	"os" | |
| 
 | |
| 	"github.com/seaweedfs/seaweedfs/weed/glog" | |
| 	"github.com/seaweedfs/seaweedfs/weed/mq/kafka/gateway" | |
| 	"github.com/seaweedfs/seaweedfs/weed/util" | |
| ) | |
| 
 | |
| var ( | |
| 	mqKafkaGatewayOptions mqKafkaGatewayOpts | |
| ) | |
| 
 | |
| type mqKafkaGatewayOpts struct { | |
| 	ip                *string | |
| 	ipBind            *string | |
| 	port              *int | |
| 	pprofPort         *int | |
| 	master            *string | |
| 	filerGroup        *string | |
| 	schemaRegistryURL *string | |
| 	defaultPartitions *int | |
| } | |
| 
 | |
| func init() { | |
| 	cmdMqKafkaGateway.Run = runMqKafkaGateway | |
| 	mqKafkaGatewayOptions.ip = cmdMqKafkaGateway.Flag.String("ip", util.DetectedHostAddress(), "Kafka gateway advertised host address") | |
| 	mqKafkaGatewayOptions.ipBind = cmdMqKafkaGateway.Flag.String("ip.bind", "", "Kafka gateway bind address (default: same as -ip)") | |
| 	mqKafkaGatewayOptions.port = cmdMqKafkaGateway.Flag.Int("port", 9092, "Kafka gateway listen port") | |
| 	mqKafkaGatewayOptions.pprofPort = cmdMqKafkaGateway.Flag.Int("port.pprof", 0, "HTTP profiling port (0 to disable)") | |
| 	mqKafkaGatewayOptions.master = cmdMqKafkaGateway.Flag.String("master", "localhost:9333", "comma-separated SeaweedFS master servers") | |
| 	mqKafkaGatewayOptions.filerGroup = cmdMqKafkaGateway.Flag.String("filerGroup", "", "filer group name") | |
| 	mqKafkaGatewayOptions.schemaRegistryURL = cmdMqKafkaGateway.Flag.String("schema-registry-url", "", "Schema Registry URL (required for schema management)") | |
| 	mqKafkaGatewayOptions.defaultPartitions = cmdMqKafkaGateway.Flag.Int("default-partitions", 4, "Default number of partitions for auto-created topics") | |
| } | |
| 
 | |
| var cmdMqKafkaGateway = &Command{ | |
| 	UsageLine: "mq.kafka.gateway [-ip=<host>] [-ip.bind=<bind_addr>] [-port=9092] [-master=<master_servers>] [-filerGroup=<group>] [-default-partitions=4] -schema-registry-url=<url>", | |
| 	Short:     "start a Kafka wire-protocol gateway for SeaweedMQ with schema management", | |
| 	Long: `Start a Kafka wire-protocol gateway translating Kafka client requests to SeaweedMQ. | |
|  | |
| Connects to SeaweedFS master servers to discover available brokers and integrates with | |
| Schema Registry for schema-aware topic management. | |
|  | |
| Options: | |
|   -ip                  Advertised host address that clients should connect to (default: auto-detected) | |
|   -ip.bind             Bind address for the gateway to listen on (default: same as -ip) | |
|                        Use 0.0.0.0 to bind to all interfaces while advertising specific IP | |
|   -port                Listen port (default: 9092) | |
|   -default-partitions  Default number of partitions for auto-created topics (default: 4) | |
|   -schema-registry-url Schema Registry URL (REQUIRED for schema management) | |
|  | |
| Examples: | |
|   weed mq.kafka.gateway -port=9092 -master=localhost:9333 -schema-registry-url=http://localhost:8081 | |
|   weed mq.kafka.gateway -ip=gateway1 -port=9092 -master=master1:9333,master2:9333 -schema-registry-url=http://schema-registry:8081 | |
|   weed mq.kafka.gateway -ip=external.host.com -ip.bind=0.0.0.0 -master=localhost:9333 -schema-registry-url=http://schema-registry:8081 | |
|  | |
| This is experimental and currently supports a minimal subset for development. | |
| `, | |
| } | |
| 
 | |
| func runMqKafkaGateway(cmd *Command, args []string) bool { | |
| 	// Validate required options | |
| 	if *mqKafkaGatewayOptions.master == "" { | |
| 		glog.Fatalf("SeaweedFS master address is required (-master)") | |
| 		return false | |
| 	} | |
| 
 | |
| 	// Schema Registry URL is required for schema management | |
| 	if *mqKafkaGatewayOptions.schemaRegistryURL == "" { | |
| 		glog.Fatalf("Schema Registry URL is required (-schema-registry-url)") | |
| 		return false | |
| 	} | |
| 
 | |
| 	// Determine bind address - default to advertised IP if not specified | |
| 	bindIP := *mqKafkaGatewayOptions.ipBind | |
| 	if bindIP == "" { | |
| 		bindIP = *mqKafkaGatewayOptions.ip | |
| 	} | |
| 
 | |
| 	// Construct listen address from bind IP and port | |
| 	listenAddr := fmt.Sprintf("%s:%d", bindIP, *mqKafkaGatewayOptions.port) | |
| 
 | |
| 	// Set advertised host for Kafka protocol handler | |
| 	if err := os.Setenv("KAFKA_ADVERTISED_HOST", *mqKafkaGatewayOptions.ip); err != nil { | |
| 		glog.Warningf("Failed to set KAFKA_ADVERTISED_HOST environment variable: %v", err) | |
| 	} | |
| 
 | |
| 	srv := gateway.NewServer(gateway.Options{ | |
| 		Listen:            listenAddr, | |
| 		Masters:           *mqKafkaGatewayOptions.master, | |
| 		FilerGroup:        *mqKafkaGatewayOptions.filerGroup, | |
| 		SchemaRegistryURL: *mqKafkaGatewayOptions.schemaRegistryURL, | |
| 		DefaultPartitions: int32(*mqKafkaGatewayOptions.defaultPartitions), | |
| 	}) | |
| 
 | |
| 	glog.Warningf("EXPERIMENTAL FEATURE: MQ Kafka Gateway is experimental and should NOT be used in production environments. It currently supports only a minimal subset of Kafka protocol for development purposes.") | |
| 
 | |
| 	// Show bind vs advertised addresses for clarity | |
| 	if bindIP != *mqKafkaGatewayOptions.ip { | |
| 		glog.V(0).Infof("Starting MQ Kafka Gateway: binding to %s, advertising %s:%d to clients", | |
| 			listenAddr, *mqKafkaGatewayOptions.ip, *mqKafkaGatewayOptions.port) | |
| 	} else { | |
| 		glog.V(0).Infof("Starting MQ Kafka Gateway on %s", listenAddr) | |
| 	} | |
| 	glog.V(0).Infof("Using SeaweedMQ brokers from masters: %s", *mqKafkaGatewayOptions.master) | |
| 
 | |
| 	// Start HTTP profiling server if enabled | |
| 	if *mqKafkaGatewayOptions.pprofPort > 0 { | |
| 		go func() { | |
| 			pprofAddr := fmt.Sprintf(":%d", *mqKafkaGatewayOptions.pprofPort) | |
| 			glog.V(0).Infof("Kafka Gateway pprof server listening on %s", pprofAddr) | |
| 			glog.V(0).Infof("Access profiling at: http://localhost:%d/debug/pprof/", *mqKafkaGatewayOptions.pprofPort) | |
| 			if err := http.ListenAndServe(pprofAddr, nil); err != nil { | |
| 				glog.Errorf("pprof server error: %v", err) | |
| 			} | |
| 		}() | |
| 	} | |
| 
 | |
| 	if err := srv.Start(); err != nil { | |
| 		glog.Fatalf("mq kafka gateway start: %v", err) | |
| 		return false | |
| 	} | |
| 
 | |
| 	// Set up graceful shutdown | |
| 	defer func() { | |
| 		glog.V(0).Infof("Shutting down MQ Kafka Gateway...") | |
| 		if err := srv.Close(); err != nil { | |
| 			glog.Errorf("mq kafka gateway close: %v", err) | |
| 		} | |
| 	}() | |
| 
 | |
| 	// Serve blocks until closed | |
| 	if err := srv.Wait(); err != nil { | |
| 		glog.Errorf("mq kafka gateway wait: %v", err) | |
| 		return false | |
| 	} | |
| 	return true | |
| }
 |