urfave-cli/app.go

377 lines
8.7 KiB
Go
Raw Normal View History

package cli
import (
"fmt"
"io"
"io/ioutil"
"os"
"path/filepath"
"sort"
2013-11-15 11:40:18 +00:00
"time"
)
var (
// DefaultSuccessExitCode is the default for use with os.Exit intended to
// indicate success
DefaultSuccessExitCode = 0
// DefaultErrorExitCode is the default for use with os.Exit intended to
// indicate an error
DefaultErrorExitCode = 1
)
// 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
type App struct {
// The name of the program. Defaults to path.Base(os.Args[0])
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
// 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
// Version of the program
Version string
// List of commands to execute
Commands []Command
2013-07-20 22:50:13 +00:00
// List of flags to parse
Flags []Flag
// 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
// Populate on app startup, only gettable throught method Categories()
categories CommandCategories
// An action to execute when the bash-completion flag is set
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
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
// It is run even if Action() panics
After AfterFunc
// The action to execute when no subcommands are specified
Action ActionFunc
// Execute this function if the proper command cannot be found
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
// List of all authors who contributed
Authors []Author
// Copyright of the binary if any
Copyright string
// 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
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-11-01 14:31:37 +00:00
// Creates a new cli Application with some reasonable defaults for Name, Usage, Version and Action.
func NewApp() *App {
return &App{
Name: filepath.Base(os.Args[0]),
HelpName: filepath.Base(os.Args[0]),
Usage: "A new cli application",
2015-10-25 05:51:06 +00:00
UsageText: "",
Version: "0.0.0",
BashComplete: DefaultAppComplete,
Action: helpCommand.Action,
Compiled: compileTime(),
2014-12-02 04:50:04 +00:00
Writer: os.Stdout,
}
}
2013-11-01 14:31:37 +00:00
// Entry point to the cli app. Parses the arguments slice and routes to the proper flag/args combination
func (a *App) Run(arguments []string) (ec int, err error) {
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-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
a.categories = CommandCategories{}
for _, command := range a.Commands {
a.categories = a.categories.AddCommand(command.Category, command)
}
sort.Sort(a.categories)
2013-07-20 15:44:09 +00:00
// append help to commands
2014-07-13 13:16:30 +00:00
if a.Command(helpCommand.Name) == nil && !a.HideHelp {
a.Commands = append(a.Commands, helpCommand)
if (HelpFlag != BoolFlag{}) {
a.appendFlag(HelpFlag)
}
}
//append version/help flags
if a.EnableBashCompletion {
a.appendFlag(BashCompletionFlag)
}
2014-11-12 11:38:58 +00:00
if !a.HideVersion {
a.appendFlag(VersionFlag)
}
2013-07-20 15:44:09 +00:00
// parse flags
2013-07-20 15:21:20 +00:00
set := flagSet(a.Name, a.Flags)
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)
context := NewContext(a, set, nil)
2013-11-20 08:05:18 +00:00
if nerr != nil {
fmt.Fprintln(a.Writer, nerr)
2013-11-20 08:05:18 +00:00
ShowAppHelp(context)
return DefaultErrorExitCode, nerr
2013-11-20 08:05:18 +00:00
}
if checkCompletions(context) {
return DefaultSuccessExitCode, nil
}
2013-07-24 14:35:45 +00:00
if err != nil {
if a.OnUsageError != nil {
err := a.OnUsageError(context, err, false)
if err != nil {
return DefaultErrorExitCode, err
}
return DefaultSuccessExitCode, err
} else {
fmt.Fprintf(a.Writer, "%s\n\n", "Incorrect Usage.")
ShowAppHelp(context)
return DefaultErrorExitCode, err
}
2013-07-24 14:35:45 +00:00
}
if !a.HideHelp && checkHelp(context) {
ShowAppHelp(context)
return DefaultSuccessExitCode, nil
}
if !a.HideVersion && checkVersion(context) {
ShowVersion(context)
return DefaultSuccessExitCode, nil
}
2013-07-20 15:44:09 +00:00
2014-11-18 22:44:21 +00:00
if a.After != nil {
defer func() {
afterEc, afterErr := a.After(context)
if afterErr != nil {
if err != nil {
err = NewMultiError(err, afterErr)
} else {
err = afterErr
}
}
ec = afterEc
2014-11-18 22:44:21 +00:00
}()
}
2014-01-01 21:00:20 +00:00
if a.Before != nil {
ec, err = a.Before(context)
2014-01-01 21:00:20 +00:00
if err != nil {
fmt.Fprintf(a.Writer, "%v\n\n", err)
ShowAppHelp(context)
return ec, err
2014-01-01 21:00:20 +00:00
}
}
args := context.Args()
2013-11-24 13:40:21 +00:00
if args.Present() {
name := args.First()
c := a.Command(name)
if c != nil {
return c.Run(context)
}
}
// Run default Action
return a.Action(context), nil
}
2014-07-13 16:26:20 +00:00
// Another entry point to the cli app, takes care of passing arguments and error handling
func (a *App) RunAndExitOnError() {
if exitCode, err := a.Run(os.Args); err != nil {
fmt.Fprintln(os.Stderr, err)
os.Exit(exitCode)
2014-07-13 16:26:20 +00:00
}
}
// Invokes the subcommand given the context, parses ctx.Args() to generate command-specific flags
func (a *App) RunAsSubcommand(ctx *Context) (ec int, err error) {
// append help to commands
if len(a.Commands) > 0 {
2014-07-13 13:16:30 +00:00
if a.Command(helpCommand.Name) == nil && !a.HideHelp {
a.Commands = append(a.Commands, helpCommand)
if (HelpFlag != BoolFlag{}) {
a.appendFlag(HelpFlag)
}
}
}
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
// append flags
if a.EnableBashCompletion {
a.appendFlag(BashCompletionFlag)
}
// parse flags
set := flagSet(a.Name, a.Flags)
set.SetOutput(ioutil.Discard)
2014-11-18 22:44:21 +00:00
err = set.Parse(ctx.Args().Tail())
nerr := normalizeFlags(a.Flags, set)
context := NewContext(a, set, ctx)
if nerr != nil {
fmt.Fprintln(a.Writer, nerr)
fmt.Fprintln(a.Writer)
if len(a.Commands) > 0 {
ShowSubcommandHelp(context)
} else {
ShowCommandHelp(ctx, context.Args().First())
}
return DefaultErrorExitCode, nerr
}
if checkCompletions(context) {
return DefaultSuccessExitCode, nil
}
if err != nil {
if a.OnUsageError != nil {
err = a.OnUsageError(context, err, true)
if err != nil {
return DefaultErrorExitCode, err
}
return DefaultSuccessExitCode, err
} else {
fmt.Fprintf(a.Writer, "%s\n\n", "Incorrect Usage.")
ShowSubcommandHelp(context)
return DefaultErrorExitCode, err
}
}
if len(a.Commands) > 0 {
if checkSubcommandHelp(context) {
return DefaultSuccessExitCode, nil
}
} else {
if checkCommandHelp(ctx, context.Args().First()) {
return DefaultSuccessExitCode, nil
}
}
2014-11-18 22:44:21 +00:00
if a.After != nil {
defer func() {
afterEc, afterErr := a.After(context)
if afterErr != nil {
if err != nil {
err = NewMultiError(err, afterErr)
} else {
err = afterErr
}
}
ec = afterEc
2014-11-18 22:44:21 +00:00
}()
}
if a.Before != nil {
ec, err = a.Before(context)
if err != nil {
return ec, err
}
}
args := context.Args()
if args.Present() {
name := args.First()
c := a.Command(name)
if c != nil {
return c.Run(context)
}
}
// Run default Action
return a.Action(context), nil
}
2013-11-01 14:31:37 +00:00
// Returns the named command on App. Returns nil if the command does not exist
func (a *App) Command(name string) *Command {
for _, c := range a.Commands {
if c.HasName(name) {
return &c
}
}
return nil
}
// Returnes the array containing all the categories with the commands they contain
func (a *App) Categories() CommandCategories {
return a.categories
}
func (a *App) hasFlag(flag Flag) bool {
for _, f := range a.Flags {
if flag == f {
return true
}
}
return false
}
func (a *App) appendFlag(flag Flag) {
if !a.hasFlag(flag) {
a.Flags = append(a.Flags, flag)
}
}
// 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 != "" {
e = "<" + a.Email + "> "
}
return fmt.Sprintf("%v %v", a.Name, e)
}