361 lines
9.7 KiB
Go
361 lines
9.7 KiB
Go
package arg
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
)
|
|
|
|
// the width of the left column
|
|
const colWidth = 25
|
|
|
|
// Fail prints usage information to p.Config.Out and exits with status code 2.
|
|
func (p *Parser) Fail(msg string) {
|
|
p.FailSubcommand(msg)
|
|
}
|
|
|
|
// FailSubcommand prints usage information for a specified subcommand to p.Config.Out,
|
|
// then exits with status code 2. To write usage information for a top-level
|
|
// subcommand, provide just the name of that subcommand. To write usage
|
|
// information for a subcommand that is nested under another subcommand, provide
|
|
// a sequence of subcommand names starting with the top-level subcommand and so
|
|
// on down the tree.
|
|
func (p *Parser) FailSubcommand(msg string, subcommand ...string) error {
|
|
err := p.WriteUsageForSubcommand(p.config.Out, subcommand...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fmt.Fprintln(p.config.Out, "error:", msg)
|
|
p.config.Exit(2)
|
|
return nil
|
|
}
|
|
|
|
// WriteUsage writes usage information to the given writer
|
|
func (p *Parser) WriteUsage(w io.Writer) {
|
|
p.WriteUsageForSubcommand(w, p.subcommand...)
|
|
}
|
|
|
|
// WriteUsageForSubcommand writes the usage information for a specified
|
|
// subcommand. To write usage information for a top-level subcommand, provide
|
|
// just the name of that subcommand. To write usage information for a subcommand
|
|
// that is nested under another subcommand, provide a sequence of subcommand
|
|
// names starting with the top-level subcommand and so on down the tree.
|
|
func (p *Parser) WriteUsageForSubcommand(w io.Writer, subcommand ...string) error {
|
|
cmd, err := p.lookupCommand(subcommand...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var positionals, longOptions, shortOptions []*spec
|
|
var hasVersionOption bool
|
|
for _, spec := range cmd.specs {
|
|
switch {
|
|
case spec.positional:
|
|
positionals = append(positionals, spec)
|
|
case spec.long != "":
|
|
longOptions = append(longOptions, spec)
|
|
if spec.long == "version" {
|
|
hasVersionOption = true
|
|
}
|
|
case spec.short != "":
|
|
shortOptions = append(shortOptions, spec)
|
|
}
|
|
}
|
|
|
|
// make a list of ancestor commands so that we print with full context
|
|
// also determine if any ancestor has a version option spec
|
|
var ancestors []string
|
|
ancestor := cmd
|
|
for ancestor != nil {
|
|
for _, spec := range ancestor.specs {
|
|
if spec.long == "version" {
|
|
hasVersionOption = true
|
|
}
|
|
}
|
|
ancestors = append(ancestors, ancestor.name)
|
|
ancestor = ancestor.parent
|
|
}
|
|
|
|
if !hasVersionOption && p.version != "" {
|
|
fmt.Fprintln(w, p.version)
|
|
}
|
|
|
|
// print the beginning of the usage string
|
|
fmt.Fprintf(w, "Usage: %s", p.cmd.name)
|
|
for _, s := range subcommand {
|
|
fmt.Fprint(w, " "+s)
|
|
}
|
|
|
|
// write the option component of the usage message
|
|
for _, spec := range shortOptions {
|
|
// prefix with a space
|
|
fmt.Fprint(w, " ")
|
|
if !spec.required {
|
|
fmt.Fprint(w, "[")
|
|
}
|
|
fmt.Fprint(w, synopsis(spec, "-"+spec.short))
|
|
if !spec.required {
|
|
fmt.Fprint(w, "]")
|
|
}
|
|
}
|
|
|
|
for _, spec := range longOptions {
|
|
// prefix with a space
|
|
fmt.Fprint(w, " ")
|
|
if !spec.required {
|
|
fmt.Fprint(w, "[")
|
|
}
|
|
fmt.Fprint(w, synopsis(spec, "--"+spec.long))
|
|
if !spec.required {
|
|
fmt.Fprint(w, "]")
|
|
}
|
|
}
|
|
|
|
// When we parse positionals, we check that:
|
|
// 1. required positionals come before non-required positionals
|
|
// 2. there is at most one multiple-value positional
|
|
// 3. if there is a multiple-value positional then it comes after all other positionals
|
|
// Here we merely print the usage string, so we do not explicitly re-enforce those rules
|
|
|
|
// write the positionals in following form:
|
|
// REQUIRED1 REQUIRED2
|
|
// REQUIRED1 REQUIRED2 [OPTIONAL1 [OPTIONAL2]]
|
|
// REQUIRED1 REQUIRED2 REPEATED [REPEATED ...]
|
|
// REQUIRED1 REQUIRED2 [REPEATEDOPTIONAL [REPEATEDOPTIONAL ...]]
|
|
// REQUIRED1 REQUIRED2 [OPTIONAL1 [REPEATEDOPTIONAL [REPEATEDOPTIONAL ...]]]
|
|
var closeBrackets int
|
|
for _, spec := range positionals {
|
|
fmt.Fprint(w, " ")
|
|
if !spec.required {
|
|
fmt.Fprint(w, "[")
|
|
closeBrackets += 1
|
|
}
|
|
if spec.cardinality == multiple {
|
|
fmt.Fprintf(w, "%s [%s ...]", spec.placeholder, spec.placeholder)
|
|
} else {
|
|
fmt.Fprint(w, spec.placeholder)
|
|
}
|
|
}
|
|
fmt.Fprint(w, strings.Repeat("]", closeBrackets))
|
|
|
|
// if the program supports subcommands, give a hint to the user about their existence
|
|
if len(cmd.subcommands) > 0 {
|
|
fmt.Fprint(w, " <command> [<args>]")
|
|
}
|
|
|
|
fmt.Fprint(w, "\n")
|
|
return nil
|
|
}
|
|
|
|
// print prints a line like this:
|
|
//
|
|
// --option FOO A description of the option [default: 123]
|
|
//
|
|
// If the text on the left is longer than a certain threshold, the description is moved to the next line:
|
|
//
|
|
// --verylongoptionoption VERY_LONG_VARIABLE
|
|
// A description of the option [default: 123]
|
|
//
|
|
// If multiple "extras" are provided then they are put inside a single set of square brackets:
|
|
//
|
|
// --option FOO A description of the option [default: 123, env: FOO]
|
|
func print(w io.Writer, item, description string, bracketed ...string) {
|
|
lhs := " " + item
|
|
fmt.Fprint(w, lhs)
|
|
if description != "" {
|
|
if len(lhs)+2 < colWidth {
|
|
fmt.Fprint(w, strings.Repeat(" ", colWidth-len(lhs)))
|
|
} else {
|
|
fmt.Fprint(w, "\n"+strings.Repeat(" ", colWidth))
|
|
}
|
|
fmt.Fprint(w, description)
|
|
}
|
|
|
|
var brack string
|
|
for _, s := range bracketed {
|
|
if s != "" {
|
|
if brack != "" {
|
|
brack += ", "
|
|
}
|
|
brack += s
|
|
}
|
|
}
|
|
|
|
if brack != "" {
|
|
fmt.Fprintf(w, " [%s]", brack)
|
|
}
|
|
fmt.Fprint(w, "\n")
|
|
}
|
|
|
|
func withDefault(s string) string {
|
|
if s == "" {
|
|
return ""
|
|
}
|
|
return "default: " + s
|
|
}
|
|
|
|
func withEnv(env string) string {
|
|
if env == "" {
|
|
return ""
|
|
}
|
|
return "env: " + env
|
|
}
|
|
|
|
// WriteHelp writes the usage string followed by the full help string for each option
|
|
func (p *Parser) WriteHelp(w io.Writer) {
|
|
p.WriteHelpForSubcommand(w, p.subcommand...)
|
|
}
|
|
|
|
// WriteHelpForSubcommand writes the usage string followed by the full help
|
|
// string for a specified subcommand. To write help for a top-level subcommand,
|
|
// provide just the name of that subcommand. To write help for a subcommand that
|
|
// is nested under another subcommand, provide a sequence of subcommand names
|
|
// starting with the top-level subcommand and so on down the tree.
|
|
func (p *Parser) WriteHelpForSubcommand(w io.Writer, subcommand ...string) error {
|
|
cmd, err := p.lookupCommand(subcommand...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
var positionals, longOptions, shortOptions, envOnlyOptions []*spec
|
|
var hasVersionOption bool
|
|
for _, spec := range cmd.specs {
|
|
switch {
|
|
case spec.positional:
|
|
positionals = append(positionals, spec)
|
|
case spec.long != "":
|
|
longOptions = append(longOptions, spec)
|
|
if spec.long == "version" {
|
|
hasVersionOption = true
|
|
}
|
|
case spec.short != "":
|
|
shortOptions = append(shortOptions, spec)
|
|
case spec.short == "" && spec.long == "":
|
|
envOnlyOptions = append(envOnlyOptions, spec)
|
|
}
|
|
}
|
|
|
|
// obtain a flattened list of options from all ancestors
|
|
// also determine if any ancestor has a version option spec
|
|
var globals []*spec
|
|
ancestor := cmd.parent
|
|
for ancestor != nil {
|
|
for _, spec := range ancestor.specs {
|
|
if spec.long == "version" {
|
|
hasVersionOption = true
|
|
break
|
|
}
|
|
}
|
|
globals = append(globals, ancestor.specs...)
|
|
ancestor = ancestor.parent
|
|
}
|
|
|
|
if p.description != "" {
|
|
fmt.Fprintln(w, p.description)
|
|
}
|
|
p.WriteUsageForSubcommand(w, subcommand...)
|
|
|
|
// write the list of positionals
|
|
if len(positionals) > 0 {
|
|
fmt.Fprint(w, "\nPositional arguments:\n")
|
|
for _, spec := range positionals {
|
|
print(w, spec.placeholder, spec.help)
|
|
}
|
|
}
|
|
|
|
// write the list of options with the short-only ones first to match the usage string
|
|
if len(shortOptions)+len(longOptions) > 0 || cmd.parent == nil {
|
|
fmt.Fprint(w, "\nOptions:\n")
|
|
for _, spec := range shortOptions {
|
|
p.printOption(w, spec)
|
|
}
|
|
for _, spec := range longOptions {
|
|
p.printOption(w, spec)
|
|
}
|
|
}
|
|
|
|
// write the list of global options
|
|
if len(globals) > 0 {
|
|
fmt.Fprint(w, "\nGlobal options:\n")
|
|
for _, spec := range globals {
|
|
p.printOption(w, spec)
|
|
}
|
|
}
|
|
|
|
// write the list of built in options
|
|
p.printOption(w, &spec{
|
|
cardinality: zero,
|
|
long: "help",
|
|
short: "h",
|
|
help: "display this help and exit",
|
|
})
|
|
if !hasVersionOption && p.version != "" {
|
|
p.printOption(w, &spec{
|
|
cardinality: zero,
|
|
long: "version",
|
|
help: "display version and exit",
|
|
})
|
|
}
|
|
|
|
// write the list of environment only variables
|
|
if len(envOnlyOptions) > 0 {
|
|
fmt.Fprint(w, "\nEnvironment variables:\n")
|
|
for _, spec := range envOnlyOptions {
|
|
p.printEnvOnlyVar(w, spec)
|
|
}
|
|
}
|
|
|
|
// write the list of subcommands
|
|
if len(cmd.subcommands) > 0 {
|
|
fmt.Fprint(w, "\nCommands:\n")
|
|
for _, subcmd := range cmd.subcommands {
|
|
names := append([]string{subcmd.name}, subcmd.aliases...)
|
|
print(w, strings.Join(names, ", "), subcmd.help)
|
|
}
|
|
}
|
|
|
|
if p.epilogue != "" {
|
|
fmt.Fprintln(w, "\n"+p.epilogue)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (p *Parser) printOption(w io.Writer, spec *spec) {
|
|
ways := make([]string, 0, 2)
|
|
if spec.long != "" {
|
|
ways = append(ways, synopsis(spec, "--"+spec.long))
|
|
}
|
|
if spec.short != "" {
|
|
ways = append(ways, synopsis(spec, "-"+spec.short))
|
|
}
|
|
if len(ways) > 0 {
|
|
print(w, strings.Join(ways, ", "), spec.help, withDefault(spec.defaultString), withEnv(spec.env))
|
|
}
|
|
}
|
|
|
|
func (p *Parser) printEnvOnlyVar(w io.Writer, spec *spec) {
|
|
ways := make([]string, 0, 2)
|
|
if spec.required {
|
|
ways = append(ways, "Required.")
|
|
} else {
|
|
ways = append(ways, "Optional.")
|
|
}
|
|
|
|
if spec.help != "" {
|
|
ways = append(ways, spec.help)
|
|
}
|
|
|
|
print(w, spec.env, strings.Join(ways, " "), withDefault(spec.defaultString))
|
|
}
|
|
|
|
func synopsis(spec *spec, form string) string {
|
|
// if the user omits the placeholder tag then we pick one automatically,
|
|
// but if the user explicitly specifies an empty placeholder then we
|
|
// leave out the placeholder in the help message
|
|
if spec.cardinality == zero || spec.placeholder == "" {
|
|
return form
|
|
}
|
|
return form + " " + spec.placeholder
|
|
}
|