// Copyright © 2015 Steve Francia . // // 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 cmd import ( "fmt" "os" "path/filepath" "unicode" "github.com/spf13/cobra" "github.com/spf13/viper" ) func init() { addCmd.Flags().StringVarP(&parentName, "parent", "p", "RootCmd", "name of parent command for this command") RootCmd.AddCommand(addCmd) } var parentName string var addCmd = &cobra.Command{ Use: "add [command name]", Aliases: []string{"command"}, Short: "Add a command to a Cobra Application", Long: `Add (cobra add) will create a new command, with a license and the appropriate structure for a Cobra-based CLI application, and register it to its parent (default RootCmd). If you want your command to be public, pass in the command name with an initial uppercase letter. Example: cobra add server -> resulting in a new cmd/server.go`, Run: func(cmd *cobra.Command, args []string) { if len(args) < 1 { er("add needs a name for the command") } wd, err := os.Getwd() if err != nil { er(err) } project := NewProjectFromPath(wd) cmdName := validateCmdName(args[0]) createCmdFile(project, cmdName) }, } // validateCmdName returns source without any dashes and underscore. // If there will be dash or underscore, next letter will be uppered. // It supports only ASCII (1-byte character) strings. func validateCmdName(source string) string { i := 0 l := len(source) // The output is initialized on demand, then first dash or underscore // occurs. var output string for i < l { if source[i] == '-' || source[i] == '_' { if output == "" { output = source[:i] } // If next character is dash or underscore, // just skip the current character. if source[i+1] == '-' || source[i+1] == '_' { i++ continue } // If the current character is dash or underscore, // upper next letter and add to output. output += string(unicode.ToUpper(rune(source[i+1]))) // We know, what source[i] is dash or underscore and source[i+1] is // uppered character, so make i = i+2. i += 2 continue } // If the current character isn't dash or underscore, // just add it. if output != "" { output += string(source[i]) } i++ } if output == "" { return source // source is initially valid name. } return output } func createCmdFile(project *Project, cmdName string) { template := `{{comment .copyright}} {{comment .license}} package cmd import ( "fmt" "github.com/spf13/cobra" ) // {{.cmdName}}Cmd represents the {{.cmdName}} command var {{.cmdName}}Cmd = &cobra.Command{ Use: "{{.cmdName}}", Short: "A brief description of your command", Long: ` + "`" + `A longer description that spans multiple lines and likely contains examples and usage of using your command. For example: Cobra is a CLI library for Go that empowers applications. This application is a tool to generate the needed files to quickly create a Cobra application.` + "`" + `, Run: func(cmd *cobra.Command, args []string) { fmt.Println("{{.cmdName}} called") }, } func init() { {{.parentName}}.AddCommand({{.cmdName}}Cmd) // Here you will define your flags and configuration settings. // Cobra supports Persistent Flags which will work for this command // and all subcommands, e.g.: // {{.cmdName}}Cmd.PersistentFlags().String("foo", "", "A help for foo") // Cobra supports local flags which will only run when this command // is called directly, e.g.: // {{.cmdName}}Cmd.Flags().BoolP("toggle", "t", false, "Help message for toggle") } ` data := make(map[string]interface{}) data["copyright"] = copyrightLine() data["license"] = project.License().Header data["viper"] = viper.GetBool("useViper") data["parentName"] = parentName data["cmdName"] = cmdName filePath := filepath.Join(project.CmdPath(), cmdName+".go") cmdScript, err := executeTemplate(template, data) if err != nil { er(err) } err = writeStringToFile(filePath, cmdScript) if err != nil { er(err) } fmt.Println(cmdName, "created at", filePath) }