From 17f23058db623046f7b1fdc5c73e0659c0322351 Mon Sep 17 00:00:00 2001 From: Brad Davidson Date: Thu, 26 Oct 2023 18:33:11 +0000 Subject: [PATCH] Remove http/tls mux from etcd client listener Upstream etcd has seen issues with grpc muxed with http and recommends against it. Ref: https://github.com/etcd-io/etcd/issues/15402 Signed-off-by: Brad Davidson --- pkg/endpoint/endpoint.go | 35 ++------------------- pkg/endpoint/http.go | 66 ---------------------------------------- 2 files changed, 2 insertions(+), 99 deletions(-) delete mode 100644 pkg/endpoint/http.go diff --git a/pkg/endpoint/endpoint.go b/pkg/endpoint/endpoint.go index 926337d3..0dfa7b1f 100644 --- a/pkg/endpoint/endpoint.go +++ b/pkg/endpoint/endpoint.go @@ -19,7 +19,6 @@ import ( "github.com/pkg/errors" "github.com/prometheus/client_golang/prometheus" "github.com/sirupsen/logrus" - "github.com/soheilhy/cmux" "go.etcd.io/etcd/server/v3/embed" "google.golang.org/grpc" "google.golang.org/grpc/credentials" @@ -88,45 +87,15 @@ func Listen(ctx context.Context, config Config) (ETCDConfig, error) { } b.Register(grpcServer) - // set up HTTP server with basic mux - httpServer := httpServer() - // Create raw listener and wrap in cmux for protocol switching listener, err := createListener(config) if err != nil { return ETCDConfig{}, errors.Wrap(err, "creating listener") } - m := cmux.New(listener) - - if config.ServerTLSConfig.CertFile != "" && config.ServerTLSConfig.KeyFile != "" { - // If using TLS, wrap handler in GRPC/HTTP switching handler and serve TLS - httpServer.Handler = grpcHandlerFunc(grpcServer, httpServer.Handler) - anyl := m.Match(cmux.Any()) - go func() { - if err := httpServer.ServeTLS(anyl, config.ServerTLSConfig.CertFile, config.ServerTLSConfig.KeyFile); err != nil { - logrus.Errorf("Kine TLS server shutdown: %v", err) - } - }() - } else { - // If using plaintext, use cmux matching for GRPC/HTTP switching - grpcl := m.Match(cmux.HTTP2()) - go func() { - if err := grpcServer.Serve(grpcl); err != nil { - logrus.Errorf("Kine GRPC server shutdown: %v", err) - } - }() - httpl := m.Match(cmux.HTTP1()) - go func() { - if err := httpServer.Serve(httpl); err != nil { - logrus.Errorf("Kine HTTP server shutdown: %v", err) - } - }() - } go func() { - if err := m.Serve(); err != nil { - logrus.Errorf("Kine listener shutdown: %v", err) - grpcServer.Stop() + if err := grpcServer.Serve(listener); err != nil { + logrus.Errorf("Kine GPRC server exited: %v", err) } }() diff --git a/pkg/endpoint/http.go b/pkg/endpoint/http.go deleted file mode 100644 index f66058da..00000000 --- a/pkg/endpoint/http.go +++ /dev/null @@ -1,66 +0,0 @@ -package endpoint - -import ( - "log" - "net/http" - "strings" - - "github.com/sirupsen/logrus" - "google.golang.org/grpc" -) - -var ( - etcdVersion = []byte(`{"etcdserver":"3.5.0","etcdcluster":"3.5.0"}`) - versionPath = "/version" -) - -// httpServer returns a HTTP server with the basic mux handler. -func httpServer() *http.Server { - // Set up root HTTP mux with basic response handlers - mux := http.NewServeMux() - handleBasic(mux) - - return &http.Server{ - Handler: mux, - ErrorLog: log.New(logrus.StandardLogger().Writer(), "kinehttp ", log.LstdFlags), - } -} - -// handleBasic binds basic HTTP response handlers to a mux. -func handleBasic(mux *http.ServeMux) { - mux.HandleFunc(versionPath, serveVersion) -} - -// serveVersion responds with a canned JSON version response. -func serveVersion(w http.ResponseWriter, r *http.Request) { - if !allowMethod(w, r, http.MethodGet) { - return - } - w.Header().Set("Content-Type", "application/json") - w.Write(etcdVersion) -} - -// allowMethod returns true if a method is allowed, or false (after sending a -// MethodNotAllowed error to the client) if it is not. -func allowMethod(w http.ResponseWriter, r *http.Request, m string) bool { - if m == r.Method { - return true - } - w.Header().Set("Allow", m) - http.Error(w, "Method Not Allowed", http.StatusMethodNotAllowed) - return false -} - -// grpcHandlerFunc takes a GRPC server and HTTP handler, and returns a handler -// function that will route GRPC requests to the GRPC server, and everything -// else to the HTTP handler. This is based on sample code provided in the GRPC -// ServeHTTP documentation for sharing a port between GRPC and HTTP handlers. -func grpcHandlerFunc(grpcServer *grpc.Server, httpHandler http.Handler) http.Handler { - return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { - if r.ProtoMajor == 2 && strings.HasPrefix(r.Header.Get("Content-Type"), "application/grpc") { - grpcServer.ServeHTTP(w, r) - } else { - httpHandler.ServeHTTP(w, r) - } - }) -}