473 lines
13 KiB
Go
473 lines
13 KiB
Go
/*
|
|
Copyright 2014 Google Inc. All rights reserved.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package config
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path"
|
|
"reflect"
|
|
"strconv"
|
|
|
|
"github.com/golang/glog"
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/GoogleCloudPlatform/kubernetes/pkg/client/clientcmd"
|
|
clientcmdapi "github.com/GoogleCloudPlatform/kubernetes/pkg/client/clientcmd/api"
|
|
)
|
|
|
|
type PathOptions struct {
|
|
Local bool
|
|
Global bool
|
|
UseEnvVar bool
|
|
|
|
LocalFile string
|
|
GlobalFile string
|
|
EnvVarFile string
|
|
|
|
EnvVar string
|
|
ExplicitFileFlag string
|
|
|
|
LoadingRules *clientcmd.ClientConfigLoadingRules
|
|
}
|
|
|
|
func NewCmdConfig(pathOptions *PathOptions, out io.Writer) *cobra.Command {
|
|
if len(pathOptions.ExplicitFileFlag) == 0 {
|
|
pathOptions.ExplicitFileFlag = clientcmd.RecommendedConfigPathFlag
|
|
}
|
|
if len(pathOptions.EnvVar) > 0 {
|
|
pathOptions.EnvVarFile = os.Getenv(pathOptions.EnvVar)
|
|
}
|
|
|
|
cmd := &cobra.Command{
|
|
Use: "config SUBCOMMAND",
|
|
Short: "config modifies kubeconfig files",
|
|
Long: `config modifies kubeconfig files using subcommands like "kubectl config set current-context my-context"`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
cmd.Help()
|
|
},
|
|
}
|
|
|
|
// file paths are common to all sub commands
|
|
cmd.PersistentFlags().BoolVar(&pathOptions.Local, "local", pathOptions.Local, "use the kubeconfig in the current directory")
|
|
cmd.PersistentFlags().BoolVar(&pathOptions.Global, "global", pathOptions.Global, "use the kubeconfig from "+pathOptions.GlobalFile)
|
|
cmd.PersistentFlags().BoolVar(&pathOptions.UseEnvVar, "envvar", pathOptions.UseEnvVar, "use the kubeconfig from $"+pathOptions.EnvVar)
|
|
cmd.PersistentFlags().StringVar(&pathOptions.LoadingRules.ExplicitPath, pathOptions.ExplicitFileFlag, pathOptions.LoadingRules.ExplicitPath, "use a particular kubeconfig file")
|
|
|
|
cmd.AddCommand(NewCmdConfigView(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigSetCluster(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigSetAuthInfo(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigSetContext(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigSet(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigUnset(out, pathOptions))
|
|
cmd.AddCommand(NewCmdConfigUseContext(out, pathOptions))
|
|
|
|
return cmd
|
|
}
|
|
|
|
func NewDefaultPathOptions() *PathOptions {
|
|
ret := &PathOptions{
|
|
LocalFile: ".kubeconfig",
|
|
GlobalFile: path.Join(os.Getenv("HOME"), "/.kube/.kubeconfig"),
|
|
EnvVar: clientcmd.RecommendedConfigPathEnvVar,
|
|
EnvVarFile: os.Getenv(clientcmd.RecommendedConfigPathEnvVar),
|
|
|
|
ExplicitFileFlag: clientcmd.RecommendedConfigPathFlag,
|
|
|
|
LoadingRules: clientcmd.NewDefaultClientConfigLoadingRules(),
|
|
}
|
|
ret.LoadingRules.DoNotResolvePaths = true
|
|
|
|
return ret
|
|
}
|
|
|
|
func (o PathOptions) Validate() error {
|
|
if len(o.LoadingRules.ExplicitPath) > 0 {
|
|
if o.Global {
|
|
return errors.New("cannot specify both --" + o.ExplicitFileFlag + " and --global")
|
|
}
|
|
if o.Local {
|
|
return errors.New("cannot specify both --" + o.ExplicitFileFlag + " and --local")
|
|
}
|
|
if o.UseEnvVar {
|
|
return errors.New("cannot specify both --" + o.ExplicitFileFlag + " and --envvar")
|
|
}
|
|
}
|
|
|
|
if o.Global {
|
|
if o.Local {
|
|
return errors.New("cannot specify both --global and --local")
|
|
}
|
|
if o.UseEnvVar {
|
|
return errors.New("cannot specify both --global and --envvar")
|
|
}
|
|
}
|
|
|
|
if o.Local {
|
|
if o.UseEnvVar {
|
|
return errors.New("cannot specify both --local and --envvar")
|
|
}
|
|
}
|
|
|
|
if o.UseEnvVar {
|
|
if len(o.EnvVarFile) == 0 {
|
|
return fmt.Errorf("environment variable %v does not have a value", o.EnvVar)
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (o *PathOptions) getStartingConfig() (*clientcmdapi.Config, error) {
|
|
if err := o.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
config := clientcmdapi.NewConfig()
|
|
|
|
switch {
|
|
case o.Global:
|
|
config = getConfigFromFileOrDie(o.GlobalFile)
|
|
|
|
case o.UseEnvVar:
|
|
config = getConfigFromFileOrDie(o.EnvVarFile)
|
|
|
|
case o.Local:
|
|
config = getConfigFromFileOrDie(o.LocalFile)
|
|
|
|
// no specific flag was set, load according to the loading rules
|
|
default:
|
|
clientConfig := clientcmd.NewNonInteractiveDeferredLoadingClientConfig(o.LoadingRules, &clientcmd.ConfigOverrides{})
|
|
rawConfig, err := clientConfig.RawConfig()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
config = &rawConfig
|
|
|
|
}
|
|
|
|
return config, nil
|
|
}
|
|
|
|
// GetDefaultFilename returns the name of the file you should write into (create if necessary), if you're trying to create
|
|
// a new stanza as opposed to updating an existing one.
|
|
func (o *PathOptions) GetDefaultFilename() string {
|
|
if o.IsExplicitFile() {
|
|
return o.GetExplicitFile()
|
|
}
|
|
|
|
if len(o.EnvVarFile) > 0 {
|
|
return o.EnvVarFile
|
|
}
|
|
|
|
if _, err := os.Stat(o.LocalFile); err == nil {
|
|
return o.LocalFile
|
|
}
|
|
|
|
return o.GlobalFile
|
|
|
|
}
|
|
|
|
func (o *PathOptions) IsExplicitFile() bool {
|
|
switch {
|
|
case len(o.LoadingRules.ExplicitPath) > 0 ||
|
|
o.Global ||
|
|
o.UseEnvVar ||
|
|
o.Local:
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (o *PathOptions) GetExplicitFile() string {
|
|
if !o.IsExplicitFile() {
|
|
return ""
|
|
}
|
|
|
|
switch {
|
|
case len(o.LoadingRules.ExplicitPath) > 0:
|
|
return o.LoadingRules.ExplicitPath
|
|
|
|
case o.Global:
|
|
return o.GlobalFile
|
|
|
|
case o.UseEnvVar:
|
|
return o.EnvVarFile
|
|
|
|
case o.Local:
|
|
return o.LocalFile
|
|
}
|
|
|
|
return ""
|
|
}
|
|
|
|
// ModifyConfig takes a Config object, iterates through Clusters, AuthInfos, and Contexts, uses the LocationOfOrigin if specified or
|
|
// uses the default destination file to write the results into. This results in multiple file reads, but it's very easy to follow.
|
|
// Preferences and CurrentContext should always be set in the default destination file. Since we can't distinguish between empty and missing values
|
|
// (no nil strings), we're forced have separate handling for them. In all the currently known cases, newConfig should have, at most, one difference,
|
|
// that means that this code will only write into a single file.
|
|
func (o *PathOptions) ModifyConfig(newConfig clientcmdapi.Config) error {
|
|
startingConfig, err := o.getStartingConfig()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// at this point, config and startingConfig should have, at most, one difference. We need to chase the difference until we find it
|
|
// then we'll build a partial config object to call write upon. Special case the test for current context and preferences since those
|
|
// always write to the default file.
|
|
switch {
|
|
case reflect.DeepEqual(*startingConfig, newConfig):
|
|
// nothing to do
|
|
|
|
case startingConfig.CurrentContext != newConfig.CurrentContext:
|
|
if err := o.writeCurrentContext(newConfig.CurrentContext); err != nil {
|
|
return err
|
|
}
|
|
|
|
case !reflect.DeepEqual(startingConfig.Preferences, newConfig.Preferences):
|
|
if err := o.writePreferences(newConfig.Preferences); err != nil {
|
|
return err
|
|
}
|
|
|
|
default:
|
|
// something is different. Search every cluster, authInfo, and context. First from new to old for differences, then from old to new for deletions
|
|
for key, cluster := range newConfig.Clusters {
|
|
startingCluster, exists := startingConfig.Clusters[key]
|
|
if !reflect.DeepEqual(cluster, startingCluster) || !exists {
|
|
destinationFile := cluster.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
configToWrite.Clusters[key] = cluster
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, context := range newConfig.Contexts {
|
|
startingContext, exists := startingConfig.Contexts[key]
|
|
if !reflect.DeepEqual(context, startingContext) || !exists {
|
|
destinationFile := context.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
configToWrite.Contexts[key] = context
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, authInfo := range newConfig.AuthInfos {
|
|
startingAuthInfo, exists := startingConfig.AuthInfos[key]
|
|
if !reflect.DeepEqual(authInfo, startingAuthInfo) || !exists {
|
|
destinationFile := authInfo.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
configToWrite.AuthInfos[key] = authInfo
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, cluster := range startingConfig.Clusters {
|
|
if _, exists := newConfig.Clusters[key]; !exists {
|
|
destinationFile := cluster.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
delete(configToWrite.Clusters, key)
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, context := range startingConfig.Contexts {
|
|
if _, exists := newConfig.Contexts[key]; !exists {
|
|
destinationFile := context.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
delete(configToWrite.Contexts, key)
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, authInfo := range startingConfig.AuthInfos {
|
|
if _, exists := newConfig.AuthInfos[key]; !exists {
|
|
destinationFile := authInfo.LocationOfOrigin
|
|
if len(destinationFile) == 0 {
|
|
destinationFile = o.GetDefaultFilename()
|
|
}
|
|
|
|
configToWrite := getConfigFromFileOrDie(destinationFile)
|
|
delete(configToWrite.AuthInfos, key)
|
|
|
|
if err := clientcmd.WriteToFile(*configToWrite, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// writeCurrentContext takes three possible paths.
|
|
// If newCurrentContext is the same as the startingConfig's current context, then we exit.
|
|
// If newCurrentContext has a value, then that value is written into the default destination file.
|
|
// If newCurrentContext is empty, then we find the config file that is setting the CurrentContext and clear the value from that file
|
|
func (o *PathOptions) writeCurrentContext(newCurrentContext string) error {
|
|
if startingConfig, err := o.getStartingConfig(); err != nil {
|
|
return err
|
|
} else if startingConfig.CurrentContext == newCurrentContext {
|
|
return nil
|
|
}
|
|
|
|
if len(newCurrentContext) > 0 {
|
|
destinationFile := o.GetDefaultFilename()
|
|
config := getConfigFromFileOrDie(destinationFile)
|
|
config.CurrentContext = newCurrentContext
|
|
|
|
if err := clientcmd.WriteToFile(*config, destinationFile); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
if o.IsExplicitFile() {
|
|
file := o.GetExplicitFile()
|
|
currConfig := getConfigFromFileOrDie(file)
|
|
currConfig.CurrentContext = newCurrentContext
|
|
if err := clientcmd.WriteToFile(*currConfig, file); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
filesToCheck := make([]string, 0, len(o.LoadingRules.Precedence)+1)
|
|
filesToCheck = append(filesToCheck, o.LoadingRules.ExplicitPath)
|
|
filesToCheck = append(filesToCheck, o.LoadingRules.Precedence...)
|
|
|
|
for _, file := range filesToCheck {
|
|
currConfig := getConfigFromFileOrDie(file)
|
|
|
|
if len(currConfig.CurrentContext) > 0 {
|
|
currConfig.CurrentContext = newCurrentContext
|
|
if err := clientcmd.WriteToFile(*currConfig, file); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (o *PathOptions) writePreferences(newPrefs clientcmdapi.Preferences) error {
|
|
if startingConfig, err := o.getStartingConfig(); err != nil {
|
|
return err
|
|
} else if reflect.DeepEqual(startingConfig.Preferences, newPrefs) {
|
|
return nil
|
|
}
|
|
|
|
if o.IsExplicitFile() {
|
|
file := o.GetExplicitFile()
|
|
currConfig := getConfigFromFileOrDie(file)
|
|
currConfig.Preferences = newPrefs
|
|
if err := clientcmd.WriteToFile(*currConfig, file); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
filesToCheck := make([]string, 0, len(o.LoadingRules.Precedence)+1)
|
|
filesToCheck = append(filesToCheck, o.LoadingRules.ExplicitPath)
|
|
filesToCheck = append(filesToCheck, o.LoadingRules.Precedence...)
|
|
|
|
for _, file := range filesToCheck {
|
|
currConfig := getConfigFromFileOrDie(file)
|
|
|
|
if !reflect.DeepEqual(currConfig.Preferences, newPrefs) {
|
|
currConfig.Preferences = newPrefs
|
|
if err := clientcmd.WriteToFile(*currConfig, file); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// getConfigFromFileOrDie tries to read a kubeconfig file and if it can't, it calls exit. One exception, missing files result in empty configs, not an exit
|
|
func getConfigFromFileOrDie(filename string) *clientcmdapi.Config {
|
|
config, err := clientcmd.LoadFromFile(filename)
|
|
if err != nil && !os.IsNotExist(err) {
|
|
glog.FatalDepth(1, err)
|
|
}
|
|
|
|
if config == nil {
|
|
return clientcmdapi.NewConfig()
|
|
}
|
|
|
|
return config
|
|
}
|
|
|
|
func toBool(propertyValue string) (bool, error) {
|
|
boolValue := false
|
|
if len(propertyValue) != 0 {
|
|
var err error
|
|
boolValue, err = strconv.ParseBool(propertyValue)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
return boolValue, nil
|
|
}
|