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 create
import (
2021-03-18 22:40:29 +00:00
"context"
"fmt"
"io/ioutil"
"os"
"path"
"strings"
"unicode/utf8"
2019-01-12 04:58:27 +00:00
"github.com/spf13/cobra"
2021-03-18 22:40:29 +00:00
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/util/validation"
2019-01-12 04:58:27 +00:00
"k8s.io/cli-runtime/pkg/genericclioptions"
2021-03-18 22:40:29 +00:00
"k8s.io/cli-runtime/pkg/resource"
corev1client "k8s.io/client-go/kubernetes/typed/core/v1"
2019-09-27 21:51:53 +00:00
cmdutil "k8s.io/kubectl/pkg/cmd/util"
2021-03-18 22:40:29 +00:00
"k8s.io/kubectl/pkg/scheme"
"k8s.io/kubectl/pkg/util"
"k8s.io/kubectl/pkg/util/hash"
2019-09-27 21:51:53 +00:00
"k8s.io/kubectl/pkg/util/i18n"
"k8s.io/kubectl/pkg/util/templates"
2019-01-12 04:58:27 +00:00
)
var (
configMapLong = templates . LongDesc ( i18n . T ( `
2021-07-02 08:43:15 +00:00
Create a config map based on a file , directory , or specified literal value .
2019-01-12 04:58:27 +00:00
2021-07-02 08:43:15 +00:00
A single config map may package one or more key / value pairs .
2019-01-12 04:58:27 +00:00
2021-07-02 08:43:15 +00:00
When creating a config map based on a file , the key will default to the basename of the file , and the value will
2019-01-12 04:58:27 +00:00
default to the file content . If the basename is an invalid key , you may specify an alternate key .
2021-07-02 08:43:15 +00:00
When creating a config map based on a directory , each file whose basename is a valid key in the directory will be
packaged into the config map . Any directory entries except regular files are ignored ( e . g . subdirectories ,
2019-01-12 04:58:27 +00:00
symlinks , devices , pipes , etc ) . ` ) )
configMapExample = templates . Examples ( i18n . T ( `
2021-07-02 08:43:15 +00:00
# Create a new config map named my - config based on folder bar
2019-01-12 04:58:27 +00:00
kubectl create configmap my - config -- from - file = path / to / bar
2021-07-02 08:43:15 +00:00
# Create a new config map named my - config with specified keys instead of file basenames on disk
2019-01-12 04:58:27 +00:00
kubectl create configmap my - config -- from - file = key1 = / path / to / bar / file1 . txt -- from - file = key2 = / path / to / bar / file2 . txt
2021-07-02 08:43:15 +00:00
# Create a new config map named my - config with key1 = config1 and key2 = config2
2019-01-12 04:58:27 +00:00
kubectl create configmap my - config -- from - literal = key1 = config1 -- from - literal = key2 = config2
2021-07-02 08:43:15 +00:00
# Create a new config map named my - config from the key = value pairs in the file
2019-01-12 04:58:27 +00:00
kubectl create configmap my - config -- from - file = path / to / bar
2021-07-02 08:43:15 +00:00
# Create a new config map named my - config from an env file
2019-01-12 04:58:27 +00:00
kubectl create configmap my - config -- from - env - file = path / to / bar . env ` ) )
)
2021-03-18 22:40:29 +00:00
// ConfigMapOptions holds properties for create configmap sub-command
type ConfigMapOptions struct {
// PrintFlags holds options necessary for obtaining a printer
PrintFlags * genericclioptions . PrintFlags
PrintObj func ( obj runtime . Object ) error
// Name of configMap (required)
Name string
// Type of configMap (optional)
Type string
// FileSources to derive the configMap from (optional)
FileSources [ ] string
// LiteralSources to derive the configMap from (optional)
LiteralSources [ ] string
// EnvFileSource to derive the configMap from (optional)
EnvFileSource string
// AppendHash; if true, derive a hash from the ConfigMap and append it to the name
AppendHash bool
FieldManager string
CreateAnnotation bool
Namespace string
EnforceNamespace bool
Client corev1client . CoreV1Interface
DryRunStrategy cmdutil . DryRunStrategy
DryRunVerifier * resource . DryRunVerifier
genericclioptions . IOStreams
2019-01-12 04:58:27 +00:00
}
2021-03-18 22:40:29 +00:00
// NewConfigMapOptions creates a new *ConfigMapOptions with default value
func NewConfigMapOptions ( ioStreams genericclioptions . IOStreams ) * ConfigMapOptions {
return & ConfigMapOptions {
FileSources : [ ] string { } ,
LiteralSources : [ ] string { } ,
PrintFlags : genericclioptions . NewPrintFlags ( "created" ) . WithTypeSetter ( scheme . Scheme ) ,
IOStreams : ioStreams ,
2019-01-12 04:58:27 +00:00
}
2021-03-18 22:40:29 +00:00
}
// NewCmdCreateConfigMap creates the `create configmap` Cobra command
func NewCmdCreateConfigMap ( f cmdutil . Factory , ioStreams genericclioptions . IOStreams ) * cobra . Command {
o := NewConfigMapOptions ( ioStreams )
2019-01-12 04:58:27 +00:00
cmd := & cobra . Command {
2020-03-26 21:07:15 +00:00
Use : "configmap NAME [--from-file=[key=]source] [--from-literal=key1=value1] [--dry-run=server|client|none]" ,
2019-01-12 04:58:27 +00:00
DisableFlagsInUseLine : true ,
Aliases : [ ] string { "cm" } ,
2021-07-02 08:43:15 +00:00
Short : i18n . T ( "Create a config map from a local file, directory or literal value" ) ,
2019-01-12 04:58:27 +00:00
Long : configMapLong ,
Example : configMapExample ,
Run : func ( cmd * cobra . Command , args [ ] string ) {
2021-03-18 22:40:29 +00:00
cmdutil . CheckErr ( o . Complete ( f , cmd , args ) )
cmdutil . CheckErr ( o . Validate ( ) )
cmdutil . CheckErr ( o . Run ( ) )
2019-01-12 04:58:27 +00:00
} ,
}
2021-03-18 22:40:29 +00:00
o . PrintFlags . AddFlags ( cmd )
2019-01-12 04:58:27 +00:00
cmdutil . AddApplyAnnotationFlags ( cmd )
cmdutil . AddValidateFlags ( cmd )
2021-03-18 22:40:29 +00:00
cmdutil . AddDryRunFlag ( cmd )
cmd . Flags ( ) . StringSliceVar ( & o . FileSources , "from-file" , o . FileSources , "Key file can be specified using its file path, in which case file basename will be used as configmap key, or optionally with a key and file path, in which case the given key will be used. Specifying a directory will iterate each named file in the directory whose basename is a valid configmap key." )
cmd . Flags ( ) . StringArrayVar ( & o . LiteralSources , "from-literal" , o . LiteralSources , "Specify a key and literal value to insert in configmap (i.e. mykey=somevalue)" )
cmd . Flags ( ) . StringVar ( & o . EnvFileSource , "from-env-file" , o . EnvFileSource , "Specify the path to a file to read lines of key=val pairs to create a configmap (i.e. a Docker .env file)." )
cmd . Flags ( ) . BoolVar ( & o . AppendHash , "append-hash" , o . AppendHash , "Append a hash of the configmap to its name." )
cmdutil . AddFieldManagerFlagVar ( cmd , & o . FieldManager , "kubectl-create" )
2019-01-12 04:58:27 +00:00
return cmd
}
2021-03-18 22:40:29 +00:00
// Complete loads data from the command line environment
func ( o * ConfigMapOptions ) Complete ( f cmdutil . Factory , cmd * cobra . Command , args [ ] string ) error {
var err error
o . Name , err = NameFromCommandArgs ( cmd , args )
2019-01-12 04:58:27 +00:00
if err != nil {
return err
}
2021-03-18 22:40:29 +00:00
restConfig , err := f . ToRESTConfig ( )
if err != nil {
return err
}
o . Client , err = corev1client . NewForConfig ( restConfig )
if err != nil {
return err
}
o . CreateAnnotation = cmdutil . GetFlagBool ( cmd , cmdutil . ApplyAnnotationsFlag )
o . DryRunStrategy , err = cmdutil . GetDryRunStrategy ( cmd )
if err != nil {
return err
}
dynamicClient , err := f . DynamicClient ( )
if err != nil {
return err
}
discoveryClient , err := f . ToDiscoveryClient ( )
if err != nil {
return err
}
o . DryRunVerifier = resource . NewDryRunVerifier ( dynamicClient , discoveryClient )
o . Namespace , o . EnforceNamespace , err = f . ToRawKubeConfigLoader ( ) . Namespace ( )
if err != nil {
return err
}
cmdutil . PrintFlagsWithDryRunStrategy ( o . PrintFlags , o . DryRunStrategy )
printer , err := o . PrintFlags . ToPrinter ( )
if err != nil {
return err
}
o . PrintObj = func ( obj runtime . Object ) error {
return printer . PrintObj ( obj , o . Out )
}
return nil
}
// Validate checks if ConfigMapOptions has sufficient value to run
func ( o * ConfigMapOptions ) Validate ( ) error {
if len ( o . Name ) == 0 {
return fmt . Errorf ( "name must be specified" )
}
if len ( o . EnvFileSource ) > 0 && ( len ( o . FileSources ) > 0 || len ( o . LiteralSources ) > 0 ) {
return fmt . Errorf ( "from-env-file cannot be combined with from-file or from-literal" )
}
return nil
}
// Run calls createConfigMap and filled in value for configMap object
func ( o * ConfigMapOptions ) Run ( ) error {
configMap , err := o . createConfigMap ( )
if err != nil {
2021-07-02 08:43:15 +00:00
return err
2021-03-18 22:40:29 +00:00
}
if err := util . CreateOrUpdateAnnotation ( o . CreateAnnotation , configMap , scheme . DefaultJSONEncoder ( ) ) ; err != nil {
return err
}
if o . DryRunStrategy != cmdutil . DryRunClient {
createOptions := metav1 . CreateOptions { }
if o . FieldManager != "" {
createOptions . FieldManager = o . FieldManager
}
if o . DryRunStrategy == cmdutil . DryRunServer {
if err := o . DryRunVerifier . HasSupport ( configMap . GroupVersionKind ( ) ) ; err != nil {
return err
}
createOptions . DryRun = [ ] string { metav1 . DryRunAll }
}
configMap , err = o . Client . ConfigMaps ( o . Namespace ) . Create ( context . TODO ( ) , configMap , createOptions )
if err != nil {
return fmt . Errorf ( "failed to create configmap: %v" , err )
}
}
return o . PrintObj ( configMap )
}
// createConfigMap fills in key value pair from the information given in
// ConfigMapOptions into *corev1.ConfigMap
func ( o * ConfigMapOptions ) createConfigMap ( ) ( * corev1 . ConfigMap , error ) {
namespace := ""
if o . EnforceNamespace {
namespace = o . Namespace
}
configMap := & corev1 . ConfigMap {
TypeMeta : metav1 . TypeMeta {
APIVersion : corev1 . SchemeGroupVersion . String ( ) ,
Kind : "ConfigMap" ,
} ,
ObjectMeta : metav1 . ObjectMeta {
Name : o . Name ,
Namespace : namespace ,
} ,
}
configMap . Name = o . Name
configMap . Data = map [ string ] string { }
configMap . BinaryData = map [ string ] [ ] byte { }
if len ( o . FileSources ) > 0 {
if err := handleConfigMapFromFileSources ( configMap , o . FileSources ) ; err != nil {
return nil , err
}
}
if len ( o . LiteralSources ) > 0 {
if err := handleConfigMapFromLiteralSources ( configMap , o . LiteralSources ) ; err != nil {
return nil , err
}
}
if len ( o . EnvFileSource ) > 0 {
if err := handleConfigMapFromEnvFileSource ( configMap , o . EnvFileSource ) ; err != nil {
return nil , err
}
}
if o . AppendHash {
hash , err := hash . ConfigMapHash ( configMap )
if err != nil {
return nil , err
}
configMap . Name = fmt . Sprintf ( "%s-%s" , configMap . Name , hash )
}
return configMap , nil
}
// handleConfigMapFromLiteralSources adds the specified literal source
// information into the provided configMap.
func handleConfigMapFromLiteralSources ( configMap * corev1 . ConfigMap , literalSources [ ] string ) error {
for _ , literalSource := range literalSources {
keyName , value , err := util . ParseLiteralSource ( literalSource )
if err != nil {
return err
}
err = addKeyFromLiteralToConfigMap ( configMap , keyName , value )
if err != nil {
return err
2019-01-12 04:58:27 +00:00
}
}
2021-03-18 22:40:29 +00:00
return nil
2019-01-12 04:58:27 +00:00
}
2021-03-18 22:40:29 +00:00
// handleConfigMapFromFileSources adds the specified file source information
// into the provided configMap
func handleConfigMapFromFileSources ( configMap * corev1 . ConfigMap , fileSources [ ] string ) error {
for _ , fileSource := range fileSources {
keyName , filePath , err := util . ParseFileSource ( fileSource )
if err != nil {
return err
}
info , err := os . Stat ( filePath )
if err != nil {
switch err := err . ( type ) {
case * os . PathError :
return fmt . Errorf ( "error reading %s: %v" , filePath , err . Err )
default :
return fmt . Errorf ( "error reading %s: %v" , filePath , err )
}
}
if info . IsDir ( ) {
if strings . Contains ( fileSource , "=" ) {
return fmt . Errorf ( "cannot give a key name for a directory path" )
}
fileList , err := ioutil . ReadDir ( filePath )
if err != nil {
return fmt . Errorf ( "error listing files in %s: %v" , filePath , err )
}
for _ , item := range fileList {
itemPath := path . Join ( filePath , item . Name ( ) )
if item . Mode ( ) . IsRegular ( ) {
keyName = item . Name ( )
err = addKeyFromFileToConfigMap ( configMap , keyName , itemPath )
if err != nil {
return err
}
}
}
} else {
if err := addKeyFromFileToConfigMap ( configMap , keyName , filePath ) ; err != nil {
return err
}
}
}
return nil
}
// handleConfigMapFromEnvFileSource adds the specified env file source information
// into the provided configMap
func handleConfigMapFromEnvFileSource ( configMap * corev1 . ConfigMap , envFileSource string ) error {
info , err := os . Stat ( envFileSource )
if err != nil {
switch err := err . ( type ) {
case * os . PathError :
return fmt . Errorf ( "error reading %s: %v" , envFileSource , err . Err )
default :
return fmt . Errorf ( "error reading %s: %v" , envFileSource , err )
}
}
if info . IsDir ( ) {
return fmt . Errorf ( "env config file cannot be a directory" )
}
return cmdutil . AddFromEnvFile ( envFileSource , func ( key , value string ) error {
return addKeyFromLiteralToConfigMap ( configMap , key , value )
} )
}
// addKeyFromFileToConfigMap adds a key with the given name to a ConfigMap, populating
// the value with the content of the given file path, or returns an error.
func addKeyFromFileToConfigMap ( configMap * corev1 . ConfigMap , keyName , filePath string ) error {
data , err := ioutil . ReadFile ( filePath )
if err != nil {
return err
}
if utf8 . Valid ( data ) {
return addKeyFromLiteralToConfigMap ( configMap , keyName , string ( data ) )
}
err = validateNewConfigMap ( configMap , keyName )
if err != nil {
return err
}
configMap . BinaryData [ keyName ] = data
return nil
}
// addKeyFromLiteralToConfigMap adds the given key and data to the given config map,
// returning an error if the key is not valid or if the key already exists.
func addKeyFromLiteralToConfigMap ( configMap * corev1 . ConfigMap , keyName , data string ) error {
err := validateNewConfigMap ( configMap , keyName )
if err != nil {
return err
}
configMap . Data [ keyName ] = data
return nil
}
// validateNewConfigMap checks whether the keyname is valid
// Note, the rules for ConfigMap keys are the exact same as the ones for SecretKeys.
func validateNewConfigMap ( configMap * corev1 . ConfigMap , keyName string ) error {
if errs := validation . IsConfigMapKey ( keyName ) ; len ( errs ) > 0 {
return fmt . Errorf ( "%q is not a valid key name for a ConfigMap: %s" , keyName , strings . Join ( errs , "," ) )
}
if _ , exists := configMap . Data [ keyName ] ; exists {
return fmt . Errorf ( "cannot add key %q, another key by that name already exists in Data for ConfigMap %q" , keyName , configMap . Name )
}
if _ , exists := configMap . BinaryData [ keyName ] ; exists {
return fmt . Errorf ( "cannot add key %q, another key by that name already exists in BinaryData for ConfigMap %q" , keyName , configMap . Name )
}
return nil
2019-01-12 04:58:27 +00:00
}