mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-04 07:49:35 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			208 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			208 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
Copyright 2017 The Kubernetes Authors.
 | 
						|
 | 
						|
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 set
 | 
						|
 | 
						|
import (
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
 | 
						|
	"k8s.io/kubernetes/pkg/printers"
 | 
						|
 | 
						|
	"github.com/spf13/cobra"
 | 
						|
 | 
						|
	"k8s.io/api/core/v1"
 | 
						|
	"k8s.io/apimachinery/pkg/runtime"
 | 
						|
	"k8s.io/apimachinery/pkg/types"
 | 
						|
	utilerrors "k8s.io/apimachinery/pkg/util/errors"
 | 
						|
	"k8s.io/kubernetes/pkg/kubectl/cmd/templates"
 | 
						|
	cmdutil "k8s.io/kubernetes/pkg/kubectl/cmd/util"
 | 
						|
	"k8s.io/kubernetes/pkg/kubectl/resource"
 | 
						|
	"k8s.io/kubernetes/pkg/kubectl/util/i18n"
 | 
						|
)
 | 
						|
 | 
						|
var (
 | 
						|
	serviceaccountResources = `
 | 
						|
	replicationcontroller (rc), deployment (deploy), daemonset (ds), job, replicaset (rs), statefulset`
 | 
						|
 | 
						|
	serviceaccountLong = templates.LongDesc(i18n.T(`
 | 
						|
	Update ServiceAccount of pod template resources.
 | 
						|
 | 
						|
	Possible resources (case insensitive) can be: 
 | 
						|
	` + serviceaccountResources))
 | 
						|
 | 
						|
	serviceaccountExample = templates.Examples(i18n.T(`
 | 
						|
	# Set Deployment nginx-deployment's ServiceAccount to serviceaccount1
 | 
						|
	kubectl set serviceaccount deployment nginx-deployment serviceaccount1
 | 
						|
 | 
						|
	# Print the result (in yaml format) of updated nginx deployment with serviceaccount from local file, without hitting apiserver
 | 
						|
	kubectl set sa -f nginx-deployment.yaml serviceaccount1 --local --dry-run -o yaml
 | 
						|
	`))
 | 
						|
)
 | 
						|
 | 
						|
// serviceAccountConfig encapsulates the data required to perform the operation.
 | 
						|
type serviceAccountConfig struct {
 | 
						|
	PrintFlags *printers.PrintFlags
 | 
						|
 | 
						|
	fileNameOptions        resource.FilenameOptions
 | 
						|
	out                    io.Writer
 | 
						|
	err                    io.Writer
 | 
						|
	dryRun                 bool
 | 
						|
	cmd                    *cobra.Command
 | 
						|
	shortOutput            bool
 | 
						|
	all                    bool
 | 
						|
	record                 bool
 | 
						|
	output                 string
 | 
						|
	changeCause            string
 | 
						|
	local                  bool
 | 
						|
	updatePodSpecForObject func(runtime.Object, func(*v1.PodSpec) error) (bool, error)
 | 
						|
	infos                  []*resource.Info
 | 
						|
	serviceAccountName     string
 | 
						|
 | 
						|
	PrintObj func(runtime.Object) error
 | 
						|
}
 | 
						|
 | 
						|
// NewCmdServiceAccount returns the "set serviceaccount" command.
 | 
						|
