2021-08-03 18:00:49 +08:00
/ *
Copyright 2021 The KubeVela 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 cli
import (
"bufio"
2021-08-06 17:10:52 +08:00
"bytes"
2021-08-03 18:00:49 +08:00
"context"
"fmt"
"os"
"os/exec"
"path"
"path/filepath"
"regexp"
"strings"
"time"
"cuelang.org/go/cue"
"cuelang.org/go/encoding/gocode/gocodec"
"github.com/pkg/errors"
"github.com/spf13/cobra"
2021-08-06 17:10:52 +08:00
"gopkg.in/yaml.v3"
2021-08-03 18:00:49 +08:00
errors2 "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
types2 "k8s.io/apimachinery/pkg/types"
"sigs.k8s.io/controller-runtime/pkg/client"
"github.com/oam-dev/kubevela/apis/types"
2021-08-30 11:43:20 +08:00
"github.com/oam-dev/kubevela/pkg/cue/model"
2021-08-03 18:00:49 +08:00
"github.com/oam-dev/kubevela/pkg/cue/model/sets"
"github.com/oam-dev/kubevela/pkg/utils/common"
common2 "github.com/oam-dev/kubevela/references/common"
)
const (
2021-08-06 17:10:52 +08:00
// HelmChartNamespacePlaceholder is used as a placeholder for rendering definitions into helm chart format
HelmChartNamespacePlaceholder = "###HELM_NAMESPACE###"
// HelmChartFormatEnvName is the name of the environment variable to enable render helm chart format YAML
HelmChartFormatEnvName = "AS_HELM_CHART"
2021-08-03 18:00:49 +08:00
)
// DefinitionCommandGroup create the command group for `vela def` command to manage definitions
func DefinitionCommandGroup ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "def" ,
Short : "Manage Definitions" ,
Long : "Manage Definitions" ,
Annotations : map [ string ] string {
types . TagCommandType : types . TypeCap ,
} ,
}
2021-08-13 18:44:01 +08:00
_ = c . SetConfig ( ) // set kubeConfig if possible, otherwise ignore it
2021-08-03 18:00:49 +08:00
cmd . AddCommand (
NewDefinitionGetCommand ( c ) ,
NewDefinitionListCommand ( c ) ,
NewDefinitionEditCommand ( c ) ,
2021-08-06 17:10:52 +08:00
NewDefinitionRenderCommand ( c ) ,
2021-08-03 18:00:49 +08:00
NewDefinitionApplyCommand ( c ) ,
NewDefinitionDelCommand ( c ) ,
NewDefinitionInitCommand ( c ) ,
NewDefinitionValidateCommand ( c ) ,
)
return cmd
}
func getPrompt ( cmd * cobra . Command , reader * bufio . Reader , description string , prompt string , validate func ( string ) error ) ( string , error ) {
cmd . Printf ( description )
for {
cmd . Printf ( prompt )
resp , err := reader . ReadString ( '\n' )
resp = strings . TrimSpace ( resp )
if err != nil {
return "" , errors . Wrapf ( err , "failed to read user response" )
}
if validate == nil {
return resp , nil
}
err = validate ( resp )
if err != nil {
cmd . Println ( err )
} else {
return resp , nil
}
}
}
func loadYAMLBytesFromFileOrHTTP ( pathOrURL string ) ( [ ] byte , error ) {
if strings . HasPrefix ( pathOrURL , "http://" ) || strings . HasPrefix ( pathOrURL , "https://" ) {
return common . HTTPGet ( context . Background ( ) , pathOrURL )
}
2021-09-06 18:33:42 +08:00
return os . ReadFile ( path . Clean ( pathOrURL ) )
2021-08-03 18:00:49 +08:00
}
func buildTemplateFromYAML ( templateYAML string , def * common2 . Definition ) error {
templateYAMLBytes , err := loadYAMLBytesFromFileOrHTTP ( templateYAML )
if err != nil {
return errors . Wrapf ( err , "failed to get template YAML file %s" , templateYAML )
}
yamlStrings := regexp . MustCompile ( ` \n---[^\n]*\n ` ) . Split ( string ( templateYAMLBytes ) , - 1 )
templateObject := map [ string ] interface { } {
2021-08-30 11:43:20 +08:00
model . OutputFieldName : map [ string ] interface { } { } ,
model . OutputsFieldName : map [ string ] interface { } { } ,
model . ParameterFieldName : map [ string ] interface { } { } ,
2021-08-03 18:00:49 +08:00
}
for index , yamlString := range yamlStrings {
var yamlObject map [ string ] interface { }
if err = yaml . Unmarshal ( [ ] byte ( yamlString ) , & yamlObject ) ; err != nil {
return errors . Wrapf ( err , "failed to unmarshal template yaml file" )
}
if index == 0 {
2021-08-30 11:43:20 +08:00
templateObject [ model . OutputFieldName ] = yamlObject
2021-08-03 18:00:49 +08:00
} else {
name , _ , _ := unstructured . NestedString ( yamlObject , "metadata" , "name" )
if name == "" {
name = fmt . Sprintf ( "output-%d" , index )
}
2021-08-30 11:43:20 +08:00
templateObject [ model . OutputsFieldName ] . ( map [ string ] interface { } ) [ name ] = yamlObject
2021-08-03 18:00:49 +08:00
}
}
codec := gocodec . New ( & cue . Runtime { } , & gocodec . Config { } )
val , err := codec . Decode ( templateObject )
if err != nil {
return errors . Wrapf ( err , "failed to decode template into cue" )
}
templateString , err := sets . ToString ( val )
if err != nil {
return errors . Wrapf ( err , "failed to encode template cue string" )
}
err = unstructured . SetNestedField ( def . Object , templateString , common2 . DefinitionTemplateKeys ... )
if err != nil {
return errors . Wrapf ( err , "failed to merge template cue string" )
}
return nil
}
// NewDefinitionInitCommand create the `vela def init` command to help user initialize a definition locally
func NewDefinitionInitCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "init DEF_NAME" ,
Short : "Init a new definition" ,
Long : "Init a new definition with given arguments or interactively\n* We support parsing a single YAML file (like kubernetes objects) into the cue-style template. However, we do not support variables in YAML file currently, which prevents users from directly feeding files like helm chart directly. We may introduce such features in the future." ,
Example : "# Command below initiate an empty TraitDefinition named my-ingress\n" +
"> vela def init my-ingress -t trait --desc \"My ingress trait definition.\" > ./my-ingress.cue\n" +
"# Command below initiate a definition named my-def interactively and save it to ./my-def.cue\n" +
"> vela def init my-def -i --output ./my-def.cue\n" +
"# Command below initiate a ComponentDefinition named my-webservice with the template parsed from ./template.yaml.\n" +
"> vela def init my-webservice -i --template-yaml ./template.yaml" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
definitionType , err := cmd . Flags ( ) . GetString ( FlagType )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagType )
}
desc , err := cmd . Flags ( ) . GetString ( FlagDescription )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagDescription )
}
templateYAML , err := cmd . Flags ( ) . GetString ( FlagTemplateYAML )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagTemplateYAML )
}
output , err := cmd . Flags ( ) . GetString ( FlagOutput )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagOutput )
}
interactive , err := cmd . Flags ( ) . GetBool ( FlagInteractive )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagInteractive )
}
if interactive {
reader := bufio . NewReader ( cmd . InOrStdin ( ) )
if definitionType == "" {
if definitionType , err = getPrompt ( cmd , reader , "Please choose one definition type from the following values: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) + "\n" , "> Definition type: " , func ( resp string ) error {
if _ , ok := common2 . DefinitionTypeToKind [ resp ] ; ! ok {
return errors . New ( "invalid definition type" )
}
return nil
} ) ; err != nil {
return err
}
}
if desc == "" {
if desc , err = getPrompt ( cmd , reader , "" , "> Definition description: " , nil ) ; err != nil {
return err
}
}
if templateYAML == "" {
if templateYAML , err = getPrompt ( cmd , reader , "Please enter the location the template YAML file to build definition. Leave it empty to generate default template.\n" , "> Definition template filename: " , func ( resp string ) error {
if resp == "" {
return nil
}
_ , err = os . Stat ( resp )
return err
} ) ; err != nil {
return err
}
}
if output == "" {
if output , err = getPrompt ( cmd , reader , "Please enter the output location of the generated definition. Leave it empty to print definition to stdout.\n" , "> Definition output filename: " , nil ) ; err != nil {
return err
}
}
}
kind , ok := common2 . DefinitionTypeToKind [ definitionType ]
if ! ok {
return errors . New ( "invalid definition type" )
}
def := common2 . Definition { Unstructured : unstructured . Unstructured { } }
def . SetGVK ( kind )
def . SetName ( args [ 0 ] )
def . SetAnnotations ( map [ string ] string {
common2 . DefinitionDescriptionKey : desc ,
} )
def . SetLabels ( map [ string ] string { } )
def . Object [ "spec" ] = common2 . GetDefinitionDefaultSpec ( def . GetKind ( ) )
if templateYAML != "" {
if err = buildTemplateFromYAML ( templateYAML , & def ) ; err != nil {
return err
}
}
cueString , err := def . ToCUEString ( )
if err != nil {
return errors . Wrapf ( err , "failed to generate cue string" )
}
if output != "" {
2021-09-06 18:33:42 +08:00
if err = os . WriteFile ( path . Clean ( output ) , [ ] byte ( cueString ) , 0600 ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to write definition into %s" , output )
}
cmd . Printf ( "Definition written to %s\n" , output )
} else if _ , err = cmd . OutOrStdout ( ) . Write ( [ ] byte ( cueString + "\n" ) ) ; err != nil {
return errors . Wrapf ( err , "failed to write out cue string" )
}
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagType , "t" , "" , "Specify the type of the new definition. Valid types: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) )
cmd . Flags ( ) . StringP ( FlagDescription , "d" , "" , "Specify the description of the new definition." )
cmd . Flags ( ) . StringP ( FlagTemplateYAML , "y" , "" , "Specify the template yaml file that definition will use to build the schema. If empty, a default template for the given definition type will be used." )
cmd . Flags ( ) . StringP ( FlagOutput , "o" , "" , "Specify the output path of the generated definition. If empty, the definition will be printed in the console." )
cmd . Flags ( ) . BoolP ( FlagInteractive , "i" , false , "Specify whether use interactive process to help generate definitions." )
return cmd
}
func getSingleDefinition ( cmd * cobra . Command , definitionName string , client client . Client , definitionType string , namespace string ) ( * common2 . Definition , error ) {
definitions , err := common2 . SearchDefinition ( definitionName , client , definitionType , namespace )
if err != nil {
return nil , err
}
if len ( definitions ) == 0 {
return nil , fmt . Errorf ( "definition not found" )
}
if len ( definitions ) > 1 {
table := newUITable ( )
table . AddRow ( "NAME" , "TYPE" , "NAMESPACE" , "DESCRIPTION" )
for _ , definition := range definitions {
desc := ""
if annotations := definition . GetAnnotations ( ) ; annotations != nil {
desc = annotations [ common2 . DefinitionDescriptionKey ]
}
table . AddRow ( definition . GetName ( ) , definition . GetKind ( ) , definition . GetNamespace ( ) , desc )
}
cmd . Println ( table )
return nil , fmt . Errorf ( "found %d definitions, please specify which one to select with more arguments" , len ( definitions ) )
}
return & common2 . Definition { Unstructured : definitions [ 0 ] } , nil
}
// NewDefinitionGetCommand create the `vela def get` command to get definition from k8s
func NewDefinitionGetCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "get NAME" ,
Short : "Get definition" ,
Long : "Get definition from kubernetes cluster" ,
Example : "# Command below will get the ComponentDefinition(or other definitions if exists) of webservice in all namespaces\n" +
"> vela def get webservice\n" +
"# Command below will get the TraitDefinition of annotations in namespace vela-system\n" +
"> vela def get annotations --type trait --namespace vela-system" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
definitionType , err := cmd . Flags ( ) . GetString ( FlagType )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagType )
}
namespace , err := cmd . Flags ( ) . GetString ( FlagNamespace )
if err != nil {
2021-11-19 18:00:03 +08:00
return errors . Wrapf ( err , "failed to get `%s`" , Namespace )
2021-08-03 18:00:49 +08:00
}
k8sClient , err := c . GetClient ( )
if err != nil {
return errors . Wrapf ( err , "failed to get k8s client" )
}
def , err := getSingleDefinition ( cmd , args [ 0 ] , k8sClient , definitionType , namespace )
if err != nil {
return err
}
cueString , err := def . ToCUEString ( )
if err != nil {
return errors . Wrapf ( err , "failed to get cue format definition" )
}
if _ , err = cmd . OutOrStdout ( ) . Write ( [ ] byte ( cueString + "\n" ) ) ; err != nil {
return errors . Wrapf ( err , "failed to write out cue string" )
}
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagType , "t" , "" , "Specify which definition type to get. If empty, all types will be searched. Valid types: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) )
2021-11-19 18:00:03 +08:00
cmd . Flags ( ) . StringP ( Namespace , "n" , "" , "Specify which namespace to get. If empty, all namespaces will be searched." )
2021-08-03 18:00:49 +08:00
return cmd
}
// NewDefinitionListCommand create the `vela def list` command to list definition from k8s
func NewDefinitionListCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "list" ,
Short : "List definitions" ,
Long : "List definitions in kubernetes cluster" ,
Example : "# Command below will list all definitions in all namespaces\n" +
"> vela def list\n" +
"# Command below will list all definitions in the vela-system namespace\n" +
"> vela def get annotations --type trait --namespace vela-system" ,
Args : cobra . ExactValidArgs ( 0 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
definitionType , err := cmd . Flags ( ) . GetString ( FlagType )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagType )
}
namespace , err := cmd . Flags ( ) . GetString ( FlagNamespace )
if err != nil {
2021-11-19 18:00:03 +08:00
return errors . Wrapf ( err , "failed to get `%s`" , Namespace )
2021-08-03 18:00:49 +08:00
}
k8sClient , err := c . GetClient ( )
if err != nil {
return errors . Wrapf ( err , "failed to get k8s client" )
}
definitions , err := common2 . SearchDefinition ( "*" , k8sClient , definitionType , namespace )
if err != nil {
return err
}
if len ( definitions ) == 0 {
cmd . Println ( "No definition found." )
return nil
}
table := newUITable ( )
table . AddRow ( "NAME" , "TYPE" , "NAMESPACE" , "DESCRIPTION" )
for _ , definition := range definitions {
desc := ""
if annotations := definition . GetAnnotations ( ) ; annotations != nil {
desc = annotations [ common2 . DefinitionDescriptionKey ]
}
table . AddRow ( definition . GetName ( ) , definition . GetKind ( ) , definition . GetNamespace ( ) , desc )
}
cmd . Println ( table )
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagType , "t" , "" , "Specify which definition type to list. If empty, all types will be searched. Valid types: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) )
2021-11-19 18:00:03 +08:00
cmd . Flags ( ) . StringP ( Namespace , "n" , "" , "Specify which namespace to list. If empty, all namespaces will be searched." )
2021-08-03 18:00:49 +08:00
return cmd
}
// NewDefinitionEditCommand create the `vela def edit` command to help user edit remote definitions
func NewDefinitionEditCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "edit NAME" ,
Short : "Edit definition" ,
Long : "Edit definition in kubernetes. If type and namespace are not specified, the command will automatically search all possible results.\n" +
"By default, this command will use the vi editor and can be altered by setting EDITOR environment variable." ,
Example : "# Command below will edit the ComponentDefinition (and other definitions if exists) of webservice in kubernetes\n" +
"> vela def edit webservice\n" +
"# Command below will edit the TraitDefinition of ingress in vela-system namespace\n" +
"> vela def edit ingress --type trait --namespace vela-system" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
definitionType , err := cmd . Flags ( ) . GetString ( FlagType )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagType )
}
namespace , err := cmd . Flags ( ) . GetString ( FlagNamespace )
if err != nil {
2021-11-19 18:00:03 +08:00
return errors . Wrapf ( err , "failed to get `%s`" , Namespace )
2021-08-03 18:00:49 +08:00
}
k8sClient , err := c . GetClient ( )
if err != nil {
return errors . Wrapf ( err , "failed to get k8s client" )
}
def , err := getSingleDefinition ( cmd , args [ 0 ] , k8sClient , definitionType , namespace )
if err != nil {
return err
}
cueString , err := def . ToCUEString ( )
if err != nil {
return errors . Wrapf ( err , "failed to get cue format definition" )
}
cleanup := func ( filePath string ) {
if err := os . Remove ( filePath ) ; err != nil {
cmd . PrintErrf ( "failed to remove file %s: %v" , filePath , err )
}
}
filename := fmt . Sprintf ( "vela-def-%d" , time . Now ( ) . UnixNano ( ) )
tempFilePath := filepath . Join ( os . TempDir ( ) , filename + ".cue" )
2021-09-06 18:33:42 +08:00
if err := os . WriteFile ( tempFilePath , [ ] byte ( cueString ) , 0600 ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to write temporary file" )
}
defer cleanup ( tempFilePath )
editor := os . Getenv ( "EDITOR" )
if editor == "" {
editor = "vi"
}
scriptFilePath := filepath . Join ( os . TempDir ( ) , filename + ".sh" )
2021-09-06 18:33:42 +08:00
if err := os . WriteFile ( scriptFilePath , [ ] byte ( editor + " " + tempFilePath ) , 0600 ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to write temporary script file" )
}
defer cleanup ( scriptFilePath )
editCmd := exec . Command ( "sh" , path . Clean ( scriptFilePath ) ) //nolint:gosec
editCmd . Stdin = os . Stdin
editCmd . Stdout = os . Stdout
editCmd . Stderr = os . Stderr
if err = editCmd . Run ( ) ; err != nil {
return errors . Wrapf ( err , "failed to run editor %s at path %s" , editor , scriptFilePath )
}
2021-09-06 18:33:42 +08:00
newBuf , err := os . ReadFile ( path . Clean ( tempFilePath ) )
2021-08-03 18:00:49 +08:00
if err != nil {
return errors . Wrapf ( err , "failed to read temporary file %s" , tempFilePath )
}
if cueString == string ( newBuf ) {
cmd . Printf ( "definition unchanged\n" )
return nil
}
2021-08-13 18:44:01 +08:00
if err := def . FromCUEString ( string ( newBuf ) , c . Config ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to load edited cue string" )
}
if err := k8sClient . Update ( context . Background ( ) , def ) ; err != nil {
return errors . Wrapf ( err , "failed to apply changes to kubernetes" )
}
cmd . Printf ( "Definition edited successfully.\n" )
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagType , "t" , "" , "Specify which definition type to get. If empty, all types will be searched. Valid types: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) )
2021-11-19 18:00:03 +08:00
cmd . Flags ( ) . StringP ( Namespace , "n" , "" , "Specify which namespace to get. If empty, all namespaces will be searched." )
2021-08-03 18:00:49 +08:00
return cmd
}
2021-08-06 17:10:52 +08:00
func prettyYAMLMarshal ( obj map [ string ] interface { } ) ( string , error ) {
var b bytes . Buffer
encoder := yaml . NewEncoder ( & b )
encoder . SetIndent ( 2 )
err := encoder . Encode ( & obj )
if err != nil {
return "" , err
}
return b . String ( ) , nil
}
// NewDefinitionRenderCommand create the `vela def render` command to help user render definition cue file into k8s YAML file, if used without kubernetes environment, set IGNORE_KUBE_CONFIG=true
func NewDefinitionRenderCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "render DEFINITION.cue" ,
Short : "Render definition" ,
Long : "Render definition with cue format into kubernetes YAML format. Could be used to check whether the cue format definition is working as expected. If a directory is used as input, all cue definitions in the directory will be rendered." ,
Example : "# Command below will render my-webservice.cue into YAML format and print it out.\n" +
"> vela def render my-webservice.cue\n" +
"# Command below will render my-webservice.cue and save it in my-webservice.yaml.\n" +
"> vela def render my-webservice.cue -o my-webservice.yaml" +
"# Command below will render all cue format definitions in the ./defs/cue/ directory and save the YAML objects in ./defs/yaml/.\n" +
"> vela def render ./defs/cue/ -o ./defs/yaml/" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
output , err := cmd . Flags ( ) . GetString ( FlagOutput )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagOutput )
}
2021-08-12 19:10:51 +08:00
message , err := cmd . Flags ( ) . GetString ( FlagMessage )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagMessage )
}
2021-08-06 17:10:52 +08:00
render := func ( inputFilename , outputFilename string ) error {
cueBytes , err := loadYAMLBytesFromFileOrHTTP ( inputFilename )
if err != nil {
return errors . Wrapf ( err , "failed to get %s" , args [ 0 ] )
}
def := common2 . Definition { Unstructured : unstructured . Unstructured { } }
2021-08-13 18:44:01 +08:00
if err := def . FromCUEString ( string ( cueBytes ) , c . Config ) ; err != nil {
2021-08-06 17:10:52 +08:00
return errors . Wrapf ( err , "failed to parse CUE" )
}
helmChartFormatEnv := strings . ToLower ( os . Getenv ( HelmChartFormatEnvName ) )
if helmChartFormatEnv == "true" {
def . SetNamespace ( HelmChartNamespacePlaceholder )
} else if helmChartFormatEnv == "system" {
2021-08-30 11:43:20 +08:00
def . SetNamespace ( types . DefaultKubeVelaNS )
2021-08-06 17:10:52 +08:00
}
if len ( def . GetLabels ( ) ) == 0 {
def . SetLabels ( nil )
}
s , err := prettyYAMLMarshal ( def . Object )
if err != nil {
return errors . Wrapf ( err , "failed to marshal CRD into YAML" )
}
s = strings . ReplaceAll ( s , "'" + HelmChartNamespacePlaceholder + "'" , "{{.Values.systemDefinitionNamespace}}" ) + "\n"
if outputFilename == "" {
s = fmt . Sprintf ( "--- %s ---\n%s" , filepath . Base ( inputFilename ) , s )
cmd . Print ( s )
} else {
2021-08-12 19:10:51 +08:00
if message != "" {
s = "# " + strings . ReplaceAll ( message , "{{INPUT_FILENAME}}" , filepath . Base ( inputFilename ) ) + "\n" + s
}
s = "# Code generated by KubeVela templates. DO NOT EDIT. Please edit the original cue file.\n" + s
2021-09-06 18:33:42 +08:00
if err := os . WriteFile ( outputFilename , [ ] byte ( s ) , 0600 ) ; err != nil {
2021-08-06 17:10:52 +08:00
return errors . Wrapf ( err , "failed to write YAML format definition to file %s" , outputFilename )
}
}
return nil
}
inputFilenames := [ ] string { args [ 0 ] }
outputFilenames := [ ] string { output }
fi , err := os . Stat ( args [ 0 ] )
if err != nil {
return errors . Wrapf ( err , "failed to get input %s" , args [ 0 ] )
}
if fi . IsDir ( ) {
inputFilenames = [ ] string { }
outputFilenames = [ ] string { }
2021-11-29 15:02:16 +08:00
err := filepath . Walk ( args [ 0 ] , func ( path string , info os . FileInfo , err error ) error {
filename := filepath . Base ( path )
fileSuffix := filepath . Ext ( path )
if fileSuffix != ".cue" {
return nil
2021-08-06 17:10:52 +08:00
}
2021-11-29 15:02:16 +08:00
inputFilenames = append ( inputFilenames , path )
2021-08-06 17:10:52 +08:00
if output != "" {
outputFilenames = append ( outputFilenames , filepath . Join ( output , strings . ReplaceAll ( filename , ".cue" , ".yaml" ) ) )
} else {
outputFilenames = append ( outputFilenames , "" )
}
2021-11-29 15:02:16 +08:00
return nil
} )
if err != nil {
return errors . Wrapf ( err , "failed to read directory %s" , args [ 0 ] )
2021-08-06 17:10:52 +08:00
}
}
for i , inputFilename := range inputFilenames {
if err = render ( inputFilename , outputFilenames [ i ] ) ; err != nil {
if _ , err = cmd . ErrOrStderr ( ) . Write ( [ ] byte ( fmt . Sprintf ( "failed to render %s, reason: %v" , inputFilename , err ) ) ) ; err != nil {
return errors . Wrapf ( err , "failed to write err" )
}
}
}
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagOutput , "o" , "" , "Specify the output path of the rendered definition YAML. If empty, the definition will be printed in the console. If input is a directory, the output path is expected to be a directory as well." )
2021-08-12 19:10:51 +08:00
cmd . Flags ( ) . StringP ( FlagMessage , "" , "" , "Specify the header message of the generated YAML file. For example, declaring author information." )
2021-08-06 17:10:52 +08:00
return cmd
}
2021-08-03 18:00:49 +08:00
// NewDefinitionApplyCommand create the `vela def apply` command to help user apply local definitions to k8s
func NewDefinitionApplyCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "apply DEFINITION.cue" ,
Short : "Apply definition" ,
Long : "Apply definition from local storage to kubernetes cluster. It will apply file to vela-system namespace by default." ,
Example : "# Command below will apply the local my-webservice.cue file to kubernetes vela-system namespace\n" +
"> vela def apply my-webservice.cue\n" +
"# Command below will apply the ./defs/my-trait.cue file to kubernetes default namespace\n" +
"> vela def apply ./defs/my-trait.cue --namespace default" +
"# Command below will convert the ./defs/my-trait.cue file to kubernetes CRD object and print it without applying it to kubernetes\n" +
"> vela def apply ./defs/my-trait.cue --dry-run" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
dryRun , err := cmd . Flags ( ) . GetBool ( FlagDryRun )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagDryRun )
}
namespace , err := cmd . Flags ( ) . GetString ( FlagNamespace )
if err != nil {
2021-11-19 18:00:03 +08:00
return errors . Wrapf ( err , "failed to get `%s`" , Namespace )
2021-08-03 18:00:49 +08:00
}
k8sClient , err := c . GetClient ( )
if err != nil {
return errors . Wrapf ( err , "failed to get k8s client" )
}
cueBytes , err := loadYAMLBytesFromFileOrHTTP ( args [ 0 ] )
if err != nil {
return errors . Wrapf ( err , "failed to get %s" , args [ 0 ] )
}
def := common2 . Definition { Unstructured : unstructured . Unstructured { } }
2021-08-13 18:44:01 +08:00
if err := def . FromCUEString ( string ( cueBytes ) , c . Config ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to parse CUE" )
}
def . SetNamespace ( namespace )
if dryRun {
2021-08-06 17:10:52 +08:00
s , err := prettyYAMLMarshal ( def . Object )
2021-08-03 18:00:49 +08:00
if err != nil {
return errors . Wrapf ( err , "failed to marshal CRD into YAML" )
}
2021-08-06 17:10:52 +08:00
cmd . Print ( s )
2021-08-03 18:00:49 +08:00
return nil
}
ctx := context . Background ( )
oldDef := common2 . Definition { Unstructured : unstructured . Unstructured { } }
oldDef . SetGroupVersionKind ( def . GroupVersionKind ( ) )
err = k8sClient . Get ( ctx , types2 . NamespacedName {
Namespace : def . GetNamespace ( ) ,
Name : def . GetName ( ) ,
} , & oldDef )
if err != nil {
if errors2 . IsNotFound ( err ) {
kind := def . GetKind ( )
if err = k8sClient . Create ( ctx , & def ) ; err != nil {
return errors . Wrapf ( err , "failed to create new definition in kubernetes" )
}
cmd . Printf ( "%s %s created in namespace %s.\n" , kind , def . GetName ( ) , def . GetNamespace ( ) )
return nil
}
return errors . Wrapf ( err , "failed to check existence of target definition in kubernetes" )
}
2021-08-13 18:44:01 +08:00
if err := oldDef . FromCUEString ( string ( cueBytes ) , c . Config ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to merge with existing definition" )
}
if err = k8sClient . Update ( ctx , & oldDef ) ; err != nil {
return errors . Wrapf ( err , "failed to update existing definition in kubernetes" )
}
cmd . Printf ( "%s %s in namespace %s updated.\n" , oldDef . GetKind ( ) , oldDef . GetName ( ) , oldDef . GetNamespace ( ) )
return nil
} ,
}
cmd . Flags ( ) . BoolP ( FlagDryRun , "" , false , "only build definition from CUE into CRB object without applying it to kubernetes clusters" )
2021-11-19 18:00:03 +08:00
cmd . Flags ( ) . StringP ( Namespace , "n" , "vela-system" , "Specify which namespace to apply." )
2021-08-03 18:00:49 +08:00
return cmd
}
// NewDefinitionDelCommand create the `vela def del` command to help user delete existing definitions conveniently
func NewDefinitionDelCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "del DEFINITION_NAME" ,
Short : "Delete definition" ,
Long : "Delete definition in kubernetes cluster." ,
Example : "# Command below will delete TraitDefinition of annotations in default namespace\n" +
"> vela def del annotations -t trait -n default" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
definitionType , err := cmd . Flags ( ) . GetString ( FlagType )
if err != nil {
return errors . Wrapf ( err , "failed to get `%s`" , FlagType )
}
namespace , err := cmd . Flags ( ) . GetString ( FlagNamespace )
if err != nil {
2021-11-19 18:00:03 +08:00
return errors . Wrapf ( err , "failed to get `%s`" , Namespace )
2021-08-03 18:00:49 +08:00
}
k8sClient , err := c . GetClient ( )
if err != nil {
return errors . Wrapf ( err , "failed to get k8s client" )
}
def , err := getSingleDefinition ( cmd , args [ 0 ] , k8sClient , definitionType , namespace )
if err != nil {
return err
}
desc := def . GetAnnotations ( ) [ common2 . DefinitionDescriptionKey ]
toDelete := false
_ , err = getPrompt ( cmd , bufio . NewReader ( cmd . InOrStdin ( ) ) ,
fmt . Sprintf ( "Are you sure to delete the following definition in namespace %s?\n" , def . GetNamespace ( ) ) +
fmt . Sprintf ( "%s %s: %s\n" , def . GetKind ( ) , def . GetName ( ) , desc ) ,
"[yes|no] > " ,
func ( resp string ) error {
switch strings . ToLower ( resp ) {
case "yes" :
toDelete = true
case "y" :
toDelete = true
case "no" :
toDelete = false
case "n" :
toDelete = false
default :
return errors . New ( "invalid input" )
}
return nil
} )
if err != nil {
return err
}
if ! toDelete {
return nil
}
if err := k8sClient . Delete ( context . Background ( ) , def ) ; err != nil {
return errors . Wrapf ( err , "failed to delete %s %s in namespace %s" , def . GetKind ( ) , def . GetName ( ) , def . GetNamespace ( ) )
}
cmd . Printf ( "%s %s in namespace %s deleted.\n" , def . GetKind ( ) , def . GetName ( ) , def . GetNamespace ( ) )
return nil
} ,
}
cmd . Flags ( ) . StringP ( FlagType , "t" , "" , "Specify the definition type of target. Valid types: " + strings . Join ( common2 . ValidDefinitionTypes ( ) , ", " ) )
2021-11-19 18:00:03 +08:00
cmd . Flags ( ) . StringP ( Namespace , "n" , "" , "Specify which namespace the definition locates." )
2021-08-03 18:00:49 +08:00
return cmd
}
// NewDefinitionValidateCommand create the `vela def vet` command to help user validate the definition
func NewDefinitionValidateCommand ( c common . Args ) * cobra . Command {
cmd := & cobra . Command {
Use : "vet DEFINITION.cue" ,
Short : "Validate definition" ,
Long : "Validate definition file by checking whether it has the valid cue format with fields set correctly\n" +
"* Currently, this command only checks the cue format. This function is still working in progress and we will support more functional validation mechanism in the future." ,
Example : "# Command below will validate the my-def.cue file.\n" +
"> vela def vet my-def.cue" ,
Args : cobra . ExactValidArgs ( 1 ) ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
2021-09-06 18:33:42 +08:00
cueBytes , err := os . ReadFile ( args [ 0 ] )
2021-08-03 18:00:49 +08:00
if err != nil {
return errors . Wrapf ( err , "failed to read %s" , args [ 0 ] )
}
def := common2 . Definition { Unstructured : unstructured . Unstructured { } }
2021-08-13 18:44:01 +08:00
if err := def . FromCUEString ( string ( cueBytes ) , c . Config ) ; err != nil {
2021-08-03 18:00:49 +08:00
return errors . Wrapf ( err , "failed to parse CUE" )
}
cmd . Println ( "Validation succeed." )
return nil
} ,
}
return cmd
}