Files
steampipe/pkg/cmdconfig/builder.go

164 lines
5.2 KiB
Go

package cmdconfig
import (
"fmt"
"github.com/spf13/cobra"
"github.com/spf13/pflag"
"github.com/spf13/viper"
pconstants "github.com/turbot/pipe-fittings/v2/constants"
"github.com/turbot/pipe-fittings/v2/utils"
"github.com/turbot/steampipe/v2/pkg/constants"
)
type CmdBuilder struct {
cmd *cobra.Command
bindings map[string]*pflag.Flag
}
// OnCmd starts a config builder wrapping over the provided *cobra.Command
func OnCmd(cmd *cobra.Command) *CmdBuilder {
cfg := new(CmdBuilder)
cfg.cmd = cmd
cfg.bindings = map[string]*pflag.Flag{}
// we will wrap over these two function - need references to call them
originalPreRun := cfg.cmd.PreRun
cfg.cmd.PreRun = func(cmd *cobra.Command, args []string) {
utils.LogTime(fmt.Sprintf("cmd.%s.PreRun start", cmd.CommandPath()))
defer utils.LogTime(fmt.Sprintf("cmd.%s.PreRun end", cmd.CommandPath()))
// bind flags
for flagName, flag := range cfg.bindings {
if flag == nil {
// we can panic here since this is bootstrap code and not execution path specific
panic(fmt.Sprintf("flag for %s cannot be nil", flagName))
}
//nolint:golint,errcheck // nil check above
viper.GetViper().BindPFlag(flagName, flag)
}
// now that we have done all the flag bindings, run the global pre run
// this will load up and populate the global config, init the logger and
// also run the daily task runner
preRunHook(cmd, args)
// run the original PreRun
if originalPreRun != nil {
originalPreRun(cmd, args)
}
}
originalPostRun := cfg.cmd.PostRun
cfg.cmd.PostRun = func(cmd *cobra.Command, args []string) {
utils.LogTime(fmt.Sprintf("cmd.%s.PostRun start", cmd.CommandPath()))
defer utils.LogTime(fmt.Sprintf("cmd.%s.PostRun end", cmd.CommandPath()))
// run the original PostRun
if originalPostRun != nil {
originalPostRun(cmd, args)
}
// run the post run
postRunHook(cmd, args)
}
// wrap over the original Run function
originalRun := cfg.cmd.Run
cfg.cmd.Run = func(cmd *cobra.Command, args []string) {
utils.LogTime(fmt.Sprintf("cmd.%s.Run start", cmd.CommandPath()))
defer utils.LogTime(fmt.Sprintf("cmd.%s.Run end", cmd.CommandPath()))
// run the original Run
if originalRun != nil {
originalRun(cmd, args)
}
}
return cfg
}
// AddStringFlag is a helper function to add a string flag to a command
func (c *CmdBuilder) AddStringFlag(name string, defaultValue string, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().String(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
// AddIntFlag is a helper function to add an integer flag to a command
func (c *CmdBuilder) AddIntFlag(name string, defaultValue int, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().Int(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
// AddBoolFlag ia s helper function to add a boolean flag to a command
func (c *CmdBuilder) AddBoolFlag(name string, defaultValue bool, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().Bool(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
// AddCloudFlags is helper function to add the cloud flags to a command
func (c *CmdBuilder) AddCloudFlags() *CmdBuilder {
return c.
AddStringFlag(pconstants.ArgPipesHost, constants.DefaultPipesHost, "Turbot Pipes host").
AddStringFlag(pconstants.ArgPipesToken, "", "Turbot Pipes authentication token")
}
// AddWorkspaceDatabaseFlag is helper function to add the workspace-databse flag to a command
func (c *CmdBuilder) AddWorkspaceDatabaseFlag() *CmdBuilder {
return c.
AddStringFlag(pconstants.ArgWorkspaceDatabase, constants.DefaultWorkspaceDatabase, "Turbot Pipes workspace database")
}
// AddStringSliceFlag is a helper function to add a flag that accepts an array of strings
func (c *CmdBuilder) AddStringSliceFlag(name string, defaultValue []string, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().StringSlice(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
// AddStringArrayFlag is a helper function to add a flag that accepts an array of strings
func (c *CmdBuilder) AddStringArrayFlag(name string, defaultValue []string, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().StringArray(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
// AddStringMapStringFlag is a helper function to add a flag that accepts a map of strings
func (c *CmdBuilder) AddStringMapStringFlag(name string, defaultValue map[string]string, desc string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().StringToString(name, defaultValue, desc)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
return c
}
func (c *CmdBuilder) AddVarFlag(value pflag.Value, name string, usage string, opts ...FlagOption) *CmdBuilder {
c.cmd.Flags().Var(value, name, usage)
c.bindings[name] = c.cmd.Flags().Lookup(name)
for _, o := range opts {
o(c.cmd, name, name)
}
//
return c
}