2019-01-12 04:58:27 +00:00
/ *
Copyright 2016 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 (
"fmt"
"github.com/spf13/cobra"
2020-08-10 17:43:49 +00:00
"k8s.io/klog/v2"
2019-01-12 04:58:27 +00:00
2020-03-26 21:07:15 +00:00
v1 "k8s.io/api/core/v1"
2019-01-12 04:58:27 +00:00
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/types"
utilerrors "k8s.io/apimachinery/pkg/util/errors"
"k8s.io/cli-runtime/pkg/genericclioptions"
2019-04-07 17:07:55 +00:00
"k8s.io/cli-runtime/pkg/printers"
"k8s.io/cli-runtime/pkg/resource"
2019-09-27 21:51:53 +00:00
cmdutil "k8s.io/kubectl/pkg/cmd/util"
"k8s.io/kubectl/pkg/polymorphichelpers"
"k8s.io/kubectl/pkg/scheme"
"k8s.io/kubectl/pkg/util/i18n"
"k8s.io/kubectl/pkg/util/templates"
2019-01-12 04:58:27 +00:00
)
// ImageOptions is the start of the data required to perform the operation. As new fields are added, add them here instead of
// referencing the cmd.Flags()
type SetImageOptions struct {
resource . FilenameOptions
PrintFlags * genericclioptions . PrintFlags
RecordFlags * genericclioptions . RecordFlags
2020-03-26 21:07:15 +00:00
Infos [ ] * resource . Info
Selector string
DryRunStrategy cmdutil . DryRunStrategy
DryRunVerifier * resource . DryRunVerifier
All bool
Output string
Local bool
ResolveImage ImageResolver
2020-08-10 17:43:49 +00:00
fieldManager string
2019-01-12 04:58:27 +00:00
PrintObj printers . ResourcePrinterFunc
Recorder genericclioptions . Recorder
UpdatePodSpecForObject polymorphichelpers . UpdatePodSpecForObjectFunc
Resources [ ] string
ContainerImages map [ string ] string
genericclioptions . IOStreams
}
var (
2020-12-01 01:06:26 +00:00
imageResources = i18n . T ( `
pod ( po ) , replicationcontroller ( rc ) , deployment ( deploy ) , daemonset ( ds ) , replicaset ( rs ) ` )
2019-01-12 04:58:27 +00:00
2020-12-01 01:06:26 +00:00
imageLong = templates . LongDesc ( i18n . T ( `
2019-01-12 04:58:27 +00:00
Update existing container image ( s ) of resources .
Possible resources include ( case insensitive ) :
2020-12-01 01:06:26 +00:00
` ) + imageResources )
2019-01-12 04:58:27 +00:00
2019-04-07 17:07:55 +00:00
imageExample = templates . Examples ( `
2019-01-12 04:58:27 +00:00
# Set a deployment ' s nginx container image to ' nginx : 1.9 .1 ' , and its busybox container image to ' busybox ' .
kubectl set image deployment / nginx busybox = busybox nginx = nginx : 1.9 .1
# Update all deployments ' and rc ' s nginx container ' s image to ' nginx : 1.9 .1 '
kubectl set image deployments , rc nginx = nginx : 1.9 .1 -- all
# Update image of all containers of daemonset abc to ' nginx : 1.9 .1 '
kubectl set image daemonset abc *= nginx : 1.9 .1
# Print result ( in yaml format ) of updating nginx container image from local file , without hitting the server
kubectl set image - f path / to / file . yaml nginx = nginx : 1.9 .1 -- local - o yaml ` )
)
2019-04-07 17:07:55 +00:00
// NewImageOptions returns an initialized SetImageOptions instance
2019-01-12 04:58:27 +00:00
func NewImageOptions ( streams genericclioptions . IOStreams ) * SetImageOptions {
return & SetImageOptions {
PrintFlags : genericclioptions . NewPrintFlags ( "image updated" ) . WithTypeSetter ( scheme . Scheme ) ,
RecordFlags : genericclioptions . NewRecordFlags ( ) ,
Recorder : genericclioptions . NoopRecorder { } ,
IOStreams : streams ,
}
}
2019-04-07 17:07:55 +00:00
// NewCmdImage returns an initialized Command instance for the 'set image' sub command
2019-01-12 04:58:27 +00:00
func NewCmdImage ( f cmdutil . Factory , streams genericclioptions . IOStreams ) * cobra . Command {
o := NewImageOptions ( streams )
cmd := & cobra . Command {
Use : "image (-f FILENAME | TYPE NAME) CONTAINER_NAME_1=CONTAINER_IMAGE_1 ... CONTAINER_NAME_N=CONTAINER_IMAGE_N" ,
DisableFlagsInUseLine : true ,
Short : i18n . T ( "Update image of a pod template" ) ,
2019-04-07 17:07:55 +00:00
Long : imageLong ,
Example : imageExample ,
2019-01-12 04:58:27 +00:00
Run : func ( cmd * cobra . Command , args [ ] string ) {
cmdutil . CheckErr ( o . Complete ( f , cmd , args ) )
cmdutil . CheckErr ( o . Validate ( ) )
cmdutil . CheckErr ( o . Run ( ) )
} ,
}
o . PrintFlags . AddFlags ( cmd )
o . RecordFlags . AddFlags ( cmd )
usage := "identifying the resource to get from a server."
cmdutil . AddFilenameOptionFlags ( cmd , & o . FilenameOptions , usage )
cmd . Flags ( ) . BoolVar ( & o . All , "all" , o . All , "Select all resources, including uninitialized ones, in the namespace of the specified resource types" )
cmd . Flags ( ) . StringVarP ( & o . Selector , "selector" , "l" , o . Selector , "Selector (label query) to filter on, not including uninitialized ones, supports '=', '==', and '!='.(e.g. -l key1=value1,key2=value2)" )
cmd . Flags ( ) . BoolVar ( & o . Local , "local" , o . Local , "If true, set image will NOT contact api-server but run locally." )
cmdutil . AddDryRunFlag ( cmd )
2020-08-10 17:43:49 +00:00
cmdutil . AddFieldManagerFlagVar ( cmd , & o . fieldManager , "kubectl-set" )
2019-01-12 04:58:27 +00:00
return cmd
}
2019-04-07 17:07:55 +00:00
// Complete completes all required options
2019-01-12 04:58:27 +00:00
func ( o * SetImageOptions ) Complete ( f cmdutil . Factory , cmd * cobra . Command , args [ ] string ) error {
var err error
o . RecordFlags . Complete ( cmd )
o . Recorder , err = o . RecordFlags . ToRecorder ( )
if err != nil {
return err
}
o . UpdatePodSpecForObject = polymorphichelpers . UpdatePodSpecForObjectFn
2020-03-26 21:07:15 +00:00
o . DryRunStrategy , err = cmdutil . GetDryRunStrategy ( cmd )
if err != nil {
return err
}
dynamicClient , err := f . DynamicClient ( )
if err != nil {
return err
}
2021-03-18 22:40:29 +00:00
o . DryRunVerifier = resource . NewDryRunVerifier ( dynamicClient , f . OpenAPIGetter ( ) )
2019-01-12 04:58:27 +00:00
o . Output = cmdutil . GetFlagString ( cmd , "output" )
o . ResolveImage = resolveImageFunc
2020-03-26 21:07:15 +00:00
cmdutil . PrintFlagsWithDryRunStrategy ( o . PrintFlags , o . DryRunStrategy )
2019-01-12 04:58:27 +00:00
printer , err := o . PrintFlags . ToPrinter ( )
if err != nil {
return err
}
o . PrintObj = printer . PrintObj
cmdNamespace , enforceNamespace , err := f . ToRawKubeConfigLoader ( ) . Namespace ( )
if err != nil {
return err
}
o . Resources , o . ContainerImages , err = getResourcesAndImages ( args )
if err != nil {
return err
}
builder := f . NewBuilder ( ) .
WithScheme ( scheme . Scheme , scheme . Scheme . PrioritizedVersionsAllGroups ( ) ... ) .
LocalParam ( o . Local ) .
ContinueOnError ( ) .
NamespaceParam ( cmdNamespace ) . DefaultNamespace ( ) .
FilenameParam ( enforceNamespace , & o . FilenameOptions ) .
Flatten ( )
if ! o . Local {
builder . LabelSelectorParam ( o . Selector ) .
ResourceTypeOrNameArgs ( o . All , o . Resources ... ) .
Latest ( )
} else {
// if a --local flag was provided, and a resource was specified in the form
// <resource>/<name>, fail immediately as --local cannot query the api server
// for the specified resource.
if len ( o . Resources ) > 0 {
return resource . LocalResourceError
}
}
o . Infos , err = builder . Do ( ) . Infos ( )
if err != nil {
return err
}
return nil
}
2019-04-07 17:07:55 +00:00
// Validate makes sure provided values in SetImageOptions are valid
2019-01-12 04:58:27 +00:00
func ( o * SetImageOptions ) Validate ( ) error {
errors := [ ] error { }
if o . All && len ( o . Selector ) > 0 {
errors = append ( errors , fmt . Errorf ( "cannot set --all and --selector at the same time" ) )
}
2019-08-30 18:33:25 +00:00
if len ( o . Resources ) < 1 && cmdutil . IsFilenameSliceEmpty ( o . Filenames , o . Kustomize ) {
2019-01-12 04:58:27 +00:00
errors = append ( errors , fmt . Errorf ( "one or more resources must be specified as <resource> <name> or <resource>/<name>" ) )
}
if len ( o . ContainerImages ) < 1 {
errors = append ( errors , fmt . Errorf ( "at least one image update is required" ) )
} else if len ( o . ContainerImages ) > 1 && hasWildcardKey ( o . ContainerImages ) {
errors = append ( errors , fmt . Errorf ( "all containers are already specified by *, but saw more than one container_name=container_image pairs" ) )
}
2020-03-26 21:07:15 +00:00
if o . Local && o . DryRunStrategy == cmdutil . DryRunServer {
errors = append ( errors , fmt . Errorf ( "cannot specify --local and --dry-run=server - did you mean --dry-run=client?" ) )
}
2019-01-12 04:58:27 +00:00
return utilerrors . NewAggregate ( errors )
}
2019-04-07 17:07:55 +00:00
// Run performs the execution of 'set image' sub command
2019-01-12 04:58:27 +00:00
func ( o * SetImageOptions ) Run ( ) error {
allErrs := [ ] error { }
patches := CalculatePatches ( o . Infos , scheme . DefaultJSONEncoder ( ) , func ( obj runtime . Object ) ( [ ] byte , error ) {
_ , err := o . UpdatePodSpecForObject ( obj , func ( spec * v1 . PodSpec ) error {
for name , image := range o . ContainerImages {
2019-04-07 17:07:55 +00:00
resolvedImageName , err := o . ResolveImage ( image )
if err != nil {
allErrs = append ( allErrs , fmt . Errorf ( "error: unable to resolve image %q for container %q: %v" , image , name , err ) )
if name == "*" {
break
2019-01-12 04:58:27 +00:00
}
2019-04-07 17:07:55 +00:00
continue
2019-01-12 04:58:27 +00:00
}
2019-04-07 17:07:55 +00:00
initContainerFound := setImage ( spec . InitContainers , name , resolvedImageName )
containerFound := setImage ( spec . Containers , name , resolvedImageName )
if ! containerFound && ! initContainerFound {
2019-01-12 04:58:27 +00:00
allErrs = append ( allErrs , fmt . Errorf ( "error: unable to find container named %q" , name ) )
}
}
return nil
} )
if err != nil {
return nil , err
}
// record this change (for rollout history)
if err := o . Recorder . Record ( obj ) ; err != nil {
klog . V ( 4 ) . Infof ( "error recording current command: %v" , err )
}
return runtime . Encode ( scheme . DefaultJSONEncoder ( ) , obj )
} )
for _ , patch := range patches {
info := patch . Info
if patch . Err != nil {
2019-04-07 17:07:55 +00:00
name := info . ObjectName ( )
allErrs = append ( allErrs , fmt . Errorf ( "error: %s %v\n" , name , patch . Err ) )
2019-01-12 04:58:27 +00:00
continue
}
// no changes
if string ( patch . Patch ) == "{}" || len ( patch . Patch ) == 0 {
continue
}
2020-03-26 21:07:15 +00:00
if o . Local || o . DryRunStrategy == cmdutil . DryRunClient {
2019-01-12 04:58:27 +00:00
if err := o . PrintObj ( info . Object , o . Out ) ; err != nil {
allErrs = append ( allErrs , err )
}
continue
}
2020-03-26 21:07:15 +00:00
if o . DryRunStrategy == cmdutil . DryRunServer {
if err := o . DryRunVerifier . HasSupport ( info . Mapping . GroupVersionKind ) ; err != nil {
return err
}
}
2019-01-12 04:58:27 +00:00
// patch the change
2020-03-26 21:07:15 +00:00
actual , err := resource .
NewHelper ( info . Client , info . Mapping ) .
DryRun ( o . DryRunStrategy == cmdutil . DryRunServer ) .
2020-08-10 17:43:49 +00:00
WithFieldManager ( o . fieldManager ) .
2020-03-26 21:07:15 +00:00
Patch ( info . Namespace , info . Name , types . StrategicMergePatchType , patch . Patch , nil )
2019-01-12 04:58:27 +00:00
if err != nil {
2019-04-07 17:07:55 +00:00
allErrs = append ( allErrs , fmt . Errorf ( "failed to patch image update to pod template: %v" , err ) )
2019-01-12 04:58:27 +00:00
continue
}
if err := o . PrintObj ( actual , o . Out ) ; err != nil {
allErrs = append ( allErrs , err )
}
}
return utilerrors . NewAggregate ( allErrs )
}
2019-04-07 17:07:55 +00:00
func setImage ( containers [ ] v1 . Container , containerName string , image string ) bool {
containerFound := false
// Find the container to update, and update its image
for i , c := range containers {
if c . Name == containerName || containerName == "*" {
containerFound = true
containers [ i ] . Image = image
}
}
return containerFound
}
2019-01-12 04:58:27 +00:00
// getResourcesAndImages retrieves resources and container name:images pair from given args
func getResourcesAndImages ( args [ ] string ) ( resources [ ] string , containerImages map [ string ] string , err error ) {
pairType := "image"
resources , imageArgs , err := cmdutil . GetResourcesAndPairs ( args , pairType )
if err != nil {
return
}
containerImages , _ , err = cmdutil . ParsePairs ( imageArgs , pairType , false )
return
}
func hasWildcardKey ( containerImages map [ string ] string ) bool {
_ , ok := containerImages [ "*" ]
return ok
}
// ImageResolver is a func that receives an image name, and
// resolves it to an appropriate / compatible image name.
// Adds flexibility for future image resolving methods.
type ImageResolver func ( in string ) ( string , error )
// implements ImageResolver
func resolveImageFunc ( in string ) ( string , error ) {
return in , nil
}