package cmd import ( "fmt" "path/filepath" "github.com/ory/viper" "github.com/spf13/cobra" bosonFunc "github.com/boson-project/func" "github.com/boson-project/func/prompt" "github.com/boson-project/func/utils" ) func init() { root.AddCommand(createCmd) createCmd.Flags().BoolP("confirm", "c", false, "Prompt to confirm all configuration options (Env: $FUNC_CONFIRM)") createCmd.Flags().StringP("runtime", "l", bosonFunc.DefaultRuntime, "Function runtime language/framework. Available runtimes: "+utils.RuntimeList()+" (Env: $FUNC_RUNTIME)") createCmd.Flags().StringP("templates", "", filepath.Join(configPath(), "templates"), "Path to additional templates (Env: $FUNC_TEMPLATES)") createCmd.Flags().StringP("template", "t", bosonFunc.DefaultTemplate, "Function template. For eample 'http' or 'events' (Env: $FUNC_TEMPLATE)") if err := createCmd.RegisterFlagCompletionFunc("runtime", CompleteRuntimeList); err != nil { fmt.Println("internal: error while calling RegisterFlagCompletionFunc: ", err) } } var createCmd = &cobra.Command{ Use: "create [PATH]", Short: "Create a function project", Long: `Create a function project Creates a new function project in PATH, or in the current directory if no PATH is given. The name of the project is determined by the directory name the project is created in. `, Example: ` # Create a Node.js function project in the current directory, choosing the # directory name as the project's name. kn func create # Create a Quarkus function project in the directory "sample-service". # The directory will be created in the local directory if non-existent and # the project is called "sample-service" kn func create --runtime quarkus myfunc # Create a function project that uses a CloudEvent based function signature kn func create -t events myfunc `, SuggestFor: []string{"inti", "new"}, PreRunE: bindEnv("runtime", "templates", "template", "confirm"), RunE: runCreate, // TODO: autocomplate or interactive prompt for runtime and template. } func runCreate(cmd *cobra.Command, args []string) error { config := newCreateConfig(args) if err := utils.ValidateFunctionName(config.Name); err != nil { return err } config = config.Prompt() function := bosonFunc.Function{ Name: config.Name, Root: config.Path, Runtime: config.Runtime, Template: config.Template, } client := bosonFunc.New( bosonFunc.WithTemplates(config.Templates), bosonFunc.WithVerbose(config.Verbose)) return client.Create(function) } type createConfig struct { // Name of the Function. Name string // Absolute path to Function on disk. Path string // Runtime language/framework. Runtime string // Templates is an optional path that, if it exists, will be used as a source // for additional templates not included in the binary. If not provided // explicitly as a flag (--templates) or env (FUNC_TEMPLATES), the default // location is $XDG_CONFIG_HOME/templates ($HOME/.config/func/templates) Templates string // Template is the code written into the new Function project, including // an implementation adhering to one of the supported function signatures. // May also include additional configuration settings or examples. // For example, embedded are 'http' for a Function whose funciton signature // is invoked via straight HTTP requests, or 'events' for a Function which // will be invoked with CloudEvents. These embedded templates contain a // minimum implementation of the signature itself and example tests. Template string // Verbose output Verbose bool // Confirm: confirm values arrived upon from environment plus flags plus defaults, // with interactive prompting (only applicable when attached to a TTY). Confirm bool } // newCreateConfig returns a config populated from the current execution context // (args, flags and environment variables) func newCreateConfig(args []string) createConfig { var path string if len(args) > 0 { path = args[0] // If explicitly provided, use. } derivedName, derivedPath := deriveNameAndAbsolutePathFromPath(path) return createConfig{ Name: derivedName, Path: derivedPath, Runtime: viper.GetString("runtime"), Templates: viper.GetString("templates"), Template: viper.GetString("template"), Confirm: viper.GetBool("confirm"), Verbose: viper.GetBool("verbose"), } } // Prompt the user with value of config members, allowing for interaractive changes. // Skipped if not in an interactive terminal (non-TTY), or if --confirm false (agree to // all prompts) was set (default). func (c createConfig) Prompt() createConfig { if !interactiveTerminal() || !c.Confirm { // Just print the basics if not confirming fmt.Printf("Project path: %v\n", c.Path) fmt.Printf("Function name: %v\n", c.Name) fmt.Printf("Runtime: %v\n", c.Runtime) fmt.Printf("Template: %v\n", c.Template) return c } var derivedName, derivedPath string for { derivedName, derivedPath = deriveNameAndAbsolutePathFromPath(prompt.ForString("Project path", c.Path, prompt.WithRequired(true))) err := utils.ValidateFunctionName(derivedName) if err == nil { break } fmt.Println("Error:", err) } return createConfig{ Name: derivedName, Path: derivedPath, Runtime: prompt.ForString("Runtime", c.Runtime), Template: prompt.ForString("Template", c.Template), } }