arg/usage.go

133 lines
3.1 KiB
Go
Raw Normal View History

2015-10-31 20:32:20 -05:00
package arg
import (
"fmt"
"io"
"os"
2015-11-21 17:58:27 -06:00
"reflect"
2015-10-31 20:32:20 -05:00
"strings"
)
2016-01-18 10:24:21 -06:00
// the width of the left column
const colWidth = 25
// Fail prints usage information to stderr and exits with non-zero status
func (p *Parser) Fail(msg string) {
p.WriteUsage(os.Stderr)
fmt.Fprintln(os.Stderr, "error:", msg)
os.Exit(-1)
2015-10-31 20:32:20 -05:00
}
// WriteUsage writes usage information to the given writer
func (p *Parser) WriteUsage(w io.Writer) {
2015-10-31 20:32:20 -05:00
var positionals, options []*spec
for _, spec := range p.spec {
2015-10-31 20:32:20 -05:00
if spec.positional {
positionals = append(positionals, spec)
} else {
options = append(options, spec)
}
}
fmt.Fprintf(w, "usage: %s", p.config.Program)
2015-10-31 20:32:20 -05:00
// write the option component of the usage message
2015-10-31 20:32:20 -05:00
for _, spec := range options {
// prefix with a space
fmt.Fprint(w, " ")
2015-10-31 20:32:20 -05:00
if !spec.required {
fmt.Fprint(w, "[")
}
fmt.Fprint(w, synopsis(spec, "--"+spec.long))
if !spec.required {
fmt.Fprint(w, "]")
}
}
// write the positional component of the usage message
2015-10-31 20:32:20 -05:00
for _, spec := range positionals {
// prefix with a space
fmt.Fprint(w, " ")
2015-10-31 20:32:20 -05:00
up := strings.ToUpper(spec.long)
if spec.multiple {
2015-10-31 20:48:38 -05:00
fmt.Fprintf(w, "[%s [%s ...]]", up, up)
2015-10-31 20:32:20 -05:00
} else {
fmt.Fprint(w, up)
}
}
fmt.Fprint(w, "\n")
}
// WriteHelp writes the usage string followed by the full help string for each option
func (p *Parser) WriteHelp(w io.Writer) {
var positionals, options []*spec
for _, spec := range p.spec {
if spec.positional {
positionals = append(positionals, spec)
} else {
options = append(options, spec)
}
}
p.WriteUsage(w)
2015-10-31 20:32:20 -05:00
// write the list of positionals
if len(positionals) > 0 {
fmt.Fprint(w, "\npositional arguments:\n")
for _, spec := range positionals {
left := " " + spec.long
fmt.Fprint(w, left)
if spec.help != "" {
if len(left)+2 < colWidth {
fmt.Fprint(w, strings.Repeat(" ", colWidth-len(left)))
} else {
fmt.Fprint(w, "\n"+strings.Repeat(" ", colWidth))
}
fmt.Fprint(w, spec.help)
}
fmt.Fprint(w, "\n")
2015-10-31 20:32:20 -05:00
}
}
2015-11-21 17:58:27 -06:00
// write the list of options
fmt.Fprint(w, "\noptions:\n")
for _, spec := range options {
printOption(w, spec)
}
// write the list of built in options
printOption(w, &spec{boolean: true, long: "help", short: "h", help: "display this help and exit"})
}
func printOption(w io.Writer, spec *spec) {
left := " " + synopsis(spec, "--"+spec.long)
if spec.short != "" {
left += ", " + synopsis(spec, "-"+spec.short)
}
fmt.Fprint(w, left)
if spec.help != "" {
if len(left)+2 < colWidth {
fmt.Fprint(w, strings.Repeat(" ", colWidth-len(left)))
} else {
fmt.Fprint(w, "\n"+strings.Repeat(" ", colWidth))
2015-10-31 20:32:20 -05:00
}
fmt.Fprint(w, spec.help)
2015-10-31 20:32:20 -05:00
}
// If spec.dest is not the zero value then a default value has been added.
v := spec.dest
2015-11-21 17:58:27 -06:00
if v.IsValid() {
z := reflect.Zero(v.Type())
if (v.Type().Comparable() && z.Type().Comparable() && v.Interface() != z.Interface()) || v.Kind() == reflect.Slice && !v.IsNil() {
2015-11-21 17:58:27 -06:00
fmt.Fprintf(w, " [default: %v]", v)
}
}
fmt.Fprint(w, "\n")
2015-10-31 20:32:20 -05:00
}
2015-11-01 01:57:26 -05:00
func synopsis(spec *spec, form string) string {
if spec.boolean {
2015-11-01 01:57:26 -05:00
return form
}
2015-11-11 07:08:28 -06:00
return form + " " + strings.ToUpper(spec.long)
2015-11-01 01:57:26 -05:00
}