package client import ( "errors" "fmt" "strings" "sync" "time" v1 "k8s.io/api/core/v1" "k8s.io/cli-runtime/pkg/genericclioptions" restclient "k8s.io/client-go/rest" clientcmd "k8s.io/client-go/tools/clientcmd" clientcmdapi "k8s.io/client-go/tools/clientcmd/api" ) const ( defaultCallTimeoutDuration time.Duration = 10 * time.Second // UsePersistentConfig caches client config to avoid reloads. UsePersistentConfig = true ) // Config tracks a kubernetes configuration. type Config struct { flags *genericclioptions.ConfigFlags mutex *sync.RWMutex } // NewConfig returns a new k8s config or an error if the flags are invalid. func NewConfig(f *genericclioptions.ConfigFlags) *Config { return &Config{ flags: f, mutex: &sync.RWMutex{}, } } // CallTimeout returns the call timeout if set or the default if not set. func (c *Config) CallTimeout() time.Duration { if !isSet(c.flags.Timeout) { return defaultCallTimeoutDuration } dur, err := time.ParseDuration(*c.flags.Timeout) if err != nil { return defaultCallTimeoutDuration } return dur } func (c *Config) RESTConfig() (*restclient.Config, error) { return c.clientConfig().ClientConfig() } // Flags returns configuration flags. func (c *Config) Flags() *genericclioptions.ConfigFlags { return c.flags } func (c *Config) RawConfig() (clientcmdapi.Config, error) { return c.clientConfig().RawConfig() } func (c *Config) clientConfig() clientcmd.ClientConfig { return c.flags.ToRawKubeConfigLoader() } func (c *Config) reset() {} // SwitchContext changes the kubeconfig context to a new cluster. func (c *Config) SwitchContext(name string) error { if _, err := c.GetContext(name); err != nil { return fmt.Errorf("context %q does not exist", name) } flags := genericclioptions.NewConfigFlags(UsePersistentConfig) flags.Context = &name flags.Timeout = c.flags.Timeout flags.KubeConfig = c.flags.KubeConfig c.flags = flags return nil } // CurrentContextName returns the currently active config context. func (c *Config) CurrentContextName() (string, error) { if isSet(c.flags.Context) { return *c.flags.Context, nil } cfg, err := c.RawConfig() if err != nil { return "", err } return cfg.CurrentContext, nil } // GetContext fetch a given context or error if it does not exists. func (c *Config) GetContext(n string) (*clientcmdapi.Context, error) { cfg, err := c.RawConfig() if err != nil { return nil, err } if c, ok := cfg.Contexts[n]; ok { return c, nil } return nil, fmt.Errorf("invalid context `%s specified", n) } // Contexts fetch all available contexts. func (c *Config) Contexts() (map[string]*clientcmdapi.Context, error) { cfg, err := c.RawConfig() if err != nil { return nil, err } return cfg.Contexts, nil } // DelContext remove a given context from the configuration. func (c *Config) DelContext(n string) error { cfg, err := c.RawConfig() if err != nil { return err } delete(cfg.Contexts, n) acc, err := c.ConfigAccess() if err != nil { return err } return clientcmd.ModifyConfig(acc, cfg, true) } // RenameContext renames a context. func (c *Config) RenameContext(old string, new string) error { cfg, err := c.RawConfig() if err != nil { return err } if _, ok := cfg.Contexts[new]; ok { return fmt.Errorf("context with name %s already exists", new) } cfg.Contexts[new] = cfg.Contexts[old] delete(cfg.Contexts, old) acc, err := c.ConfigAccess() if err != nil { return err } if err := clientcmd.ModifyConfig(acc, cfg, true); err != nil { return err } current, err := c.CurrentContextName() if err != nil { return err } if current == old { return c.SwitchContext(new) } return nil } // ContextNames fetch all available contexts. func (c *Config) ContextNames() ([]string, error) { cfg, err := c.RawConfig() if err != nil { return nil, err } cc := make([]string, 0, len(cfg.Contexts)) for n := range cfg.Contexts { cc = append(cc, n) } return cc, nil } // ClusterNameFromContext returns the cluster associated with the given context. func (c *Config) ClusterNameFromContext(context string) (string, error) { cfg, err := c.RawConfig() if err != nil { return "", err } if ctx, ok := cfg.Contexts[context]; ok { return ctx.Cluster, nil } return "", fmt.Errorf("unable to locate cluster from context %s", context) } // CurrentClusterName returns the active cluster name. func (c *Config) CurrentClusterName() (string, error) { if isSet(c.flags.ClusterName) { return *c.flags.ClusterName, nil } cfg, err := c.RawConfig() if err != nil { return "", err } context, err := c.CurrentContextName() if err != nil { context = cfg.CurrentContext } if ctx, ok := cfg.Contexts[context]; ok { return ctx.Cluster, nil } return "", errors.New("unable to locate current cluster") } // ClusterNames fetch all kubeconfig defined clusters. func (c *Config) ClusterNames() (map[string]struct{}, error) { cfg, err := c.RawConfig() if err != nil { return nil, err } cc := make(map[string]struct{}, len(cfg.Clusters)) for name := range cfg.Clusters { cc[name] = struct{}{} } return cc, nil } // CurrentGroupNames retrieves the active group names. func (c *Config) CurrentGroupNames() ([]string, error) { if areSet(c.flags.ImpersonateGroup) { return *c.flags.ImpersonateGroup, nil } return []string{}, errors.New("unable to locate current group") } // ImpersonateGroups retrieves the active groups if set on the CLI. func (c *Config) ImpersonateGroups() (string, error) { if areSet(c.flags.ImpersonateGroup) { return strings.Join(*c.flags.ImpersonateGroup, ","), nil } return "", errors.New("no groups set") } // ImpersonateUser retrieves the active user name if set on the CLI. func (c *Config) ImpersonateUser() (string, error) { if isSet(c.flags.Impersonate) { return *c.flags.Impersonate, nil } return "", errors.New("no user set") } // CurrentUserName retrieves the active user name. func (c *Config) CurrentUserName() (string, error) { if isSet(c.flags.Impersonate) { return *c.flags.Impersonate, nil } if isSet(c.flags.AuthInfoName) { return *c.flags.AuthInfoName, nil } cfg, err := c.RawConfig() if err != nil { return "", err } current := cfg.CurrentContext if isSet(c.flags.Context) { current = *c.flags.Context } if ctx, ok := cfg.Contexts[current]; ok { return ctx.AuthInfo, nil } return "", errors.New("unable to locate current user") } // CurrentNamespaceName retrieves the active namespace. func (c *Config) CurrentNamespaceName() (string, error) { ns, _, err := c.clientConfig().Namespace() return ns, err } // ConfigAccess return the current kubeconfig api server access configuration. func (c *Config) ConfigAccess() (clientcmd.ConfigAccess, error) { c.mutex.RLock() defer c.mutex.RUnlock() return c.clientConfig().ConfigAccess(), nil } // ---------------------------------------------------------------------------- // Helpers... // NamespaceNames fetch all available namespaces on current cluster. func NamespaceNames(nns []v1.Namespace) []string { nn := make([]string, 0, len(nns)) for _, ns := range nns { nn = append(nn, ns.Name) } return nn } func isSet(s *string) bool { return s != nil && len(*s) != 0 } func areSet(s *[]string) bool { return s != nil && len(*s) != 0 }