func NewCmdServiceAccount(f cmdutil.Factory, out, err io.Writer) *cobra.Command {
 | 
						|
	saConfig := &serviceAccountConfig{
 | 
						|
		PrintFlags: printers.NewPrintFlags("serviceaccount updated"),
 | 
						|
 | 
						|
		out: out,
 | 
						|
		err: err,
 | 
						|
	}
 | 
						|
 | 
						|
	cmd := &cobra.Command{
 | 
						|
		Use: "serviceaccount (-f FILENAME | TYPE NAME) SERVICE_ACCOUNT",
 | 
						|
		DisableFlagsInUseLine: true,
 | 
						|
		Aliases:               []string{"sa"},
 | 
						|
		Short:                 i18n.T("Update ServiceAccount of a resource"),
 | 
						|
		Long:                  serviceaccountLong,
 | 
						|
		Example:               serviceaccountExample,
 | 
						|
		Run: func(cmd *cobra.Command, args []string) {
 | 
						|
			cmdutil.CheckErr(saConfig.Complete(f, cmd, args))
 | 
						|
			cmdutil.CheckErr(saConfig.Run())
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	saConfig.PrintFlags.AddFlags(cmd)
 | 
						|
 | 
						|
	usage := "identifying the resource to get from a server."
 | 
						|
	cmdutil.AddFilenameOptionFlags(cmd, &saConfig.fileNameOptions, usage)
 | 
						|
	cmd.Flags().BoolVar(&saConfig.all, "all", saConfig.all, "Select all resources, including uninitialized ones, in the namespace of the specified resource types")
 | 
						|
	cmd.Flags().BoolVar(&saConfig.local, "local", saConfig.local, "If true, set serviceaccount will NOT contact api-server but run locally.")
 | 
						|
	cmdutil.AddRecordFlag(cmd)
 | 
						|
	cmdutil.AddDryRunFlag(cmd)
 | 
						|
	cmdutil.AddIncludeUninitializedFlag(cmd)
 | 
						|
	return cmd
 | 
						|
}
 | 
						|
 | 
						|
// Complete configures serviceAccountConfig from command line args.
 | 
						|
func (saConfig *serviceAccountConfig) Complete(f cmdutil.Factory, cmd *cobra.Command, args []string) error {
 | 
						|
	saConfig.shortOutput = cmdutil.GetFlagString(cmd, "output") == "name"
 | 
						|
	saConfig.record = cmdutil.GetRecordFlag(cmd)
 | 
						|
	saConfig.changeCause = f.Command(cmd, false)
 | 
						|
	saConfig.dryRun = cmdutil.GetDryRunFlag(cmd)
 | 
						|
	saConfig.output = cmdutil.GetFlagString(cmd, "output")
 | 
						|
	saConfig.updatePodSpecForObject = f.UpdatePodSpecForObject
 | 
						|
	saConfig.cmd = cmd
 | 
						|
 | 
						|
	saConfig.PrintFlags.Complete(saConfig.dryRun)
 | 
						|
 | 
						|
	printer, err := saConfig.PrintFlags.ToPrinter()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	saConfig.PrintObj = func(obj runtime.Object) error {
 | 
						|
		return printer.PrintObj(obj, saConfig.out)
 | 
						|
	}
 | 
						|
 | 
						|
	cmdNamespace, enforceNamespace, err := f.DefaultNamespace()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	if len(args) == 0 {
 | 
						|
		return errors.New("serviceaccount is required")
 | 
						|
	}
 | 
						|
	saConfig.serviceAccountName = args[len(args)-1]
 | 
						|
	resources := args[:len(args)-1]
 | 
						|
	includeUninitialized := cmdutil.ShouldIncludeUninitialized(cmd, false)
 | 
						|
	builder := f.NewBuilder().
 | 
						|
		Internal().
 | 
						|
		LocalParam(saConfig.local).
 | 
						|
		ContinueOnError().
 | 
						|
		NamespaceParam(cmdNamespace).DefaultNamespace().
 | 
						|
		FilenameParam(enforceNamespace, &saConfig.fileNameOptions).
 | 
						|
		IncludeUninitialized(includeUninitialized).
 | 
						|
		Flatten()
 | 
						|
	if !saConfig.local {
 | 
						|
		builder.ResourceTypeOrNameArgs(saConfig.all, resources...).
 | 
						|
			Latest()
 | 
						|
	}
 | 
						|
	saConfig.infos, err = builder.Do().Infos()
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// Run creates and applies the patch either locally or calling apiserver.
 | 
						|
func (saConfig *serviceAccountConfig) Run() error {
 | 
						|
	patchErrs := []error{}
 | 
						|
	patchFn := func(info *resource.Info) ([]byte, error) {
 | 
						|
		info.Object = info.AsVersioned()
 | 
						|
		saConfig.updatePodSpecForObject(info.Object, func(podSpec *v1.PodSpec) error {
 | 
						|
			podSpec.ServiceAccountName = saConfig.serviceAccountName
 | 
						|
			return nil
 | 
						|
		})
 | 
						|
		return runtime.Encode(cmdutil.InternalVersionJSONEncoder(), info.Object)
 | 
						|
	}
 | 
						|
 | 
						|
	patches := CalculatePatches(saConfig.infos, cmdutil.InternalVersionJSONEncoder(), patchFn)
 | 
						|
	for _, patch := range patches {
 | 
						|
		info := patch.Info
 | 
						|
		if patch.Err != nil {
 | 
						|
			patchErrs = append(patchErrs, fmt.Errorf("error: %s/%s %v\n", info.Mapping.Resource, info.Name, patch.Err))
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		if saConfig.local || saConfig.dryRun {
 | 
						|
			if err := saConfig.PrintObj(patch.Info.AsVersioned()); err != nil {
 | 
						|
				return err
 | 
						|
			}
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		patched, err := resource.NewHelper(info.Client, info.Mapping).Patch(info.Namespace, info.Name, types.StrategicMergePatchType, patch.Patch)
 | 
						|
		if err != nil {
 | 
						|
			patchErrs = append(patchErrs, fmt.Errorf("failed to patch ServiceAccountName %v", err))
 | 
						|
			continue
 | 
						|
		}
 | 
						|
		info.Refresh(patched, true)
 | 
						|
		if saConfig.record || cmdutil.ContainsChangeCause(info) {
 | 
						|
			if patch, patchType, err := cmdutil.ChangeResourcePatch(info, saConfig.changeCause); err == nil {
 | 
						|
				if patched, err = resource.NewHelper(info.Client, info.Mapping).Patch(info.Namespace, info.Name, patchType, patch); err != nil {
 | 
						|
					fmt.Fprintf(saConfig.err, "WARNING: changes to %s/%s can't be recorded: %v\n", info.Mapping.Resource, info.Name, err)
 | 
						|
				}
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if err := saConfig.PrintObj(info.AsVersioned()); err != nil {
 | 
						|
			return err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return utilerrors.NewAggregate(patchErrs)
 | 
						|
}
 |