package main import ( "crypto/tls" "crypto/x509" "flag" "fmt" "net/http" "os" "path/filepath" "time" "github.com/open-policy-agent/gatekeeper-external-data-provider/pkg/handler" "k8s.io/klog/v2" ) const ( defaultPort = 8090 certName = "tls.crt" keyName = "tls.key" ) var ( certDir string clientCAFile string port int ) func init() { klog.InitFlags(nil) flag.StringVar(&certDir, "cert-dir", "", "path to directory containing TLS certificates") flag.StringVar(&clientCAFile, "client-ca-file", "", "path to client CA certificate") flag.IntVar(&port, "port", defaultPort, "Port for the server to listen on") flag.Parse() } func main() { mux := http.NewServeMux() mux.HandleFunc("/", handler.Handler) server := &http.Server{ Addr: fmt.Sprintf(":%d", port), Handler: mux, ReadHeaderTimeout: time.Duration(15) * time.Second, } config := &tls.Config{ MinVersion: tls.VersionTLS13, } if clientCAFile != "" { klog.InfoS("loading Gatekeeper's CA certificate", "clientCAFile", clientCAFile) caCert, err := os.ReadFile(clientCAFile) if err != nil { klog.ErrorS(err, "unable to load Gatekeeper's CA certificate", "clientCAFile", clientCAFile) os.Exit(1) } clientCAs := x509.NewCertPool() clientCAs.AppendCertsFromPEM(caCert) config.ClientCAs = clientCAs config.ClientAuth = tls.RequireAndVerifyClientCert server.TLSConfig = config } if certDir != "" { certFile := filepath.Join(certDir, certName) keyFile := filepath.Join(certDir, keyName) klog.InfoS("starting external data provider server", "port", port, "certFile", certFile, "keyFile", keyFile) if err := server.ListenAndServeTLS(certFile, keyFile); err != nil { klog.ErrorS(err, "unable to start external data provider server") os.Exit(1) } } else { klog.Error("TLS certificates are not provided, the server will not be started") os.Exit(1) } } // TODO: root cause Content-Length header error // func processTimeout(h http.HandlerFunc, duration time.Duration) http.HandlerFunc { // return func(w http.ResponseWriter, r *http.Request) { // ctx, cancel := context.WithTimeout(r.Context(), duration) // defer cancel() // r = r.WithContext(ctx) // processDone := make(chan bool) // go func() { // h(w, r) // processDone <- true // }() // select { // case <-ctx.Done(): // utils.SendResponse(nil, "operation timed out", w) // case <-processDone: // } // } // }