279 lines
6.2 KiB
Go
279 lines
6.2 KiB
Go
package kong
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"go/doc"
|
|
"io"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
defaultIndent = 2
|
|
defaultColumnPadding = 4
|
|
)
|
|
|
|
// Help flag.
|
|
type helpValue bool
|
|
|
|
func (h helpValue) BeforeApply(ctx *Context) error {
|
|
options := ctx.Kong.helpOptions
|
|
options.Summary = false
|
|
err := ctx.Kong.help(options, ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ctx.Kong.Exit(1)
|
|
return nil
|
|
}
|
|
|
|
// HelpOptions for HelpPrinters.
|
|
type HelpOptions struct {
|
|
// Don't print top-level usage summary.
|
|
NoAppSummary bool
|
|
|
|
// Write a one-line summary of the context.
|
|
Summary bool
|
|
|
|
// Write help in a more compact, but still fully-specified, form.
|
|
Compact bool
|
|
}
|
|
|
|
// HelpProvider can be implemented by commands/args to provide detailed help.
|
|
type HelpProvider interface {
|
|
// This string is formatted by go/doc and thus has the same formatting rules.
|
|
Help() string
|
|
}
|
|
|
|
// HelpPrinter is used to print context-sensitive help.
|
|
type HelpPrinter func(options HelpOptions, ctx *Context) error
|
|
|
|
// DefaultHelpPrinter is the default HelpPrinter.
|
|
func DefaultHelpPrinter(options HelpOptions, ctx *Context) error {
|
|
if ctx.Empty() {
|
|
options.Summary = false
|
|
}
|
|
w := newHelpWriter(ctx, options)
|
|
selected := ctx.Selected()
|
|
if selected == nil {
|
|
printApp(w, ctx.Model)
|
|
} else {
|
|
printCommand(w, ctx.Model, selected)
|
|
}
|
|
return w.Write(ctx.Stdout)
|
|
}
|
|
|
|
func printApp(w *helpWriter, app *Application) {
|
|
if !w.NoAppSummary {
|
|
w.Printf("Usage: %s%s", app.Name, app.Summary())
|
|
}
|
|
printNodeDetail(w, app.Node)
|
|
cmds := app.Leaves(true)
|
|
if len(cmds) > 0 && app.HelpFlag != nil {
|
|
w.Print("")
|
|
if w.Summary {
|
|
w.Printf(`Run "%s --help" for more information.`, app.Name)
|
|
} else {
|
|
w.Printf(`Run "%s <command> --help" for more information on a command.`, app.Name)
|
|
}
|
|
}
|
|
}
|
|
|
|
func printCommand(w *helpWriter, app *Application, cmd *Command) {
|
|
if !w.NoAppSummary {
|
|
w.Printf("Usage: %s %s", app.Name, cmd.Summary())
|
|
}
|
|
printNodeDetail(w, cmd)
|
|
if w.Summary && app.HelpFlag != nil {
|
|
w.Print("")
|
|
w.Printf(`Run "%s --help" for more information.`, cmd.FullPath())
|
|
}
|
|
}
|
|
|
|
func printNodeDetail(w *helpWriter, node *Node) {
|
|
if node.Help != "" {
|
|
w.Print("")
|
|
w.Wrap(node.Help)
|
|
}
|
|
if w.Summary {
|
|
return
|
|
}
|
|
if node.Detail != "" {
|
|
w.Print("")
|
|
w.Wrap(node.Detail)
|
|
}
|
|
if len(node.Positional) > 0 {
|
|
w.Print("")
|
|
w.Print("Arguments:")
|
|
writePositionals(w.Indent(), node.Positional)
|
|
}
|
|
if flags := node.AllFlags(true); len(flags) > 0 {
|
|
w.Print("")
|
|
w.Print("Flags:")
|
|
writeFlags(w.Indent(), flags)
|
|
}
|
|
cmds := node.Leaves(true)
|
|
if len(cmds) > 0 {
|
|
w.Print("")
|
|
w.Print("Commands:")
|
|
iw := w.Indent()
|
|
if w.Compact {
|
|
rows := [][2]string{}
|
|
for _, cmd := range cmds {
|
|
rows = append(rows, [2]string{cmd.Path(), cmd.Help})
|
|
}
|
|
writeTwoColumns(iw, defaultColumnPadding, rows)
|
|
} else {
|
|
for i, cmd := range cmds {
|
|
printCommandSummary(iw, cmd)
|
|
if i != len(cmds)-1 {
|
|
iw.Print("")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func printCommandSummary(w *helpWriter, cmd *Command) {
|
|
w.Print(cmd.Summary())
|
|
if cmd.Help != "" {
|
|
w.Indent().Wrap(cmd.Help)
|
|
}
|
|
}
|
|
|
|
type helpWriter struct {
|
|
indent string
|
|
width int
|
|
lines *[]string
|
|
HelpOptions
|
|
}
|
|
|
|
func newHelpWriter(ctx *Context, options HelpOptions) *helpWriter {
|
|
lines := []string{}
|
|
w := &helpWriter{
|
|
indent: "",
|
|
width: guessWidth(ctx.Stdout),
|
|
lines: &lines,
|
|
HelpOptions: options,
|
|
}
|
|
return w
|
|
}
|
|
|
|
func (h *helpWriter) Printf(format string, args ...interface{}) {
|
|
h.Print(fmt.Sprintf(format, args...))
|
|
}
|
|
|
|
func (h *helpWriter) Print(text string) {
|
|
*h.lines = append(*h.lines, strings.TrimRight(h.indent+text, " "))
|
|
}
|
|
|
|
func (h *helpWriter) Indent() *helpWriter {
|
|
return &helpWriter{indent: h.indent + " ", lines: h.lines, width: h.width - 2, HelpOptions: h.HelpOptions}
|
|
}
|
|
|
|
func (h *helpWriter) String() string {
|
|
return strings.Join(*h.lines, "\n")
|
|
}
|
|
|
|
func (h *helpWriter) Write(w io.Writer) error {
|
|
for _, line := range *h.lines {
|
|
_, err := io.WriteString(w, line+"\n")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (h *helpWriter) Wrap(text string) {
|
|
w := bytes.NewBuffer(nil)
|
|
doc.ToText(w, strings.TrimSpace(text), "", " ", h.width)
|
|
for _, line := range strings.Split(strings.TrimSpace(w.String()), "\n") {
|
|
h.Print(line)
|
|
}
|
|
}
|
|
|
|
func writePositionals(w *helpWriter, args []*Positional) {
|
|
rows := [][2]string{}
|
|
for _, arg := range args {
|
|
rows = append(rows, [2]string{arg.Summary(), arg.Help})
|
|
}
|
|
writeTwoColumns(w, defaultColumnPadding, rows)
|
|
}
|
|
|
|
func writeFlags(w *helpWriter, groups [][]*Flag) {
|
|
rows := [][2]string{}
|
|
haveShort := false
|
|
for _, group := range groups {
|
|
for _, flag := range group {
|
|
if flag.Short != 0 {
|
|
haveShort = true
|
|
break
|
|
}
|
|
}
|
|
}
|
|
for i, group := range groups {
|
|
if i > 0 {
|
|
rows = append(rows, [2]string{"", ""})
|
|
}
|
|
for _, flag := range group {
|
|
if !flag.Hidden {
|
|
rows = append(rows, [2]string{formatFlag(haveShort, flag), flag.Help})
|
|
}
|
|
}
|
|
}
|
|
writeTwoColumns(w, defaultColumnPadding, rows)
|
|
}
|
|
|
|
func writeTwoColumns(w *helpWriter, padding int, rows [][2]string) {
|
|
maxLeft := 375 * w.width / 1000
|
|
if maxLeft < 30 {
|
|
maxLeft = 30
|
|
}
|
|
// Find size of first column.
|
|
leftSize := 0
|
|
for _, row := range rows {
|
|
if c := len(row[0]); c > leftSize && c < maxLeft {
|
|
leftSize = c
|
|
}
|
|
}
|
|
|
|
offsetStr := strings.Repeat(" ", leftSize+padding)
|
|
|
|
for _, row := range rows {
|
|
buf := bytes.NewBuffer(nil)
|
|
doc.ToText(buf, row[1], "", strings.Repeat(" ", defaultIndent), w.width-leftSize-padding)
|
|
lines := strings.Split(strings.TrimRight(buf.String(), "\n"), "\n")
|
|
|
|
line := fmt.Sprintf("%-*s", leftSize, row[0])
|
|
if len(row[0]) < maxLeft {
|
|
line += fmt.Sprintf("%*s%s", padding, "", lines[0])
|
|
lines = lines[1:]
|
|
}
|
|
w.Print(line)
|
|
for _, line := range lines {
|
|
w.Printf("%s%s", offsetStr, line)
|
|
}
|
|
}
|
|
}
|
|
|
|
// haveShort will be true if there are short flags present at all in the help. Useful for column alignment.
|
|
func formatFlag(haveShort bool, flag *Flag) string {
|
|
flagString := ""
|
|
name := flag.Name
|
|
isBool := flag.IsBool()
|
|
if flag.Short != 0 {
|
|
flagString += fmt.Sprintf("-%c, --%s", flag.Short, name)
|
|
} else {
|
|
if haveShort {
|
|
flagString += fmt.Sprintf(" --%s", name)
|
|
} else {
|
|
flagString += fmt.Sprintf("--%s", name)
|
|
}
|
|
}
|
|
if !isBool {
|
|
flagString += fmt.Sprintf("=%s", flag.FormatPlaceHolder())
|
|
}
|
|
return flagString
|
|
}
|