package helper import ( "bytes" "fmt" "io" "io/ioutil" "os" "path/filepath" "strings" "text/template" "github.com/SAP/jenkins-library/pkg/config" "github.com/SAP/jenkins-library/pkg/piperutils" ) type stepInfo struct { CobraCmdFuncName string CreateCmdVar string ExportPrefix string FlagsFunc string Long string StepParameters []config.StepParameters StepAliases []config.Alias OSImport bool OutputResources []map[string]string Short string StepFunc string StepName string StepSecrets []string } //StepGoTemplate ... const stepGoTemplate = `// Code generated by piper's step-generator. DO NOT EDIT. package cmd import ( "fmt" "os" {{ if .OutputResources -}} "path/filepath" {{ end -}} "time" {{ if .ExportPrefix -}} {{ .ExportPrefix }} "github.com/SAP/jenkins-library/cmd" {{ end -}} "github.com/SAP/jenkins-library/pkg/config" "github.com/SAP/jenkins-library/pkg/log" {{ if .OutputResources -}} "github.com/SAP/jenkins-library/pkg/piperenv" {{ end -}} "github.com/SAP/jenkins-library/pkg/telemetry" "github.com/spf13/cobra" ) type {{ .StepName }}Options struct { {{- range $key, $value := .StepParameters }} {{ $value.Name | golangName }} {{ $value.Type }} ` + "`json:\"{{$value.Name}},omitempty\"`" + `{{end}} } {{ range $notused, $oRes := .OutputResources }} {{ index $oRes "def"}} {{ end }} // {{.CobraCmdFuncName}} {{.Short}} func {{.CobraCmdFuncName}}() *cobra.Command { const STEP_NAME = "{{ .StepName }}" metadata := {{ .StepName }}Metadata() var stepConfig {{.StepName}}Options var startTime time.Time {{- range $notused, $oRes := .OutputResources }} var {{ index $oRes "name" }} {{ index $oRes "objectname" }}{{ end }} var {{.CreateCmdVar}} = &cobra.Command{ Use: STEP_NAME, Short: "{{.Short}}", Long: {{ $tick := "` + "`" + `" }}{{ $tick }}{{.Long | longName }}{{ $tick }}, PreRunE: func(cmd *cobra.Command, args []string) error { startTime = time.Now() log.SetStepName(STEP_NAME) log.SetVerbose({{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.Verbose) path, _ := os.Getwd() fatalHook := &log.FatalHook{CorrelationID: {{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.CorrelationID, Path: path} log.RegisterHook(fatalHook) err := {{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}PrepareConfig(cmd, &metadata, STEP_NAME, &stepConfig, config.OpenPiperFile) if err != nil { return err } {{- range $key, $value := .StepSecrets }} log.RegisterSecret(stepConfig.{{ $value | golangName }}){{end}} if len({{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.HookConfig.SentryConfig.Dsn) > 0 { sentryHook := log.NewSentryHook({{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.HookConfig.SentryConfig.Dsn, {{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.CorrelationID) log.RegisterHook(&sentryHook) } return nil }, Run: func(cmd *cobra.Command, args []string) { telemetryData := telemetry.CustomData{} telemetryData.ErrorCode = "1" handler := func() { {{- range $notused, $oRes := .OutputResources }} {{ index $oRes "name" }}.persist({{if $.ExportPrefix}}{{ $.ExportPrefix }}.{{end}}GeneralConfig.EnvRootPath, "{{ index $oRes "name" }}"){{ end }} telemetryData.Duration = fmt.Sprintf("%v", time.Since(startTime).Milliseconds()) telemetry.Send(&telemetryData) } log.DeferExitHandler(handler) defer handler() telemetry.Initialize({{if .ExportPrefix}}{{ .ExportPrefix }}.{{end}}GeneralConfig.NoTelemetry, STEP_NAME) {{.StepName}}(stepConfig, &telemetryData{{ range $notused, $oRes := .OutputResources}}, &{{ index $oRes "name" }}{{ end }}) telemetryData.ErrorCode = "0" log.Entry().Info("SUCCESS") }, } {{.FlagsFunc}}({{.CreateCmdVar}}, &stepConfig) return {{.CreateCmdVar}} } func {{.FlagsFunc}}(cmd *cobra.Command, stepConfig *{{.StepName}}Options) { {{- range $key, $value := .StepParameters }} cmd.Flags().{{ $value.Type | flagType }}(&stepConfig.{{ $value.Name | golangName }}, "{{ $value.Name }}", {{ $value.Default }}, "{{ $value.Description }}"){{ end }} {{- printf "\n" }} {{- range $key, $value := .StepParameters }}{{ if $value.Mandatory }} cmd.MarkFlagRequired("{{ $value.Name }}"){{ end }}{{ end }} } // retrieve step metadata func {{ .StepName }}Metadata() config.StepData { var theMetaData = config.StepData{ Metadata: config.StepMetadata{ Name: "{{ .StepName }}", Aliases: []config.Alias{{ "{" }}{{ range $notused, $alias := .StepAliases }}{{ "{" }}Name: "{{ $alias.Name }}", Deprecated: {{ $alias.Deprecated }}{{ "}" }},{{ end }}{{ "}" }}, }, Spec: config.StepSpec{ Inputs: config.StepInputs{ Parameters: []config.StepParameters{ {{- range $key, $value := .StepParameters }} { Name: "{{ $value.Name }}", ResourceRef: []config.ResourceReference{{ "{" }}{{ range $notused, $ref := $value.ResourceRef }}{{ "{" }}Name: "{{ $ref.Name }}", Param: "{{ $ref.Param }}"{{ "}" }},{{ end }}{{ "}" }}, Scope: []string{{ "{" }}{{ range $notused, $scope := $value.Scope }}"{{ $scope }}",{{ end }}{{ "}" }}, Type: "{{ $value.Type }}", Mandatory: {{ $value.Mandatory }}, Aliases: []config.Alias{{ "{" }}{{ range $notused, $alias := $value.Aliases }}{{ "{" }}Name: "{{ $alias.Name }}"{{ "}" }},{{ end }}{{ "}" }}, },{{ end }} }, }, }, } return theMetaData } ` //StepTestGoTemplate ... const stepTestGoTemplate = `package cmd import ( "testing" "github.com/stretchr/testify/assert" ) func Test{{.CobraCmdFuncName}}(t *testing.T) { testCmd := {{.CobraCmdFuncName}}() // only high level testing performed - details are tested in step generation procudure assert.Equal(t, "{{ .StepName }}", testCmd.Use, "command name incorrect") } ` const stepGoImplementationTemplate = `package cmd import ( "github.com/SAP/jenkins-library/pkg/command" "github.com/SAP/jenkins-library/pkg/log" "github.com/SAP/jenkins-library/pkg/telemetry" ) func {{.StepName}}(config {{ .StepName }}Options, telemetryData *telemetry.CustomData{{ range $notused, $oRes := .OutputResources}}, {{ index $oRes "name" }} *{{ index $oRes "objectname" }}{{ end }}) { // for command execution use Command c := command.Command{} // reroute command output to logging framework c.Stdout(log.Writer()) c.Stderr(log.Writer()) // for http calls import piperhttp "github.com/SAP/jenkins-library/pkg/http" // and use a &piperhttp.Client{} in a custom system // Example: step checkmarxExecuteScan.go // error situations should stop execution through log.Entry().Fatal() call which leads to an os.Exit(1) in the end err := run{{.StepName | title}}(&config, telemetryData, &c{{ range $notused, $oRes := .OutputResources}}, &{{ index $oRes "name" }}{{ end }}) if err != nil { log.Entry().WithError(err).Fatal("step execution failed") } } func run{{.StepName | title}}(config *{{ .StepName }}Options, telemetryData *telemetry.CustomData, command execRunner{{ range $notused, $oRes := .OutputResources}}, {{ index $oRes "name" }} *{{ index $oRes "objectname" }} {{ end }}) error { log.Entry().WithField("LogField", "Log field content").Info("This is just a demo for a simple step.") return nil } ` // ProcessMetaFiles generates step coding based on step configuration provided in yaml files func ProcessMetaFiles(metadataFiles []string, stepHelperData StepHelperData, docuHelperData DocuHelperData) error { for key := range metadataFiles { var stepData config.StepData configFilePath := metadataFiles[key] metadataFile, err := stepHelperData.OpenFile(configFilePath) checkError(err) defer metadataFile.Close() fmt.Printf("Reading file %v\n", configFilePath) err = stepData.ReadPipelineStepData(metadataFile) checkError(err) fmt.Printf("Step name: %v\n", stepData.Metadata.Name) //Switch Docu or Step Files if !docuHelperData.IsGenerateDocu { osImport := false osImport, err = setDefaultParameters(&stepData) checkError(err) myStepInfo, err := getStepInfo(&stepData, osImport, stepHelperData.ExportPrefix) checkError(err) step := stepTemplate(myStepInfo) err = stepHelperData.WriteFile(fmt.Sprintf("cmd/%v_generated.go", stepData.Metadata.Name), step, 0644) checkError(err) test := stepTestTemplate(myStepInfo) err = stepHelperData.WriteFile(fmt.Sprintf("cmd/%v_generated_test.go", stepData.Metadata.Name), test, 0644) checkError(err) exists, _ := piperutils.FileExists(fmt.Sprintf("cmd/%v.go", stepData.Metadata.Name)) if !exists { impl := stepImplementation(myStepInfo) err = stepHelperData.WriteFile(fmt.Sprintf("cmd/%v.go", stepData.Metadata.Name), impl, 0644) checkError(err) } } else { err = generateStepDocumentation(stepData, docuHelperData) if err != nil { fmt.Printf("%v\n", err) } } } return nil } func openMetaFile(name string) (io.ReadCloser, error) { return os.Open(name) } func fileWriter(filename string, data []byte, perm os.FileMode) error { return ioutil.WriteFile(filename, data, perm) } func setDefaultParameters(stepData *config.StepData) (bool, error) { //ToDo: custom function for default handling, support all relevant parameter types osImportRequired := false for k, param := range stepData.Spec.Inputs.Parameters { if param.Default == nil { switch param.Type { case "bool": // ToDo: Check if default should be read from env param.Default = "false" case "int": param.Default = "0" case "string": param.Default = fmt.Sprintf("os.Getenv(\"PIPER_%v\")", param.Name) osImportRequired = true case "[]string": // ToDo: Check if default should be read from env param.Default = "[]string{}" default: return false, fmt.Errorf("Meta data type not set or not known: '%v'", param.Type) } } else { switch param.Type { case "bool": boolVal := "false" if param.Default.(bool) == true { boolVal = "true" } param.Default = boolVal case "int": param.Default = fmt.Sprintf("%v", param.Default) case "string": param.Default = fmt.Sprintf("\"%v\"", param.Default) case "[]string": param.Default = fmt.Sprintf("[]string{\"%v\"}", strings.Join(getStringSliceFromInterface(param.Default), "\", \"")) default: return false, fmt.Errorf("Meta data type not set or not known: '%v'", param.Type) } } stepData.Spec.Inputs.Parameters[k] = param } return osImportRequired, nil } func getStepInfo(stepData *config.StepData, osImport bool, exportPrefix string) (stepInfo, error) { oRes, err := getOutputResourceDetails(stepData) return stepInfo{ StepName: stepData.Metadata.Name, CobraCmdFuncName: fmt.Sprintf("%vCommand", strings.Title(stepData.Metadata.Name)), CreateCmdVar: fmt.Sprintf("create%vCmd", strings.Title(stepData.Metadata.Name)), Short: stepData.Metadata.Description, Long: stepData.Metadata.LongDescription, StepParameters: stepData.Spec.Inputs.Parameters, StepAliases: stepData.Metadata.Aliases, FlagsFunc: fmt.Sprintf("add%vFlags", strings.Title(stepData.Metadata.Name)), OSImport: osImport, OutputResources: oRes, ExportPrefix: exportPrefix, StepSecrets: getSecretFields(stepData), }, err } func getSecretFields(stepData *config.StepData) []string { var secretFields []string for _, parameter := range stepData.Spec.Inputs.Parameters { if parameter.Secret { secretFields = append(secretFields, parameter.Name) } } return secretFields } func getOutputResourceDetails(stepData *config.StepData) ([]map[string]string, error) { outputResources := []map[string]string{} for _, res := range stepData.Spec.Outputs.Resources { currentResource := map[string]string{} currentResource["name"] = res.Name switch res.Type { case "piperEnvironment": var envResource PiperEnvironmentResource envResource.Name = res.Name envResource.StepName = stepData.Metadata.Name for _, param := range res.Parameters { paramSections := strings.Split(fmt.Sprintf("%v", param["name"]), "/") category := "" name := paramSections[0] if len(paramSections) > 1 { name = strings.Join(paramSections[1:], "_") category = paramSections[0] if !contains(envResource.Categories, category) { envResource.Categories = append(envResource.Categories, category) } } envParam := PiperEnvironmentParameter{Category: category, Name: name} envResource.Parameters = append(envResource.Parameters, envParam) } def, err := envResource.StructString() if err != nil { return outputResources, err } currentResource["def"] = def currentResource["objectname"] = envResource.StructName() outputResources = append(outputResources, currentResource) case "influx": var influxResource InfluxResource influxResource.Name = res.Name influxResource.StepName = stepData.Metadata.Name for _, measurement := range res.Parameters { influxMeasurement := InfluxMeasurement{Name: fmt.Sprintf("%v", measurement["name"])} if fields, ok := measurement["fields"].([]interface{}); ok { for _, field := range fields { if fieldParams, ok := field.(map[string]interface{}); ok { influxMeasurement.Fields = append(influxMeasurement.Fields, InfluxMetric{Name: fmt.Sprintf("%v", fieldParams["name"])}) } } } if tags, ok := measurement["tags"].([]interface{}); ok { for _, tag := range tags { if tagParams, ok := tag.(map[string]interface{}); ok { influxMeasurement.Tags = append(influxMeasurement.Tags, InfluxMetric{Name: fmt.Sprintf("%v", tagParams["name"])}) } } } influxResource.Measurements = append(influxResource.Measurements, influxMeasurement) } def, err := influxResource.StructString() if err != nil { return outputResources, err } currentResource["def"] = def currentResource["objectname"] = influxResource.StructName() outputResources = append(outputResources, currentResource) } } return outputResources, nil } // MetadataFiles provides a list of all step metadata files func MetadataFiles(sourceDirectory string) ([]string, error) { var metadataFiles []string err := filepath.Walk(sourceDirectory, func(path string, info os.FileInfo, err error) error { if filepath.Ext(path) == ".yaml" { metadataFiles = append(metadataFiles, path) } return nil }) if err != nil { return metadataFiles, nil } return metadataFiles, nil } func stepTemplate(myStepInfo stepInfo) []byte { funcMap := template.FuncMap{ "flagType": flagType, "golangName": golangNameTitle, "title": strings.Title, "longName": longName, } tmpl, err := template.New("step").Funcs(funcMap).Parse(stepGoTemplate) checkError(err) var generatedCode bytes.Buffer err = tmpl.Execute(&generatedCode, myStepInfo) checkError(err) return generatedCode.Bytes() } func stepTestTemplate(myStepInfo stepInfo) []byte { funcMap := template.FuncMap{ "flagType": flagType, "golangName": golangNameTitle, "title": strings.Title, } tmpl, err := template.New("stepTest").Funcs(funcMap).Parse(stepTestGoTemplate) checkError(err) var generatedCode bytes.Buffer err = tmpl.Execute(&generatedCode, myStepInfo) checkError(err) return generatedCode.Bytes() } func stepImplementation(myStepInfo stepInfo) []byte { funcMap := template.FuncMap{ "title": strings.Title, } tmpl, err := template.New("impl").Funcs(funcMap).Parse(stepGoImplementationTemplate) checkError(err) var generatedCode bytes.Buffer err = tmpl.Execute(&generatedCode, myStepInfo) checkError(err) return generatedCode.Bytes() } func longName(long string) string { l := strings.ReplaceAll(long, "`", "` + \"`\" + `") l = strings.TrimSpace(l) return l } func golangName(name string) string { properName := strings.Replace(name, "Api", "API", -1) properName = strings.Replace(properName, "api", "API", -1) properName = strings.Replace(properName, "Url", "URL", -1) properName = strings.Replace(properName, "Id", "ID", -1) properName = strings.Replace(properName, "Json", "JSON", -1) properName = strings.Replace(properName, "json", "JSON", -1) properName = strings.Replace(properName, "Tls", "TLS", -1) return properName } func golangNameTitle(name string) string { return strings.Title(golangName(name)) } func flagType(paramType string) string { var theFlagType string switch paramType { case "bool": theFlagType = "BoolVar" case "int": theFlagType = "IntVar" case "string": theFlagType = "StringVar" case "[]string": theFlagType = "StringSliceVar" default: fmt.Printf("Meta data type not set or not known: '%v'\n", paramType) os.Exit(1) } return theFlagType } func getStringSliceFromInterface(iSlice interface{}) []string { s := []string{} t, ok := iSlice.([]interface{}) if ok { for _, v := range t { s = append(s, fmt.Sprintf("%v", v)) } } else { s = append(s, fmt.Sprintf("%v", iSlice)) } return s }