// Copyright (c) 2020 TypeFox 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 ( "encoding/json" "fmt" "io/ioutil" "os" "github.com/prometheus/client_golang/prometheus" wsk8s "github.com/gitpod-io/gitpod/common-go/kubernetes" "github.com/gitpod-io/gitpod/common-go/log" "github.com/gitpod-io/gitpod/common-go/tracing" "github.com/gitpod-io/gitpod/content-service/pkg/storage" "github.com/gitpod-io/gitpod/ws-manager/pkg/manager" "github.com/spf13/cobra" "k8s.io/client-go/kubernetes" "k8s.io/client-go/rest" "k8s.io/client-go/tools/clientcmd" ) var ( // ServiceName is the name we use for tracing/logging ServiceName = "ws-manager" // Version of this service - set during build Version = "" ) var cfgFile string var kubeconfig string var jsonLog bool // rootCmd represents the base command when called without any subcommands var rootCmd = &cobra.Command{ Use: "ws-manager", Short: "ws-manager starts/stops/controls workspace deployments in Kubernetes", PersistentPreRun: func(cmd *cobra.Command, args []string) { log.Init(ServiceName, Version, jsonLog, jsonLog) }, } // Execute adds all child commands to the root command and sets flags appropriately. // This is called by main.main(). It only needs to happen once to the rootCmd. func Execute() { promrep := &tracing.PromReporter{ Operations: map[string]tracing.SpanMetricMapping{ "StartWorkspace": { Name: "wsman_start_workspace", Help: "time it takes to service a StartWorkspace request", Buckets: prometheus.LinearBuckets(0, 500, 10), // 10 buckets, each 500ms wide }, }, } closer := tracing.Init(ServiceName, tracing.WithPrometheusReporter(promrep)) if closer != nil { defer closer.Close() } if err := rootCmd.Execute(); err != nil { fmt.Println(err) os.Exit(1) } } func init() { rootCmd.PersistentFlags().StringVar(&cfgFile, "config", "", "config file (default is $HOME/.kedgei.yaml)") rootCmd.PersistentFlags().BoolVarP(&jsonLog, "json-log", "v", false, "produce JSON log output on verbose level") rootCmd.PersistentFlags().StringVar(&kubeconfig, "kubeconfig", "", "path to the kubeconfig file to use (defaults to in-cluster config)") } func getConfig() *config { ctnt, err := ioutil.ReadFile(cfgFile) if err != nil { log.WithError(err).Error("cannot read configuration. Maybe missing --config?") os.Exit(1) } var cfg config err = json.Unmarshal(ctnt, &cfg) if err != nil { log.WithError(err).Error("cannot read configuration. Maybe missing --config?") os.Exit(1) } return &cfg } type config struct { Manager manager.Configuration `json:"manager"` Content struct { Storage storage.Config `json:"storage"` } `json:"content"` RPCServerAddr string `json:"rpcServerAddr"` TLS struct { Certificate string `json:"crt"` PrivateKey string `json:"key"` } `json:"tls"` PProfAddr string `json:"pprofAddr"` PrometheusAddr string `json:"prometheusAddr"` } func newClientSet() (*kubernetes.Clientset, error) { if kubeconfig != "" { res, err := clientcmd.BuildConfigFromFlags("", kubeconfig) if err != nil { return nil, err } res.RateLimiter = &wsk8s.UnlimitedRateLimiter{} return kubernetes.NewForConfig(res) } k8s, err := rest.InClusterConfig() if err != nil { return nil, err } k8s.RateLimiter = &wsk8s.UnlimitedRateLimiter{} return kubernetes.NewForConfig(k8s) }