2013-07-19 15:34:01 +00:00
|
|
|
package cli
|
|
|
|
|
|
|
|
import (
|
2013-09-04 19:58:31 +00:00
|
|
|
"fmt"
|
2014-06-12 07:39:13 +00:00
|
|
|
"io"
|
2013-09-14 22:29:57 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2016-04-03 09:50:08 +00:00
|
|
|
"path/filepath"
|
2015-08-21 08:58:14 +00:00
|
|
|
"sort"
|
2013-11-15 11:40:18 +00:00
|
|
|
"time"
|
2013-07-19 15:34:01 +00:00
|
|
|
)
|
|
|
|
|
2016-04-28 14:15:04 +00:00
|
|
|
var (
|
2016-05-22 21:42:23 +00:00
|
|
|
changeLogURL = "https://github.com/urfave/cli/blob/master/CHANGELOG.md"
|
2016-05-01 13:09:54 +00:00
|
|
|
appActionDeprecationURL = fmt.Sprintf("%s#deprecated-cli-app-action-signature", changeLogURL)
|
|
|
|
runAndExitOnErrorDeprecationURL = fmt.Sprintf("%s#deprecated-cli-app-runandexitonerror", changeLogURL)
|
2016-04-28 15:03:10 +00:00
|
|
|
|
2016-04-29 07:01:57 +00:00
|
|
|
contactSysadmin = "This is an error in the application. Please contact the distributor of this application if this is not you."
|
|
|
|
|
2016-11-02 03:33:12 +00:00
|
|
|
errInvalidActionType = NewExitError("ERROR invalid Action type. "+
|
|
|
|
fmt.Sprintf("Must be `func(*Context`)` or `func(*Context) error). %s", contactSysadmin)+
|
2016-04-28 15:03:10 +00:00
|
|
|
fmt.Sprintf("See %s", appActionDeprecationURL), 2)
|
2016-04-28 14:15:04 +00:00
|
|
|
)
|
|
|
|
|
2016-02-09 16:36:13 +00:00
|
|
|
// App is the main structure of a cli application. It is recommended that
|
2015-06-11 10:24:06 +00:00
|
|
|
// an app be created with the cli.NewApp() function
|
2013-07-19 15:34:01 +00:00
|
|
|
type App struct {
|
2015-12-18 17:58:32 +00:00
|
|
|
// The name of the program. Defaults to path.Base(os.Args[0])
|
2013-07-19 15:34:01 +00:00
|
|
|
Name string
|
2015-08-13 05:14:26 +00:00
|
|
|
// Full name of command for help, defaults to Name
|
2015-08-13 04:43:14 +00:00
|
|
|
HelpName string
|
2013-07-19 15:34:01 +00:00
|
|
|
// Description of the program.
|
|
|
|
Usage string
|
2015-10-25 05:51:06 +00:00
|
|
|
// Text to override the USAGE section of help
|
|
|
|
UsageText string
|
2015-08-03 23:51:11 +00:00
|
|
|
// Description of the program argument format.
|
|
|
|
ArgsUsage string
|
2013-07-19 15:34:01 +00:00
|
|
|
// Version of the program
|
|
|
|
Version string
|
2016-10-12 15:06:41 +00:00
|
|
|
// Description of the program
|
|
|
|
Description string
|
2013-07-19 15:34:01 +00:00
|
|
|
// List of commands to execute
|
|
|
|
Commands []Command
|
2013-07-20 22:50:13 +00:00
|
|
|
// List of flags to parse
|
|
|
|
Flags []Flag
|
2014-04-12 13:32:53 +00:00
|
|
|
// Boolean to enable bash completion commands
|
|
|
|
EnableBashCompletion bool
|
2014-07-13 13:16:30 +00:00
|
|
|
// Boolean to hide built-in help command
|
|
|
|
HideHelp bool
|
2016-03-11 16:25:30 +00:00
|
|
|
// Boolean to hide built-in version flag and the VERSION section of help
|
2014-11-12 11:38:58 +00:00
|
|
|
HideVersion bool
|
2016-05-09 13:40:09 +00:00
|
|
|
// Populate on app startup, only gettable through method Categories()
|
2015-08-21 11:25:37 +00:00
|
|
|
categories CommandCategories
|
2019-01-27 16:41:54 +00:00
|
|
|
// Populate on app startup, only gettable through method Categories()
|
|
|
|
flagCategories FlagCategories
|
2014-04-12 13:32:53 +00:00
|
|
|
// An action to execute when the bash-completion flag is set
|
2016-04-25 22:29:05 +00:00
|
|
|
BashComplete BashCompleteFunc
|
2014-01-01 21:00:20 +00:00
|
|
|
// An action to execute before any subcommands are run, but after the context is ready
|
|
|
|
// If a non-nil error is returned, no subcommands are run
|
2016-04-25 22:29:05 +00:00
|
|
|
Before BeforeFunc
|
2014-11-18 22:44:21 +00:00
|
|
|
// An action to execute after any subcommands are run, but after the subcommand has finished
|
2015-02-20 21:21:27 +00:00
|
|
|
// It is run even if Action() panics
|
2016-04-25 22:29:05 +00:00
|
|
|
After AfterFunc
|
2016-08-21 20:51:55 +00:00
|
|
|
|
2013-07-19 15:34:01 +00:00
|
|
|
// The action to execute when no subcommands are specified
|
2016-08-21 20:51:55 +00:00
|
|
|
// Expects a `cli.ActionFunc` but will accept the *deprecated* signature of `func(*cli.Context) {}`
|
|
|
|
// *Note*: support for the deprecated `Action` signature will be removed in a future version
|
2016-04-27 15:34:01 +00:00
|
|
|
Action interface{}
|
|
|
|
|
2014-03-31 03:40:46 +00:00
|
|
|
// Execute this function if the proper command cannot be found
|
2016-04-25 22:29:05 +00:00
|
|
|
CommandNotFound CommandNotFoundFunc
|
|
|
|
// Execute this function if an usage error occurs
|
|
|
|
OnUsageError OnUsageErrorFunc
|
2013-11-15 11:40:18 +00:00
|
|
|
// Compilation date
|
|
|
|
Compiled time.Time
|
2015-02-20 23:44:00 +00:00
|
|
|
// List of all authors who contributed
|
2015-01-30 23:04:52 +00:00
|
|
|
Authors []Author
|
2015-06-09 22:35:50 +00:00
|
|
|
// Copyright of the binary if any
|
|
|
|
Copyright string
|
2015-02-20 23:44:00 +00:00
|
|
|
// Name of Author (Note: Use App.Authors, this is deprecated)
|
|
|
|
Author string
|
|
|
|
// Email of Author (Note: Use App.Authors, this is deprecated)
|
|
|
|
Email string
|
2014-12-02 04:50:04 +00:00
|
|
|
// Writer writer to write output to
|
|
|
|
Writer io.Writer
|
2016-05-06 16:14:26 +00:00
|
|
|
// ErrWriter writes error output
|
|
|
|
ErrWriter io.Writer
|
2017-04-25 16:33:54 +00:00
|
|
|
// Execute this function to handle ExitErrors. If not provided, HandleExitCoder is provided to
|
|
|
|
// function as a default, so this is optional.
|
2017-04-25 16:29:43 +00:00
|
|
|
ExitErrHandler ExitErrHandlerFunc
|
2016-03-02 02:45:13 +00:00
|
|
|
// Other custom info
|
2016-04-30 02:42:07 +00:00
|
|
|
Metadata map[string]interface{}
|
2016-11-25 09:07:42 +00:00
|
|
|
// Carries a function which returns app specific info.
|
|
|
|
ExtraInfo func() map[string]string
|
2016-11-25 08:16:48 +00:00
|
|
|
// CustomAppHelpTemplate the text template for app help topic.
|
|
|
|
// cli.go uses text/template to render templates. You can
|
|
|
|
// render custom help text by setting this variable.
|
|
|
|
CustomAppHelpTemplate string
|
2016-05-03 10:54:05 +00:00
|
|
|
|
|
|
|
didSetup bool
|
2013-11-15 11:40:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tries to find out when this binary was compiled.
|
|
|
|
// Returns the current time if it fails to find it.
|
|
|
|
func compileTime() time.Time {
|
|
|
|
info, err := os.Stat(os.Args[0])
|
|
|
|
if err != nil {
|
|
|
|
return time.Now()
|
|
|
|
}
|
|
|
|
return info.ModTime()
|
2013-07-19 15:34:01 +00:00
|
|
|
}
|
|
|
|
|
2016-05-09 14:12:59 +00:00
|
|
|
// NewApp creates a new cli Application with some reasonable defaults for Name,
|
|
|
|
// Usage, Version and Action.
|
2013-07-19 15:34:01 +00:00
|
|
|
func NewApp() *App {
|
|
|
|
return &App{
|
2016-04-03 09:50:08 +00:00
|
|
|
Name: filepath.Base(os.Args[0]),
|
|
|
|
HelpName: filepath.Base(os.Args[0]),
|
2014-04-12 13:32:53 +00:00
|
|
|
Usage: "A new cli application",
|
2015-10-25 05:51:06 +00:00
|
|
|
UsageText: "",
|
2014-04-12 13:32:53 +00:00
|
|
|
Version: "0.0.0",
|
|
|
|
BashComplete: DefaultAppComplete,
|
|
|
|
Action: helpCommand.Action,
|
|
|
|
Compiled: compileTime(),
|
2014-12-02 04:50:04 +00:00
|
|
|
Writer: os.Stdout,
|
2013-07-19 15:34:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-03 10:54:05 +00:00
|
|
|
// Setup runs initialization code to ensure all data structures are ready for
|
|
|
|
// `Run` or inspection prior to `Run`. It is internally called by `Run`, but
|
|
|
|
// will return early if setup has already happened.
|
|
|
|
func (a *App) Setup() {
|
|
|
|
if a.didSetup {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
a.didSetup = true
|
|
|
|
|
2015-03-10 14:59:59 +00:00
|
|
|
if a.Author != "" || a.Email != "" {
|
2015-03-10 15:12:44 +00:00
|
|
|
a.Authors = append(a.Authors, Author{Name: a.Author, Email: a.Email})
|
2015-02-20 23:44:00 +00:00
|
|
|
}
|
|
|
|
|
2015-08-13 04:43:14 +00:00
|
|
|
newCmds := []Command{}
|
|
|
|
for _, c := range a.Commands {
|
2015-08-13 04:58:25 +00:00
|
|
|
if c.HelpName == "" {
|
|
|
|
c.HelpName = fmt.Sprintf("%s %s", a.HelpName, c.Name)
|
|
|
|
}
|
2019-01-27 20:48:00 +00:00
|
|
|
|
|
|
|
fc := FlagCategories{}
|
|
|
|
for _, flag := range c.Flags {
|
|
|
|
fc = fc.AddFlag(flag.GetCategory(), flag)
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Sort(fc)
|
|
|
|
c.FlagCategories = fc
|
2015-08-13 04:43:14 +00:00
|
|
|
newCmds = append(newCmds, c)
|
|
|
|
}
|
|
|
|
a.Commands = newCmds
|
|
|
|
|
2014-07-13 13:16:30 +00:00
|
|
|
if a.Command(helpCommand.Name) == nil && !a.HideHelp {
|
2013-09-14 22:29:57 +00:00
|
|
|
a.Commands = append(a.Commands, helpCommand)
|
2014-12-02 04:20:21 +00:00
|
|
|
if (HelpFlag != BoolFlag{}) {
|
|
|
|
a.appendFlag(HelpFlag)
|
|
|
|
}
|
2013-09-14 22:29:57 +00:00
|
|
|
}
|
|
|
|
|
2014-11-12 11:38:58 +00:00
|
|
|
if !a.HideVersion {
|
|
|
|
a.appendFlag(VersionFlag)
|
|
|
|
}
|
2016-06-22 13:34:24 +00:00
|
|
|
|
|
|
|
a.categories = CommandCategories{}
|
|
|
|
for _, command := range a.Commands {
|
|
|
|
a.categories = a.categories.AddCommand(command.Category, command)
|
|
|
|
}
|
|
|
|
sort.Sort(a.categories)
|
2016-07-11 06:39:58 +00:00
|
|
|
|
2019-01-27 20:48:00 +00:00
|
|
|
if a.Metadata == nil {
|
2016-07-11 06:39:58 +00:00
|
|
|
a.Metadata = make(map[string]interface{})
|
|
|
|
}
|
2016-10-19 03:56:31 +00:00
|
|
|
|
|
|
|
if a.Writer == nil {
|
|
|
|
a.Writer = os.Stdout
|
|
|
|
}
|
2016-05-03 10:54:05 +00:00
|
|
|
}
|
|
|
|
|
2016-05-09 14:12:59 +00:00
|
|
|
// Run is the entry point to the cli app. Parses the arguments slice and routes
|
|
|
|
// to the proper flag/args combination
|
2016-05-03 10:54:05 +00:00
|
|
|
func (a *App) Run(arguments []string) (err error) {
|
|
|
|
a.Setup()
|
2013-07-20 15:44:09 +00:00
|
|
|
|
2016-11-04 20:56:28 +00:00
|
|
|
// handle the completion flag separately from the flagset since
|
|
|
|
// completion could be attempted after a flag, but before its value was put
|
|
|
|
// on the command line. this causes the flagset to interpret the completion
|
|
|
|
// flag name as the value of the flag before it which is undesirable
|
|
|
|
// note that we can only do this because the shell autocomplete function
|
|
|
|
// always appends the completion flag at the end of the command
|
2016-11-14 16:35:22 +00:00
|
|
|
shellComplete, arguments := checkShellCompleteFlag(a, arguments)
|
2016-11-04 20:56:28 +00:00
|
|
|
|
2013-07-19 22:10:34 +00:00
|
|
|
// parse flags
|
2016-09-17 23:54:29 +00:00
|
|
|
set, err := flagSet(a.Name, a.Flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-09-04 19:58:31 +00:00
|
|
|
set.SetOutput(ioutil.Discard)
|
2014-11-18 22:44:21 +00:00
|
|
|
err = set.Parse(arguments[1:])
|
2013-11-20 08:05:18 +00:00
|
|
|
nerr := normalizeFlags(a.Flags, set)
|
2016-02-04 07:25:41 +00:00
|
|
|
context := NewContext(a, set, nil)
|
2013-11-20 08:05:18 +00:00
|
|
|
if nerr != nil {
|
2014-12-02 04:57:35 +00:00
|
|
|
fmt.Fprintln(a.Writer, nerr)
|
2013-11-20 08:05:18 +00:00
|
|
|
ShowAppHelp(context)
|
2016-04-27 13:12:34 +00:00
|
|
|
return nerr
|
2013-11-20 08:05:18 +00:00
|
|
|
}
|
2016-11-14 16:35:22 +00:00
|
|
|
context.shellComplete = shellComplete
|
2013-09-04 19:58:31 +00:00
|
|
|
|
2015-12-12 23:31:32 +00:00
|
|
|
if checkCompletions(context) {
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2015-12-12 23:31:32 +00:00
|
|
|
}
|
|
|
|
|
2013-07-24 14:35:45 +00:00
|
|
|
if err != nil {
|
2016-01-23 11:47:24 +00:00
|
|
|
if a.OnUsageError != nil {
|
|
|
|
err := a.OnUsageError(context, err, false)
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, err)
|
2016-04-27 13:12:34 +00:00
|
|
|
return err
|
2016-01-23 11:47:24 +00:00
|
|
|
}
|
2016-10-07 14:38:36 +00:00
|
|
|
fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error())
|
2016-05-09 14:15:05 +00:00
|
|
|
ShowAppHelp(context)
|
|
|
|
return err
|
2013-07-24 14:35:45 +00:00
|
|
|
}
|
2013-07-19 15:34:01 +00:00
|
|
|
|
2015-09-28 06:42:17 +00:00
|
|
|
if !a.HideHelp && checkHelp(context) {
|
2015-10-19 00:02:23 +00:00
|
|
|
ShowAppHelp(context)
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2013-11-01 13:45:19 +00:00
|
|
|
}
|
|
|
|
|
2015-09-28 06:42:17 +00:00
|
|
|
if !a.HideVersion && checkVersion(context) {
|
2015-10-19 00:02:23 +00:00
|
|
|
ShowVersion(context)
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2013-11-01 13:45:19 +00:00
|
|
|
}
|
2013-07-20 15:44:09 +00:00
|
|
|
|
2014-11-18 22:44:21 +00:00
|
|
|
if a.After != nil {
|
|
|
|
defer func() {
|
2016-04-27 15:34:01 +00:00
|
|
|
if afterErr := a.After(context); afterErr != nil {
|
2015-06-02 04:11:20 +00:00
|
|
|
if err != nil {
|
|
|
|
err = NewMultiError(err, afterErr)
|
|
|
|
} else {
|
|
|
|
err = afterErr
|
|
|
|
}
|
|
|
|
}
|
2014-11-18 22:44:21 +00:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2014-01-01 21:00:20 +00:00
|
|
|
if a.Before != nil {
|
2016-04-28 21:15:16 +00:00
|
|
|
beforeErr := a.Before(context)
|
|
|
|
if beforeErr != nil {
|
2017-04-25 20:02:05 +00:00
|
|
|
fmt.Fprintf(a.Writer, "%v\n\n", beforeErr)
|
2015-12-25 21:08:22 +00:00
|
|
|
ShowAppHelp(context)
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, beforeErr)
|
2016-04-28 21:15:16 +00:00
|
|
|
err = beforeErr
|
2016-04-27 13:12:34 +00:00
|
|
|
return err
|
2014-01-01 21:00:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-19 15:34:01 +00:00
|
|
|
args := context.Args()
|
2013-11-24 13:40:21 +00:00
|
|
|
if args.Present() {
|
|
|
|
name := args.First()
|
2013-09-14 22:29:57 +00:00
|
|
|
c := a.Command(name)
|
|
|
|
if c != nil {
|
2013-11-01 13:45:19 +00:00
|
|
|
return c.Run(context)
|
2013-07-19 15:34:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-12 21:37:07 +00:00
|
|
|
if a.Action == nil {
|
|
|
|
a.Action = helpCommand.Action
|
|
|
|
}
|
|
|
|
|
2013-07-19 15:34:01 +00:00
|
|
|
// Run default Action
|
2016-04-27 15:34:01 +00:00
|
|
|
err = HandleAction(a.Action, context)
|
2016-04-27 13:12:34 +00:00
|
|
|
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, err)
|
2016-04-27 13:12:34 +00:00
|
|
|
return err
|
2013-07-19 15:34:01 +00:00
|
|
|
}
|
2013-09-14 22:29:57 +00:00
|
|
|
|
2016-08-21 20:51:55 +00:00
|
|
|
// RunAndExitOnError calls .Run() and exits non-zero if an error was returned
|
|
|
|
//
|
|
|
|
// Deprecated: instead you should return an error that fulfills cli.ExitCoder
|
|
|
|
// to cli.App.Run. This will cause the application to exit with the given eror
|
|
|
|
// code in the cli.ExitCoder
|
2016-04-27 16:23:09 +00:00
|
|
|
func (a *App) RunAndExitOnError() {
|
|
|
|
if err := a.Run(os.Args); err != nil {
|
2016-05-06 16:14:26 +00:00
|
|
|
fmt.Fprintln(a.errWriter(), err)
|
2016-05-02 15:25:37 +00:00
|
|
|
OsExiter(1)
|
2016-04-27 16:23:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-09 14:12:59 +00:00
|
|
|
// RunAsSubcommand invokes the subcommand given the context, parses ctx.Args() to
|
|
|
|
// generate command-specific flags
|
2016-04-27 13:12:34 +00:00
|
|
|
func (a *App) RunAsSubcommand(ctx *Context) (err error) {
|
2014-04-16 16:18:00 +00:00
|
|
|
// append help to commands
|
2014-04-17 16:48:00 +00:00
|
|
|
if len(a.Commands) > 0 {
|
2014-07-13 13:16:30 +00:00
|
|
|
if a.Command(helpCommand.Name) == nil && !a.HideHelp {
|
2014-04-17 16:48:00 +00:00
|
|
|
a.Commands = append(a.Commands, helpCommand)
|
2014-12-02 04:20:21 +00:00
|
|
|
if (HelpFlag != BoolFlag{}) {
|
|
|
|
a.appendFlag(HelpFlag)
|
|
|
|
}
|
2014-04-17 16:48:00 +00:00
|
|
|
}
|
2014-04-16 16:18:00 +00:00
|
|
|
}
|
|
|
|
|
2015-08-13 04:43:14 +00:00
|
|
|
newCmds := []Command{}
|
|
|
|
for _, c := range a.Commands {
|
2015-08-13 04:58:25 +00:00
|
|
|
if c.HelpName == "" {
|
|
|
|
c.HelpName = fmt.Sprintf("%s %s", a.HelpName, c.Name)
|
|
|
|
}
|
2015-08-13 04:43:14 +00:00
|
|
|
newCmds = append(newCmds, c)
|
|
|
|
}
|
|
|
|
a.Commands = newCmds
|
|
|
|
|
2014-04-16 16:18:00 +00:00
|
|
|
// parse flags
|
2016-09-17 23:54:29 +00:00
|
|
|
set, err := flagSet(a.Name, a.Flags)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2014-04-16 16:18:00 +00:00
|
|
|
set.SetOutput(ioutil.Discard)
|
2014-11-18 22:44:21 +00:00
|
|
|
err = set.Parse(ctx.Args().Tail())
|
2014-04-16 16:18:00 +00:00
|
|
|
nerr := normalizeFlags(a.Flags, set)
|
2015-05-18 15:39:48 +00:00
|
|
|
context := NewContext(a, set, ctx)
|
2014-04-16 16:18:00 +00:00
|
|
|
|
|
|
|
if nerr != nil {
|
2014-12-02 04:57:35 +00:00
|
|
|
fmt.Fprintln(a.Writer, nerr)
|
2015-06-25 05:46:33 +00:00
|
|
|
fmt.Fprintln(a.Writer)
|
2014-04-17 16:48:00 +00:00
|
|
|
if len(a.Commands) > 0 {
|
|
|
|
ShowSubcommandHelp(context)
|
|
|
|
} else {
|
|
|
|
ShowCommandHelp(ctx, context.Args().First())
|
|
|
|
}
|
2016-04-27 13:12:34 +00:00
|
|
|
return nerr
|
2014-04-16 16:18:00 +00:00
|
|
|
}
|
|
|
|
|
2015-12-12 23:31:32 +00:00
|
|
|
if checkCompletions(context) {
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2015-12-12 23:31:32 +00:00
|
|
|
}
|
|
|
|
|
2014-04-16 16:18:00 +00:00
|
|
|
if err != nil {
|
2016-01-23 11:47:24 +00:00
|
|
|
if a.OnUsageError != nil {
|
|
|
|
err = a.OnUsageError(context, err, true)
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, err)
|
2016-04-28 21:15:16 +00:00
|
|
|
return err
|
2016-01-23 11:47:24 +00:00
|
|
|
}
|
2016-10-07 14:38:36 +00:00
|
|
|
fmt.Fprintf(a.Writer, "%s %s\n\n", "Incorrect Usage.", err.Error())
|
2016-05-09 14:15:05 +00:00
|
|
|
ShowSubcommandHelp(context)
|
|
|
|
return err
|
2014-04-16 16:18:00 +00:00
|
|
|
}
|
|
|
|
|
2014-04-17 16:48:00 +00:00
|
|
|
if len(a.Commands) > 0 {
|
|
|
|
if checkSubcommandHelp(context) {
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2014-04-17 16:48:00 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if checkCommandHelp(ctx, context.Args().First()) {
|
2016-04-27 13:12:34 +00:00
|
|
|
return nil
|
2014-04-17 16:48:00 +00:00
|
|
|
}
|
2014-04-16 16:18:00 +00:00
|
|
|
}
|
|
|
|
|
2014-11-18 22:44:21 +00:00
|
|
|
if a.After != nil {
|
|
|
|
defer func() {
|
2016-04-27 13:12:34 +00:00
|
|
|
afterErr := a.After(context)
|
2015-06-02 04:11:20 +00:00
|
|
|
if afterErr != nil {
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, err)
|
2015-06-02 04:11:20 +00:00
|
|
|
if err != nil {
|
|
|
|
err = NewMultiError(err, afterErr)
|
|
|
|
} else {
|
|
|
|
err = afterErr
|
|
|
|
}
|
|
|
|
}
|
2014-11-18 22:44:21 +00:00
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
2014-04-16 20:26:28 +00:00
|
|
|
if a.Before != nil {
|
2016-04-28 21:15:16 +00:00
|
|
|
beforeErr := a.Before(context)
|
|
|
|
if beforeErr != nil {
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, beforeErr)
|
2016-04-28 21:15:16 +00:00
|
|
|
err = beforeErr
|
2016-04-27 13:12:34 +00:00
|
|
|
return err
|
2014-04-16 20:26:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-16 16:18:00 +00:00
|
|
|
args := context.Args()
|
|
|
|
if args.Present() {
|
|
|
|
name := args.First()
|
|
|
|
c := a.Command(name)
|
|
|
|
if c != nil {
|
|
|
|
return c.Run(context)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Run default Action
|
2016-04-27 15:34:01 +00:00
|
|
|
err = HandleAction(a.Action, context)
|
|
|
|
|
2017-06-28 16:52:12 +00:00
|
|
|
a.handleExitCoder(context, err)
|
2016-04-27 15:34:01 +00:00
|
|
|
return err
|
2014-04-16 16:18:00 +00:00
|
|
|
}
|
|
|
|
|
2016-05-09 14:12:59 +00:00
|
|
|
// Command returns the named command on App. Returns nil if the command does not exist
|
2013-09-14 22:29:57 +00:00
|
|
|
func (a *App) Command(name string) *Command {
|
|
|
|
for _, c := range a.Commands {
|
|
|
|
if c.HasName(name) {
|
|
|
|
return &c
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-05-03 10:54:05 +00:00
|
|
|
// Categories returns a slice containing all the categories with the commands they contain
|
2015-08-21 11:25:37 +00:00
|
|
|
func (a *App) Categories() CommandCategories {
|
|
|
|
return a.categories
|
|
|
|
}
|
|
|
|
|
2016-05-03 10:54:05 +00:00
|
|
|
// VisibleCategories returns a slice of categories and commands that are
|
|
|
|
// Hidden=false
|
|
|
|
func (a *App) VisibleCategories() []*CommandCategory {
|
|
|
|
ret := []*CommandCategory{}
|
|
|
|
for _, category := range a.categories {
|
|
|
|
if visible := func() *CommandCategory {
|
|
|
|
for _, command := range category.Commands {
|
|
|
|
if !command.Hidden {
|
|
|
|
return category
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}(); visible != nil {
|
|
|
|
ret = append(ret, visible)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
|
|
|
// VisibleCommands returns a slice of the Commands with Hidden=false
|
|
|
|
func (a *App) VisibleCommands() []Command {
|
|
|
|
ret := []Command{}
|
|
|
|
for _, command := range a.Commands {
|
|
|
|
if !command.Hidden {
|
|
|
|
ret = append(ret, command)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2019-01-27 16:41:54 +00:00
|
|
|
// Categories returns a slice containing all the categories with the commands they contain
|
2019-01-27 20:48:00 +00:00
|
|
|
func (a *App) VisibleFlagCategories() FlagCategories {
|
2019-01-27 16:41:54 +00:00
|
|
|
return a.flagCategories
|
|
|
|
}
|
|
|
|
|
2016-05-01 12:36:17 +00:00
|
|
|
// VisibleFlags returns a slice of the Flags with Hidden=false
|
|
|
|
func (a *App) VisibleFlags() []Flag {
|
|
|
|
return visibleFlags(a.Flags)
|
|
|
|
}
|
|
|
|
|
2013-09-14 22:29:57 +00:00
|
|
|
func (a *App) hasFlag(flag Flag) bool {
|
|
|
|
for _, f := range a.Flags {
|
|
|
|
if flag == f {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2016-05-06 16:14:26 +00:00
|
|
|
func (a *App) errWriter() io.Writer {
|
|
|
|
// When the app ErrWriter is nil use the package level one.
|
|
|
|
if a.ErrWriter == nil {
|
|
|
|
return ErrWriter
|
|
|
|
}
|
|
|
|
|
|
|
|
return a.ErrWriter
|
|
|
|
}
|
|
|
|
|
2013-09-14 22:29:57 +00:00
|
|
|
func (a *App) appendFlag(flag Flag) {
|
|
|
|
if !a.hasFlag(flag) {
|
|
|
|
a.Flags = append(a.Flags, flag)
|
|
|
|
}
|
|
|
|
}
|
2015-01-30 23:04:52 +00:00
|
|
|
|
2017-06-28 17:07:25 +00:00
|
|
|
func (a *App) handleExitCoder(context *Context, err error) {
|
2017-04-25 20:02:05 +00:00
|
|
|
if a.ExitErrHandler != nil {
|
2017-06-28 16:52:12 +00:00
|
|
|
a.ExitErrHandler(context, err)
|
2017-04-25 20:02:05 +00:00
|
|
|
} else {
|
|
|
|
HandleExitCoder(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-30 23:04:52 +00:00
|
|
|
// Author represents someone who has contributed to a cli project.
|
|
|
|
type Author struct {
|
|
|
|
Name string // The Authors name
|
|
|
|
Email string // The Authors email
|
|
|
|
}
|
|
|
|
|
|
|
|
// String makes Author comply to the Stringer interface, to allow an easy print in the templating process
|
|
|
|
func (a Author) String() string {
|
|
|
|
e := ""
|
|
|
|
if a.Email != "" {
|
2016-10-22 22:58:07 +00:00
|
|
|
e = " <" + a.Email + ">"
|
2015-01-30 23:04:52 +00:00
|
|
|
}
|
|
|
|
|
2016-10-22 22:58:07 +00:00
|
|
|
return fmt.Sprintf("%v%v", a.Name, e)
|
2015-01-30 23:04:52 +00:00
|
|
|
}
|
2016-04-27 15:34:01 +00:00
|
|
|
|
2016-11-02 03:33:12 +00:00
|
|
|
// HandleAction attempts to figure out which Action signature was used. If
|
|
|
|
// it's an ActionFunc or a func with the legacy signature for Action, the func
|
|
|
|
// is run!
|
2016-04-28 15:03:10 +00:00
|
|
|
func HandleAction(action interface{}, context *Context) (err error) {
|
2016-11-21 15:47:23 +00:00
|
|
|
if a, ok := action.(ActionFunc); ok {
|
|
|
|
return a(context)
|
|
|
|
} else if a, ok := action.(func(*Context) error); ok {
|
2016-11-02 03:33:12 +00:00
|
|
|
return a(context)
|
|
|
|
} else if a, ok := action.(func(*Context)); ok { // deprecated function signature
|
|
|
|
a(context)
|
2016-04-29 07:01:57 +00:00
|
|
|
return nil
|
2016-04-27 15:34:01 +00:00
|
|
|
}
|
2017-10-06 05:29:13 +00:00
|
|
|
|
|
|
|
return errInvalidActionType
|
2016-04-27 15:34:01 +00:00
|
|
|
}
|