134 lines
4.4 KiB
Go

// Copyright (c) 2020 Gitpod GmbH. All rights reserved.
// Licensed under the GNU Affero General Public License (AGPL).
// See License-AGPL.txt in the project root for license information.
package cmd
import (
"fmt"
"net/http"
"os"
"os/signal"
"syscall"
"time"
"github.com/gitpod-io/gitpod/common-go/log"
"github.com/gitpod-io/gitpod/common-go/pprof"
"github.com/gitpod-io/gitpod/ws-proxy/pkg/proxy"
"github.com/prometheus/client_golang/prometheus"
"github.com/prometheus/client_golang/prometheus/promhttp"
"github.com/spf13/cobra"
)
var jsonLog bool
// runCmd represents the run command
var runCmd = &cobra.Command{
Use: "run <config.json>",
Short: "Starts ws-proxy",
Args: cobra.ExactArgs(1),
Run: func(cmd *cobra.Command, args []string) {
cfg, err := getConfig(args[0])
if err != nil {
log.WithError(err).WithField("filename", args[0]).Fatal("cannot load config")
}
const wsmanConnectionAttempts = 5
workspaceInfoProvider := proxy.NewRemoteWorkspaceInfoProvider(cfg.WorkspaceInfoProviderConfig)
for i := 0; i < wsmanConnectionAttempts; i++ {
err = workspaceInfoProvider.Run()
if err == nil {
break
}
if i == wsmanConnectionAttempts-1 {
continue
}
log.WithError(err).Error("cannot start workspace info provider - will retry in 10 seconds")
time.Sleep(10 * time.Second)
}
if err != nil {
log.WithError(err).Fatal("cannot start workspace info provider")
}
log.Infof("workspace info provider started")
switch cfg.Ingress.Kind {
case HostBasedIngress:
addr := cfg.Ingress.HostBasedIngress.Address
go proxy.NewWorkspaceProxy(addr, cfg.Proxy, proxy.HostBasedRouter(cfg.Ingress.HostBasedIngress.Header, cfg.Proxy.GitpodInstallation.WorkspaceHostSuffix), workspaceInfoProvider).MustServe()
log.WithField("ingress", cfg.Ingress.Kind).Infof("started proxying on %s", addr)
case PathAndHostIngress:
addr := cfg.Ingress.PathAndHostIngress.Address
go proxy.NewWorkspaceProxy(addr, cfg.Proxy, proxy.PathAndHostRouter(cfg.Ingress.PathAndHostIngress.TrimPrefix, cfg.Ingress.PathAndHostIngress.Header, cfg.Proxy.GitpodInstallation.WorkspaceHostSuffix), workspaceInfoProvider).MustServe()
log.WithField("ingress", cfg.Ingress.Kind).Infof("started proxying on %s", addr)
case PathAndPortIngress:
var (
addr = cfg.Ingress.PathAndPortIngress.Address
router = proxy.PathAndPortRouter(cfg.Ingress.PathAndPortIngress.TrimPrefix)
)
go proxy.NewWorkspaceProxy(addr, cfg.Proxy, router, workspaceInfoProvider).MustServe()
log.WithField("ingress", cfg.Ingress.Kind).Infof("started proxying on %s", addr)
for port := cfg.Ingress.PathAndPortIngress.Start; port <= cfg.Ingress.PathAndPortIngress.End; port++ {
go proxy.
NewWorkspaceProxy(fmt.Sprintf(":%d", port), cfg.Proxy, router, workspaceInfoProvider).
MustServe()
}
log.WithField("ingress", cfg.Ingress.Kind).Infof("started proxying on port range :%d-:%d", cfg.Ingress.PathAndPortIngress.Start, cfg.Ingress.PathAndPortIngress.End)
default:
log.Fatalf("unknown ingress kind %s", cfg.Ingress.Kind)
}
if cfg.PProfAddr != "" {
go pprof.Serve(cfg.PProfAddr)
}
if cfg.PrometheusAddr != "" {
reg := prometheus.NewRegistry()
reg.MustRegister(
prometheus.NewGoCollector(),
prometheus.NewProcessCollector(prometheus.ProcessCollectorOpts{}),
)
handler := http.NewServeMux()
handler.Handle("/metrics", promhttp.HandlerFor(reg, promhttp.HandlerOpts{}))
go func() {
err := http.ListenAndServe(cfg.PrometheusAddr, handler)
if err != nil {
log.WithError(err).Error("Prometheus metrics server failed")
}
}()
log.WithField("addr", cfg.PrometheusAddr).Info("started Prometheus metrics server")
}
if cfg.ReadinessProbeAddr != "" {
go func() {
err = http.ListenAndServe(cfg.ReadinessProbeAddr, http.HandlerFunc(func(resp http.ResponseWriter, req *http.Request) {
if workspaceInfoProvider.Ready() {
resp.WriteHeader(http.StatusOK)
} else {
resp.WriteHeader(http.StatusServiceUnavailable)
}
}))
if err != nil {
log.WithError(err).Fatal("readiness endpoint server failed")
}
}()
}
log.Info("🚪 ws-proxy is up and running")
sigChan := make(chan os.Signal, 1)
signal.Notify(sigChan, os.Interrupt, syscall.SIGTERM)
<-sigChan
log.Info("received SIGTERM, ws-proxy is stopping...")
defer func() {
log.Info("ws-proxy stopped.")
}()
},
}
func init() {
rootCmd.AddCommand(runCmd)
}