2022-05-08 15:38:41 +00:00
|
|
|
|
# v2 guide
|
2020-04-04 10:09:14 +00:00
|
|
|
|
|
2019-10-24 05:15:30 +00:00
|
|
|
|
## Getting Started
|
|
|
|
|
|
|
|
|
|
One of the philosophies behind cli is that an API should be playful and full of
|
|
|
|
|
discovery. So a cli app can be as little as one line of code in `main()`.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "A new cli application"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
(&cli.App{}).Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
This app will run and show help text, but is not very useful. Let's give an
|
|
|
|
|
action to execute and some help documentation:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "boom! I say!"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Name: "boom",
|
|
|
|
|
Usage: "make an explosive entrance",
|
|
|
|
|
Action: func(*cli.Context) error {
|
|
|
|
|
fmt.Println("boom! I say!")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Running this already gives you a ton of functionality, plus support for things
|
|
|
|
|
like subcommands and flags, which are covered below.
|
|
|
|
|
|
|
|
|
|
## Examples
|
|
|
|
|
|
|
|
|
|
Being a programmer can be a lonely job. Thankfully by the power of automation
|
|
|
|
|
that is not the case! Let's create a greeter app to fend off our demons of
|
|
|
|
|
loneliness!
|
|
|
|
|
|
|
|
|
|
Start by creating a directory named `greet`, and within it, add a file,
|
|
|
|
|
`greet.go` with the following code in it:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "Hello friend!"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Name: "greet",
|
|
|
|
|
Usage: "fight the loneliness!",
|
|
|
|
|
Action: func(*cli.Context) error {
|
|
|
|
|
fmt.Println("Hello friend!")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Install our command to the `$GOPATH/bin` directory:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
2019-10-24 05:15:30 +00:00
|
|
|
|
$ go install
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Finally run our new command:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
2019-10-24 05:15:30 +00:00
|
|
|
|
$ greet
|
|
|
|
|
Hello friend!
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
cli also generates neat help text:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
2019-10-24 05:15:30 +00:00
|
|
|
|
$ greet help
|
|
|
|
|
NAME:
|
|
|
|
|
greet - fight the loneliness!
|
|
|
|
|
|
|
|
|
|
USAGE:
|
|
|
|
|
greet [global options] command [command options] [arguments...]
|
|
|
|
|
|
|
|
|
|
COMMANDS:
|
|
|
|
|
help, h Shows a list of commands or help for one command
|
|
|
|
|
|
|
|
|
|
GLOBAL OPTIONS
|
2019-11-28 07:01:14 +00:00
|
|
|
|
--help, -h show help (default: false)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Arguments
|
|
|
|
|
|
|
|
|
|
You can lookup arguments by calling the `Args` function on `cli.Context`, e.g.:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "Hello \""
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Printf("Hello %q", cCtx.Args().Get(0))
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Flags
|
|
|
|
|
|
|
|
|
|
Setting and querying flags is simple.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "Hello Nefertiti"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
name := "Nefertiti"
|
|
|
|
|
if cCtx.NArg() > 0 {
|
|
|
|
|
name = cCtx.Args().Get(0)
|
|
|
|
|
}
|
|
|
|
|
if cCtx.String("lang") == "spanish" {
|
|
|
|
|
fmt.Println("Hola", name)
|
|
|
|
|
} else {
|
|
|
|
|
fmt.Println("Hello", name)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
You can also set a destination variable for a flag, to which the content will be
|
|
|
|
|
scanned.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "Hello someone"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
var language string
|
|
|
|
|
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
Destination: &language,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
name := "someone"
|
|
|
|
|
if cCtx.NArg() > 0 {
|
|
|
|
|
name = cCtx.Args().Get(0)
|
|
|
|
|
}
|
|
|
|
|
if language == "spanish" {
|
|
|
|
|
fmt.Println("Hola", name)
|
|
|
|
|
} else {
|
|
|
|
|
fmt.Println("Hello", name)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-06-15 15:04:06 +00:00
|
|
|
|
See full list of flags at https://pkg.go.dev/github.com/urfave/cli/v2
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
#### Placeholder Values
|
|
|
|
|
|
|
|
|
|
Sometimes it's useful to specify a flag's value within the usage string itself.
|
|
|
|
|
Such placeholders are indicated with back quotes.
|
|
|
|
|
|
|
|
|
|
For example this:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "--config FILE, -c FILE"
|
|
|
|
|
} -->
|
|
|
|
|
```go
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "config",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "Load configuration from `FILE`",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Will result in help output like:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
--config FILE, -c FILE Load configuration from FILE
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Note that only the first placeholder is used. Subsequent back-quoted words will
|
|
|
|
|
be left as-is.
|
|
|
|
|
|
|
|
|
|
#### Alternate Names
|
|
|
|
|
|
|
|
|
|
You can set alternate (or short) names for flags by providing a comma-delimited
|
|
|
|
|
list for the `Name`. e.g.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "--lang value, -l value.*language for the greeting.*default: \"english\""
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Aliases: []string{"l"},
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
That flag can then be set with `--lang spanish` or `-l spanish`. Note that
|
|
|
|
|
giving two different forms of the same flag in the same command invocation is an
|
|
|
|
|
error.
|
|
|
|
|
|
|
|
|
|
#### Ordering
|
|
|
|
|
|
|
|
|
|
Flags for the application and commands are shown in the order they are defined.
|
|
|
|
|
However, it's possible to sort them from outside this library by using `FlagsByName`
|
|
|
|
|
or `CommandsByName` with `sort`.
|
|
|
|
|
|
|
|
|
|
For example this:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
2022-05-23 11:44:58 +00:00
|
|
|
|
"output": ".*Load configuration from FILE\n.*\n.*Language for the greeting.*"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
|
|
|
|
"sort"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Aliases: []string{"l"},
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "Language for the greeting",
|
|
|
|
|
},
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "config",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "Load configuration from `FILE`",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "complete",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "complete a task on the list",
|
|
|
|
|
Action: func(*cli.Context) error {
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Aliases: []string{"a"},
|
|
|
|
|
Usage: "add a task to the list",
|
|
|
|
|
Action: func(*cli.Context) error {
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sort.Sort(cli.FlagsByName(app.Flags))
|
|
|
|
|
sort.Sort(cli.CommandsByName(app.Commands))
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Will result in help output like:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
--config FILE, -c FILE Load configuration from FILE
|
|
|
|
|
--lang value, -l value Language for the greeting (default: "english")
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
#### Values from the Environment
|
|
|
|
|
|
|
|
|
|
You can also have the default value set from the environment via `EnvVars`. e.g.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "language for the greeting.*APP_LANG"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Aliases: []string{"l"},
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
EnvVars: []string{"APP_LANG"},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
If `EnvVars` contains more than one string, the first environment variable that
|
2021-09-16 03:16:31 +00:00
|
|
|
|
resolves is used.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "language for the greeting.*LEGACY_COMPAT_LANG.*APP_LANG.*LANG"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Aliases: []string{"l"},
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
EnvVars: []string{"LEGACY_COMPAT_LANG", "APP_LANG", "LANG"},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
#### Values from files
|
|
|
|
|
|
|
|
|
|
You can also have the default value set from file via `FilePath`. e.g.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "password for the mysql database"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "password",
|
|
|
|
|
Aliases: []string{"p"},
|
|
|
|
|
Usage: "password for the mysql database",
|
|
|
|
|
FilePath: "/etc/mysql/password",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Note that default values set from file (e.g. `FilePath`) take precedence over
|
|
|
|
|
default values set from the environment (e.g. `EnvVar`).
|
|
|
|
|
|
|
|
|
|
#### Values from alternate input sources (YAML, TOML, and others)
|
|
|
|
|
|
|
|
|
|
There is a separate package altsrc that adds support for getting flag values
|
|
|
|
|
from other file input sources.
|
|
|
|
|
|
|
|
|
|
Currently supported input source formats:
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2022-06-22 12:29:12 +00:00
|
|
|
|
- YAML
|
|
|
|
|
- JSON
|
|
|
|
|
- TOML
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
In order to get values for a flag from an alternate input source the following
|
|
|
|
|
code would be added to wrap an existing cli.Flag like below:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
|
|
|
|
// --- >8 ---
|
2019-10-24 05:15:30 +00:00
|
|
|
|
altsrc.NewIntFlag(&cli.IntFlag{Name: "test"})
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Initialization must also occur for these flags. Below is an example initializing
|
|
|
|
|
getting data from a yaml file below.
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
|
|
|
|
// --- >8 ---
|
2019-10-24 05:15:30 +00:00
|
|
|
|
command.Before = altsrc.InitInputSourceWithContext(command.Flags, NewYamlSourceFromFlagFunc("load"))
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The code above will use the "load" string as a flag name to get the file name of
|
|
|
|
|
a yaml file from the cli.Context. It will then use that file name to initialize
|
|
|
|
|
the yaml input source for any flags that are defined on that command. As a note
|
|
|
|
|
the "load" flag used would also have to be defined on the command flags in order
|
|
|
|
|
for this code snippet to work.
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Currently only YAML, JSON, and TOML files are supported but developers can add
|
|
|
|
|
support for other input sources by implementing the altsrc.InputSourceContext
|
|
|
|
|
for their given sources.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
Here is a more complete sample of a command using YAML support:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
2021-01-25 01:54:48 +00:00
|
|
|
|
"args": ["--help"],
|
2019-10-24 05:15:30 +00:00
|
|
|
|
"output": "--test value.*default: 0"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
|
|
|
|
"github.com/urfave/cli/v2/altsrc"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
flags := []cli.Flag{
|
|
|
|
|
altsrc.NewIntFlag(&cli.IntFlag{Name: "test"}),
|
|
|
|
|
&cli.StringFlag{Name: "load"},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
Action: func(*cli.Context) error {
|
|
|
|
|
fmt.Println("--test value.*default: 0")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
Before: altsrc.InitInputSourceWithContext(flags, altsrc.NewYamlSourceFromFlagFunc("load")),
|
|
|
|
|
Flags: flags,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app.Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2019-12-21 22:36:02 +00:00
|
|
|
|
#### Required Flags
|
|
|
|
|
|
2019-12-21 22:47:18 +00:00
|
|
|
|
You can make a flag required by setting the `Required` field to `true`. If a user
|
|
|
|
|
does not provide a required flag, they will be shown an error message.
|
2019-12-21 22:36:02 +00:00
|
|
|
|
|
2020-07-01 20:20:25 +00:00
|
|
|
|
Take for example this app that requires the `lang` flag:
|
2019-12-21 22:36:02 +00:00
|
|
|
|
|
2019-12-21 23:48:07 +00:00
|
|
|
|
<!-- {
|
|
|
|
|
"error": "Required flag \"lang\" not set"
|
|
|
|
|
} -->
|
2019-12-21 22:36:02 +00:00
|
|
|
|
```go
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2022-04-20 18:45:04 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-12-21 22:36:02 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.StringFlag{
|
|
|
|
|
Name: "lang",
|
|
|
|
|
Value: "english",
|
|
|
|
|
Usage: "language for the greeting",
|
|
|
|
|
Required: true,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
output := "Hello"
|
|
|
|
|
if cCtx.String("lang") == "spanish" {
|
|
|
|
|
output = "Hola"
|
|
|
|
|
}
|
|
|
|
|
fmt.Println(output)
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-12-21 22:36:02 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2019-12-21 22:47:18 +00:00
|
|
|
|
If the app is run without the `lang` flag, the user will see the following message
|
2019-12-21 22:36:02 +00:00
|
|
|
|
|
|
|
|
|
```
|
2019-12-21 22:47:18 +00:00
|
|
|
|
Required flag "lang" not set
|
2019-12-21 22:36:02 +00:00
|
|
|
|
```
|
|
|
|
|
|
2019-10-24 05:15:30 +00:00
|
|
|
|
#### Default Values for help output
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Sometimes it's useful to specify a flag's default help-text value within the
|
|
|
|
|
flag declaration. This can be useful if the default value for a flag is a
|
|
|
|
|
computed value. The default value can be set via the `DefaultText` struct field.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
For example this:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--help"],
|
|
|
|
|
"output": "--port value"
|
|
|
|
|
} -->
|
|
|
|
|
```go
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.IntFlag{
|
|
|
|
|
Name: "port",
|
|
|
|
|
Usage: "Use a randomized port",
|
|
|
|
|
Value: 0,
|
|
|
|
|
DefaultText: "random",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Will result in help output like:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
--port value Use a randomized port (default: random)
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
#### Precedence
|
|
|
|
|
|
|
|
|
|
The precedence for flag value sources is as follows (highest to lowest):
|
|
|
|
|
|
|
|
|
|
0. Command line flag value from user
|
|
|
|
|
0. Environment variable (if specified)
|
|
|
|
|
0. Configuration file (if specified)
|
|
|
|
|
0. Default defined on the flag
|
|
|
|
|
|
|
|
|
|
### Subcommands
|
|
|
|
|
|
|
|
|
|
Subcommands can be defined for a more git-like command line app.
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["template", "add"],
|
|
|
|
|
"output": "new task template: .+"
|
|
|
|
|
} -->
|
|
|
|
|
```go
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Aliases: []string{"a"},
|
|
|
|
|
Usage: "add a task to the list",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("added task: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "complete",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "complete a task on the list",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("completed task: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "template",
|
|
|
|
|
Aliases: []string{"t"},
|
|
|
|
|
Usage: "options for task templates",
|
|
|
|
|
Subcommands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Usage: "add a new template",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("new task template: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "remove",
|
|
|
|
|
Usage: "remove an existing template",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("removed task template: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Subcommands categories
|
|
|
|
|
|
|
|
|
|
For additional organization in apps that have many subcommands, you can
|
|
|
|
|
associate a category for each command to group them together in the help
|
2022-06-20 21:08:10 +00:00
|
|
|
|
output, e.g.:
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
```go
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "noop",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Category: "template",
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
Name: "remove",
|
|
|
|
|
Category: "template",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Will include:
|
|
|
|
|
|
|
|
|
|
```
|
|
|
|
|
COMMANDS:
|
|
|
|
|
noop
|
|
|
|
|
|
|
|
|
|
Template actions:
|
|
|
|
|
add
|
|
|
|
|
remove
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Exit code
|
|
|
|
|
|
|
|
|
|
Calling `App.Run` will not automatically call `os.Exit`, which means that by
|
|
|
|
|
default the exit code will "fall through" to being `0`. An explicit exit code
|
|
|
|
|
may be set by returning a non-nil error that fulfills `cli.ExitCoder`, *or* a
|
|
|
|
|
`cli.MultiError` that includes an error that fulfills `cli.ExitCoder`, e.g.:
|
|
|
|
|
<!-- {
|
|
|
|
|
"error": "Ginger croutons are not in the soup"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.BoolFlag{
|
|
|
|
|
Name: "ginger-crouton",
|
|
|
|
|
Usage: "is it in the soup?",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Action: func(ctx *cli.Context) error {
|
|
|
|
|
if !ctx.Bool("ginger-crouton") {
|
|
|
|
|
return cli.Exit("Ginger croutons are not in the soup", 86)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Combining short options
|
|
|
|
|
|
|
|
|
|
Traditional use of options using their shortnames look like this:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
2019-10-24 05:15:30 +00:00
|
|
|
|
$ cmd -s -o -m "Some message"
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
Suppose you want users to be able to combine options with their shortnames. This
|
|
|
|
|
can be done using the `UseShortOptionHandling` bool in your app configuration,
|
|
|
|
|
or for individual commands by attaching it to the command configuration. For
|
|
|
|
|
example:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["short", "-som", "Some message"],
|
|
|
|
|
"output": "serve: true\noption: true\nmessage: Some message\n"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
UseShortOptionHandling: true,
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "short",
|
|
|
|
|
Usage: "complete a task on the list",
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.BoolFlag{Name: "serve", Aliases: []string{"s"}},
|
|
|
|
|
&cli.BoolFlag{Name: "option", Aliases: []string{"o"}},
|
|
|
|
|
&cli.StringFlag{Name: "message", Aliases: []string{"m"}},
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("serve:", cCtx.Bool("serve"))
|
|
|
|
|
fmt.Println("option:", cCtx.Bool("option"))
|
|
|
|
|
fmt.Println("message:", cCtx.String("message"))
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
If your program has any number of bool flags such as `serve` and `option`, and
|
|
|
|
|
optionally one non-bool flag `message`, with the short options of `-s`, `-o`,
|
|
|
|
|
and `-m` respectively, setting `UseShortOptionHandling` will also support the
|
|
|
|
|
following syntax:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
2019-10-24 05:15:30 +00:00
|
|
|
|
$ cmd -som "Some message"
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
If you enable `UseShortOptionHandling`, then you must not use any flags that
|
|
|
|
|
have a single leading `-` or this will result in failures. For example,
|
|
|
|
|
`-option` can no longer be used. Flags with two leading dashes (such as
|
|
|
|
|
`--options`) are still valid.
|
|
|
|
|
|
|
|
|
|
### Bash Completion
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
You can enable completion commands by setting the `EnableBashCompletion` flag on
|
|
|
|
|
the `App` object to `true`. By default, this setting will allow auto-completion
|
|
|
|
|
for an app's subcommands, but you can write your own completion methods for the
|
|
|
|
|
App or its subcommands as well.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
#### Default auto-completion
|
|
|
|
|
|
|
|
|
|
```go
|
|
|
|
|
package main
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2020-03-18 06:32:57 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2020-02-06 00:51:33 +00:00
|
|
|
|
)
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
EnableBashCompletion: true,
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Aliases: []string{"a"},
|
|
|
|
|
Usage: "add a task to the list",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("added task: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
2020-02-06 00:51:33 +00:00
|
|
|
|
},
|
2022-06-20 21:08:10 +00:00
|
|
|
|
{
|
|
|
|
|
Name: "complete",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "complete a task on the list",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("completed task: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
2020-02-06 00:51:33 +00:00
|
|
|
|
},
|
2022-06-20 21:08:10 +00:00
|
|
|
|
{
|
|
|
|
|
Name: "template",
|
|
|
|
|
Aliases: []string{"t"},
|
|
|
|
|
Usage: "options for task templates",
|
|
|
|
|
Subcommands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "add",
|
|
|
|
|
Usage: "add a new template",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("new task template: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
2020-02-06 00:51:33 +00:00
|
|
|
|
},
|
2022-06-20 21:08:10 +00:00
|
|
|
|
{
|
|
|
|
|
Name: "remove",
|
|
|
|
|
Usage: "remove an existing template",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("removed task template: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
2020-02-06 00:51:33 +00:00
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
2020-02-06 00:51:33 +00:00
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
![](/docs/v2/images/default-bash-autocomplete.gif)
|
|
|
|
|
|
|
|
|
|
#### Custom auto-completion
|
2019-10-24 05:15:30 +00:00
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["complete", "--generate-bash-completion"],
|
|
|
|
|
"output": "laundry"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
tasks := []string{"cook", "clean", "laundry", "eat", "sleep", "code"}
|
|
|
|
|
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
EnableBashCompletion: true,
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "complete",
|
|
|
|
|
Aliases: []string{"c"},
|
|
|
|
|
Usage: "complete a task on the list",
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Println("completed task: ", cCtx.Args().First())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
BashComplete: func(cCtx *cli.Context) {
|
|
|
|
|
// This will complete if no args are passed
|
|
|
|
|
if cCtx.NArg() > 0 {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
for _, t := range tasks {
|
|
|
|
|
fmt.Println(t)
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
2020-02-06 00:51:33 +00:00
|
|
|
|
![](/docs/v2/images/custom-bash-autocomplete.gif)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
#### Enabling
|
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
To enable auto-completion for the current shell session, a bash script,
|
|
|
|
|
`autocomplete/bash_autocomplete` is included in this repo.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
To use `autocomplete/bash_autocomplete` set an environment variable named `PROG`
|
|
|
|
|
to the name of your program and then `source` the
|
|
|
|
|
`autocomplete/bash_autocomplete` file.
|
2020-02-06 00:51:33 +00:00
|
|
|
|
|
|
|
|
|
For example, if your cli program is called `myprogram`:
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
|
|
|
|
$ PROG=myprogram source path/to/cli/autocomplete/bash_autocomplete
|
|
|
|
|
```
|
2020-02-06 00:51:33 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Auto-completion is now enabled for the current shell, but will not persist into
|
|
|
|
|
a new shell.
|
2020-02-06 00:51:33 +00:00
|
|
|
|
|
|
|
|
|
#### Distribution and Persistent Autocompletion
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
Copy `autocomplete/bash_autocomplete` into `/etc/bash_completion.d/` and rename
|
|
|
|
|
it to the name of the program you wish to add autocomplete support for (or
|
|
|
|
|
automatically install it there if you are distributing a package). Don't forget
|
2020-02-06 00:51:33 +00:00
|
|
|
|
to source the file or restart your shell to activate the auto-completion.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
|
|
|
|
$ sudo cp path/to/autocomplete/bash_autocomplete /etc/bash_completion.d/<myprogram>
|
|
|
|
|
$ source /etc/bash_completion.d/<myprogram>
|
2019-10-24 05:15:30 +00:00
|
|
|
|
```
|
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
Alternatively, you can just document that users should `source` the generic
|
2022-06-20 21:08:10 +00:00
|
|
|
|
`autocomplete/bash_autocomplete` and set `$PROG` within their bash configuration
|
2020-02-06 00:51:33 +00:00
|
|
|
|
file, adding these lines:
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
|
|
|
|
$ PROG=<myprogram>
|
|
|
|
|
$ source path/to/cli/autocomplete/bash_autocomplete
|
2020-02-06 00:51:33 +00:00
|
|
|
|
```
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Keep in mind that if they are enabling auto-completion for more than one
|
|
|
|
|
program, they will need to set `PROG` and source
|
|
|
|
|
`autocomplete/bash_autocomplete` for each program, like so:
|
|
|
|
|
|
|
|
|
|
```sh-session
|
|
|
|
|
$ PROG=<program1>
|
|
|
|
|
$ source path/to/cli/autocomplete/bash_autocomplete
|
|
|
|
|
|
|
|
|
|
$ PROG=<program2>
|
|
|
|
|
$ source path/to/cli/autocomplete/bash_autocomplete
|
2020-02-06 00:51:33 +00:00
|
|
|
|
```
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
#### Customization
|
|
|
|
|
|
|
|
|
|
The default shell completion flag (`--generate-bash-completion`) is defined as
|
|
|
|
|
`cli.EnableBashCompletion`, and may be redefined if desired, e.g.:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--generate-bash-completion"],
|
|
|
|
|
"output": "wat\nhelp\nh"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
EnableBashCompletion: true,
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
{
|
|
|
|
|
Name: "wat",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
#### ZSH Support
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Auto-completion for ZSH is also supported using the
|
|
|
|
|
`autocomplete/zsh_autocomplete` file included in this repo. One environment
|
|
|
|
|
variable is used, `PROG`. Set `PROG` to the program name as before, and then
|
|
|
|
|
`source path/to/autocomplete/zsh_autocomplete`. Adding the following lines to
|
|
|
|
|
your ZSH configuration file (usually `.zshrc`) will allow the auto-completion to
|
|
|
|
|
persist across new shells:
|
|
|
|
|
|
|
|
|
|
```sh-session
|
|
|
|
|
$ PROG=<myprogram>
|
|
|
|
|
$ source path/to/autocomplete/zsh_autocomplete
|
2020-02-06 00:51:33 +00:00
|
|
|
|
```
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
#### ZSH default auto-complete example
|
|
|
|
|
![](/docs/v2/images/default-zsh-autocomplete.gif)
|
2022-06-20 21:08:10 +00:00
|
|
|
|
|
2020-02-06 00:51:33 +00:00
|
|
|
|
#### ZSH custom auto-complete example
|
|
|
|
|
![](/docs/v2/images/custom-zsh-autocomplete.gif)
|
|
|
|
|
|
2020-05-18 03:19:00 +00:00
|
|
|
|
#### PowerShell Support
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Auto-completion for PowerShell is also supported using the
|
|
|
|
|
`autocomplete/powershell_autocomplete.ps1` file included in this repo.
|
2020-05-18 03:19:00 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Rename the script to `<my program>.ps1` and move it anywhere in your file
|
|
|
|
|
system. The location of script does not matter, only the file name of the
|
|
|
|
|
script has to match the your program's binary name.
|
2020-05-18 03:19:00 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
To activate it, enter:
|
|
|
|
|
|
|
|
|
|
```powershell
|
2020-05-18 03:19:00 +00:00
|
|
|
|
& path/to/autocomplete/<my program>.ps1
|
|
|
|
|
```
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
To persist across new shells, open the PowerShell profile (with `code $profile`
|
|
|
|
|
or `notepad $profile`) and add the line:
|
|
|
|
|
|
|
|
|
|
```powershell
|
|
|
|
|
& path/to/autocomplete/<my program>.ps1
|
|
|
|
|
```
|
2020-05-18 03:19:00 +00:00
|
|
|
|
|
2019-10-24 05:15:30 +00:00
|
|
|
|
### Generated Help Text
|
|
|
|
|
|
|
|
|
|
The default help flag (`-h/--help`) is defined as `cli.HelpFlag` and is checked
|
|
|
|
|
by the cli internals in order to print generated help text for the app, command,
|
|
|
|
|
or subcommand, and break execution.
|
|
|
|
|
|
|
|
|
|
#### Customization
|
|
|
|
|
|
|
|
|
|
All of the help text generation may be customized, and at multiple levels. The
|
|
|
|
|
templates are exposed as variables `AppHelpTemplate`, `CommandHelpTemplate`, and
|
|
|
|
|
`SubcommandHelpTemplate` which may be reassigned or augmented, and full override
|
|
|
|
|
is possible by assigning a compatible func to the `cli.HelpPrinter` variable,
|
|
|
|
|
e.g.:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "Ha HA. I pwnd the help!!1"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
// EXAMPLE: Append to an existing template
|
|
|
|
|
cli.AppHelpTemplate = fmt.Sprintf(`%s
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
WEBSITE: http://awesometown.example.com
|
|
|
|
|
|
|
|
|
|
SUPPORT: support@awesometown.example.com
|
|
|
|
|
|
|
|
|
|
`, cli.AppHelpTemplate)
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
// EXAMPLE: Override a template
|
|
|
|
|
cli.AppHelpTemplate = `NAME:
|
2019-10-24 05:15:30 +00:00
|
|
|
|
{{.Name}} - {{.Usage}}
|
|
|
|
|
USAGE:
|
|
|
|
|
{{.HelpName}} {{if .VisibleFlags}}[global options]{{end}}{{if .Commands}} command [command options]{{end}} {{if .ArgsUsage}}{{.ArgsUsage}}{{else}}[arguments...]{{end}}
|
|
|
|
|
{{if len .Authors}}
|
|
|
|
|
AUTHOR:
|
|
|
|
|
{{range .Authors}}{{ . }}{{end}}
|
|
|
|
|
{{end}}{{if .Commands}}
|
|
|
|
|
COMMANDS:
|
|
|
|
|
{{range .Commands}}{{if not .HideHelp}} {{join .Names ", "}}{{ "\t"}}{{.Usage}}{{ "\n" }}{{end}}{{end}}{{end}}{{if .VisibleFlags}}
|
|
|
|
|
GLOBAL OPTIONS:
|
|
|
|
|
{{range .VisibleFlags}}{{.}}
|
|
|
|
|
{{end}}{{end}}{{if .Copyright }}
|
|
|
|
|
COPYRIGHT:
|
|
|
|
|
{{.Copyright}}
|
|
|
|
|
{{end}}{{if .Version}}
|
|
|
|
|
VERSION:
|
|
|
|
|
{{.Version}}
|
|
|
|
|
{{end}}
|
|
|
|
|
`
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
// EXAMPLE: Replace the `HelpPrinter` func
|
|
|
|
|
cli.HelpPrinter = func(w io.Writer, templ string, data interface{}) {
|
|
|
|
|
fmt.Println("Ha HA. I pwnd the help!!1")
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
(&cli.App{}).Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
The default flag may be customized to something other than `-h/--help` by
|
|
|
|
|
setting `cli.HelpFlag`, e.g.:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--halp"],
|
|
|
|
|
"output": "haaaaalp.*HALP"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
cli.HelpFlag = &cli.BoolFlag{
|
|
|
|
|
Name: "haaaaalp",
|
|
|
|
|
Aliases: []string{"halp"},
|
|
|
|
|
Usage: "HALP",
|
|
|
|
|
EnvVars: []string{"SHOW_HALP", "HALPPLZ"},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
(&cli.App{}).Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
### Version Flag
|
|
|
|
|
|
|
|
|
|
The default version flag (`-v/--version`) is defined as `cli.VersionFlag`, which
|
|
|
|
|
is checked by the cli internals in order to print the `App.Version` via
|
|
|
|
|
`cli.VersionPrinter` and break execution.
|
|
|
|
|
|
|
|
|
|
#### Customization
|
|
|
|
|
|
|
|
|
|
The default flag may be customized to something other than `-v/--version` by
|
|
|
|
|
setting `cli.VersionFlag`, e.g.:
|
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--print-version"],
|
|
|
|
|
"output": "partay version v19\\.99\\.0"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
cli.VersionFlag = &cli.BoolFlag{
|
|
|
|
|
Name: "print-version",
|
|
|
|
|
Aliases: []string{"V"},
|
|
|
|
|
Usage: "print only the version",
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
Name: "partay",
|
|
|
|
|
Version: "v19.99.0",
|
|
|
|
|
}
|
|
|
|
|
app.Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Alternatively, the version printer at `cli.VersionPrinter` may be overridden,
|
|
|
|
|
e.g.:
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--version"],
|
|
|
|
|
"output": "version=v19\\.99\\.0 revision=fafafaf"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"os"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Revision = "fafafaf"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
cli.VersionPrinter = func(cCtx *cli.Context) {
|
|
|
|
|
fmt.Printf("version=%s revision=%s\n", cCtx.App.Version, Revision)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
Name: "partay",
|
|
|
|
|
Version: "v19.99.0",
|
|
|
|
|
}
|
|
|
|
|
app.Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2020-01-26 14:19:43 +00:00
|
|
|
|
### Timestamp Flag
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Using the timestamp flag is simple. Please refer to
|
|
|
|
|
[`time.Parse`](https://golang.org/pkg/time/#example_Parse) to get possible
|
|
|
|
|
formats.
|
2020-01-26 14:19:43 +00:00
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"args": ["--meeting", "2019-08-12T15:04:05"],
|
|
|
|
|
"output": "2019\\-08\\-12 15\\:04\\:05 \\+0000 UTC"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2020-01-26 14:19:43 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"fmt"
|
|
|
|
|
"log"
|
|
|
|
|
"os"
|
2020-01-26 14:19:43 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"github.com/urfave/cli/v2"
|
2020-01-26 14:19:43 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.TimestampFlag{Name: "meeting", Layout: "2006-01-02T15:04:05"},
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Printf("%s", cCtx.Timestamp("meeting").String())
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
|
log.Fatal(err)
|
|
|
|
|
}
|
2020-01-26 14:19:43 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
In this example the flag could be used like this:
|
2020-01-26 14:19:43 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```sh-session
|
|
|
|
|
$ myapp --meeting 2019-08-12T15:04:05
|
|
|
|
|
```
|
2020-01-26 14:19:43 +00:00
|
|
|
|
|
2022-07-09 11:43:14 +00:00
|
|
|
|
When the layout doesn't contain timezones, timestamp will render with UTC. To
|
|
|
|
|
change behavior, a default timezone can be provided with flag definition:
|
|
|
|
|
|
|
|
|
|
```go
|
|
|
|
|
app := &cli.App{
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.TimestampFlag{Name: "meeting", Layout: "2006-01-02T15:04:05", Timezone: time.Local},
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
```
|
|
|
|
|
|
|
|
|
|
(time.Local contains the system's local time zone.)
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
Side note: quotes may be necessary around the date depending on your layout (if
|
|
|
|
|
you have spaces for instance)
|
2020-01-26 14:19:43 +00:00
|
|
|
|
|
2019-11-29 13:09:46 +00:00
|
|
|
|
### Suggestions
|
|
|
|
|
|
|
|
|
|
To enable flag and command suggestions, set `app.Suggest = true`. If the suggest
|
|
|
|
|
feature is enabled, then the help output of the corresponding command will
|
|
|
|
|
provide an appropriate suggestion for the provided flag or subcommand if
|
|
|
|
|
available.
|
|
|
|
|
|
2019-10-24 05:15:30 +00:00
|
|
|
|
### Full API Example
|
|
|
|
|
|
|
|
|
|
**Notice**: This is a contrived (functioning) example meant strictly for API
|
2022-06-20 21:08:10 +00:00
|
|
|
|
demonstration purposes. Use of one's imagination is encouraged.
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
|
|
|
|
<!-- {
|
|
|
|
|
"output": "made it!\nPhew!"
|
|
|
|
|
} -->
|
2022-06-20 21:08:10 +00:00
|
|
|
|
```go
|
2019-10-24 05:15:30 +00:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2022-06-20 21:08:10 +00:00
|
|
|
|
"errors"
|
|
|
|
|
"flag"
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
|
|
|
|
"io/ioutil"
|
|
|
|
|
"os"
|
|
|
|
|
"time"
|
|
|
|
|
|
|
|
|
|
"github.com/urfave/cli/v2"
|
2019-10-24 05:15:30 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func init() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
cli.AppHelpTemplate += "\nCUSTOMIZED: you bet ur muffins\n"
|
|
|
|
|
cli.CommandHelpTemplate += "\nYMMV\n"
|
|
|
|
|
cli.SubcommandHelpTemplate += "\nor something\n"
|
|
|
|
|
|
|
|
|
|
cli.HelpFlag = &cli.BoolFlag{Name: "halp"}
|
|
|
|
|
cli.VersionFlag = &cli.BoolFlag{Name: "print-version", Aliases: []string{"V"}}
|
|
|
|
|
|
|
|
|
|
cli.HelpPrinter = func(w io.Writer, templ string, data interface{}) {
|
|
|
|
|
fmt.Fprintf(w, "best of luck to you\n")
|
|
|
|
|
}
|
|
|
|
|
cli.VersionPrinter = func(cCtx *cli.Context) {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "version=%s\n", cCtx.App.Version)
|
|
|
|
|
}
|
|
|
|
|
cli.OsExiter = func(cCtx int) {
|
|
|
|
|
fmt.Fprintf(cli.ErrWriter, "refusing to exit %d\n", cCtx)
|
|
|
|
|
}
|
|
|
|
|
cli.ErrWriter = ioutil.Discard
|
|
|
|
|
cli.FlagStringer = func(fl cli.Flag) string {
|
|
|
|
|
return fmt.Sprintf("\t\t%s", fl.Names()[0])
|
|
|
|
|
}
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type hexWriter struct{}
|
|
|
|
|
|
|
|
|
|
func (w *hexWriter) Write(p []byte) (int, error) {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
for _, b := range p {
|
|
|
|
|
fmt.Printf("%x", b)
|
|
|
|
|
}
|
|
|
|
|
fmt.Printf("\n")
|
2019-10-24 05:15:30 +00:00
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
return len(p), nil
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type genericType struct {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
s string
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (g *genericType) Set(value string) error {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
g.s = value
|
|
|
|
|
return nil
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (g *genericType) String() string {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
return g.s
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func main() {
|
2022-06-20 21:08:10 +00:00
|
|
|
|
app := &cli.App{
|
|
|
|
|
Name: "kənˈtrīv",
|
|
|
|
|
Version: "v19.99.0",
|
|
|
|
|
Compiled: time.Now(),
|
|
|
|
|
Authors: []*cli.Author{
|
|
|
|
|
&cli.Author{
|
|
|
|
|
Name: "Example Human",
|
|
|
|
|
Email: "human@example.com",
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Copyright: "(c) 1999 Serious Enterprise",
|
|
|
|
|
HelpName: "contrive",
|
|
|
|
|
Usage: "demonstrate available API",
|
|
|
|
|
UsageText: "contrive - demonstrating the available API",
|
|
|
|
|
ArgsUsage: "[args and such]",
|
|
|
|
|
Commands: []*cli.Command{
|
|
|
|
|
&cli.Command{
|
|
|
|
|
Name: "doo",
|
|
|
|
|
Aliases: []string{"do"},
|
|
|
|
|
Category: "motion",
|
|
|
|
|
Usage: "do the doo",
|
|
|
|
|
UsageText: "doo - does the dooing",
|
|
|
|
|
Description: "no really, there is a lot of dooing to be done",
|
|
|
|
|
ArgsUsage: "[arrgh]",
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.BoolFlag{Name: "forever", Aliases: []string{"forevvarr"}},
|
|
|
|
|
},
|
|
|
|
|
Subcommands: []*cli.Command{
|
|
|
|
|
&cli.Command{
|
|
|
|
|
Name: "wop",
|
|
|
|
|
Action: wopAction,
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
SkipFlagParsing: false,
|
|
|
|
|
HideHelp: false,
|
|
|
|
|
Hidden: false,
|
|
|
|
|
HelpName: "doo!",
|
|
|
|
|
BashComplete: func(cCtx *cli.Context) {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "--better\n")
|
|
|
|
|
},
|
|
|
|
|
Before: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "brace for impact\n")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
After: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "did we lose anyone?\n")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
cCtx.Command.FullName()
|
|
|
|
|
cCtx.Command.HasName("wop")
|
|
|
|
|
cCtx.Command.Names()
|
|
|
|
|
cCtx.Command.VisibleFlags()
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "dodododododoodododddooooododododooo\n")
|
|
|
|
|
if cCtx.Bool("forever") {
|
|
|
|
|
cCtx.Command.Run(cCtx)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
OnUsageError: func(cCtx *cli.Context, err error, isSubcommand bool) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "for shame\n")
|
|
|
|
|
return err
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
Flags: []cli.Flag{
|
|
|
|
|
&cli.BoolFlag{Name: "fancy"},
|
|
|
|
|
&cli.BoolFlag{Value: true, Name: "fancier"},
|
|
|
|
|
&cli.DurationFlag{Name: "howlong", Aliases: []string{"H"}, Value: time.Second * 3},
|
|
|
|
|
&cli.Float64Flag{Name: "howmuch"},
|
|
|
|
|
&cli.GenericFlag{Name: "wat", Value: &genericType{}},
|
|
|
|
|
&cli.Int64Flag{Name: "longdistance"},
|
|
|
|
|
&cli.Int64SliceFlag{Name: "intervals"},
|
|
|
|
|
&cli.IntFlag{Name: "distance"},
|
|
|
|
|
&cli.IntSliceFlag{Name: "times"},
|
|
|
|
|
&cli.StringFlag{Name: "dance-move", Aliases: []string{"d"}},
|
|
|
|
|
&cli.StringSliceFlag{Name: "names", Aliases: []string{"N"}},
|
|
|
|
|
&cli.UintFlag{Name: "age"},
|
|
|
|
|
&cli.Uint64Flag{Name: "bigage"},
|
|
|
|
|
},
|
|
|
|
|
EnableBashCompletion: true,
|
|
|
|
|
HideHelp: false,
|
|
|
|
|
HideVersion: false,
|
|
|
|
|
BashComplete: func(cCtx *cli.Context) {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "lipstick\nkiss\nme\nlipstick\nringo\n")
|
|
|
|
|
},
|
|
|
|
|
Before: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "HEEEERE GOES\n")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
After: func(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "Phew!\n")
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
CommandNotFound: func(cCtx *cli.Context, command string) {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "Thar be no %q here.\n", command)
|
|
|
|
|
},
|
|
|
|
|
OnUsageError: func(cCtx *cli.Context, err error, isSubcommand bool) error {
|
|
|
|
|
if isSubcommand {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "WRONG: %#v\n", err)
|
|
|
|
|
return nil
|
|
|
|
|
},
|
|
|
|
|
Action: func(cCtx *cli.Context) error {
|
|
|
|
|
cli.DefaultAppComplete(cCtx)
|
|
|
|
|
cli.HandleExitCoder(errors.New("not an exit coder, though"))
|
|
|
|
|
cli.ShowAppHelp(cCtx)
|
|
|
|
|
cli.ShowCommandCompletions(cCtx, "nope")
|
|
|
|
|
cli.ShowCommandHelp(cCtx, "also-nope")
|
|
|
|
|
cli.ShowCompletions(cCtx)
|
|
|
|
|
cli.ShowSubcommandHelp(cCtx)
|
|
|
|
|
cli.ShowVersion(cCtx)
|
|
|
|
|
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.App.Command("doo"))
|
|
|
|
|
if cCtx.Bool("infinite") {
|
|
|
|
|
cCtx.App.Run([]string{"app", "doo", "wop"})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if cCtx.Bool("forevar") {
|
|
|
|
|
cCtx.App.RunAsSubcommand(cCtx)
|
|
|
|
|
}
|
|
|
|
|
cCtx.App.Setup()
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.App.VisibleCategories())
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.App.VisibleCommands())
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.App.VisibleFlags())
|
|
|
|
|
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.Args().First())
|
|
|
|
|
if cCtx.Args().Len() > 0 {
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.Args().Get(1))
|
|
|
|
|
}
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.Args().Present())
|
|
|
|
|
fmt.Printf("%#v\n", cCtx.Args().Tail())
|
|
|
|
|
|
|
|
|
|
set := flag.NewFlagSet("contrive", 0)
|
|
|
|
|
nc := cli.NewContext(cCtx.App, set, cCtx)
|
|
|
|
|
|
|
|
|
|
fmt.Printf("%#v\n", nc.Args())
|
|
|
|
|
fmt.Printf("%#v\n", nc.Bool("nope"))
|
|
|
|
|
fmt.Printf("%#v\n", !nc.Bool("nerp"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Duration("howlong"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Float64("hay"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Generic("bloop"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Int64("bonk"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Int64Slice("burnks"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Int("bips"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.IntSlice("blups"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.String("snurt"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.StringSlice("snurkles"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Uint("flub"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Uint64("florb"))
|
|
|
|
|
|
|
|
|
|
fmt.Printf("%#v\n", nc.FlagNames())
|
|
|
|
|
fmt.Printf("%#v\n", nc.IsSet("wat"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.Set("wat", "nope"))
|
|
|
|
|
fmt.Printf("%#v\n", nc.NArg())
|
|
|
|
|
fmt.Printf("%#v\n", nc.NumFlags())
|
|
|
|
|
fmt.Printf("%#v\n", nc.Lineage()[1])
|
|
|
|
|
nc.Set("wat", "also-nope")
|
|
|
|
|
|
|
|
|
|
ec := cli.Exit("ohwell", 86)
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, "%d", ec.ExitCode())
|
|
|
|
|
fmt.Printf("made it!\n")
|
|
|
|
|
return ec
|
|
|
|
|
},
|
|
|
|
|
Metadata: map[string]interface{}{
|
|
|
|
|
"layers": "many",
|
|
|
|
|
"explicable": false,
|
|
|
|
|
"whatever-values": 19.99,
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if os.Getenv("HEXY") != "" {
|
|
|
|
|
app.Writer = &hexWriter{}
|
|
|
|
|
app.ErrWriter = &hexWriter{}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
app.Run(os.Args)
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
func wopAction(cCtx *cli.Context) error {
|
|
|
|
|
fmt.Fprintf(cCtx.App.Writer, ":wave: over here, eh\n")
|
|
|
|
|
return nil
|
2019-10-24 05:15:30 +00:00
|
|
|
|
}
|
|
|
|
|
```
|
2022-05-08 15:38:41 +00:00
|
|
|
|
|
|
|
|
|
## Migrating From Older Releases
|
|
|
|
|
|
2022-06-20 21:08:10 +00:00
|
|
|
|
There are a small set of breaking changes between v1 and v2. Converting is
|
|
|
|
|
relatively straightforward and typically takes less than an hour. Specific steps
|
|
|
|
|
are included in [Migration Guide: v1 to v2](../migrate-v1-to-v2.md). Also see
|
|
|
|
|
the [pkg.go.dev docs](https://pkg.go.dev/github.com/urfave/cli/v2) for v2 API
|
|
|
|
|
documentation.
|