mirror of
https://github.com/VictoriaMetrics/VictoriaMetrics.git
synced 2025-01-20 15:16:42 +00:00
478 lines
11 KiB
Go
478 lines
11 KiB
Go
package cli
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"strings"
|
|
"text/tabwriter"
|
|
"text/template"
|
|
"unicode/utf8"
|
|
)
|
|
|
|
const (
|
|
helpName = "help"
|
|
helpAlias = "h"
|
|
)
|
|
|
|
var helpCommand = &Command{
|
|
Name: helpName,
|
|
Aliases: []string{helpAlias},
|
|
Usage: "Shows a list of commands or help for one command",
|
|
ArgsUsage: "[command]",
|
|
Action: func(cCtx *Context) error {
|
|
args := cCtx.Args()
|
|
if args.Present() {
|
|
return ShowCommandHelp(cCtx, args.First())
|
|
}
|
|
|
|
_ = ShowAppHelp(cCtx)
|
|
return nil
|
|
},
|
|
}
|
|
|
|
var helpSubcommand = &Command{
|
|
Name: helpName,
|
|
Aliases: []string{helpAlias},
|
|
Usage: "Shows a list of commands or help for one command",
|
|
ArgsUsage: "[command]",
|
|
Action: func(cCtx *Context) error {
|
|
args := cCtx.Args()
|
|
if args.Present() {
|
|
return ShowCommandHelp(cCtx, args.First())
|
|
}
|
|
|
|
return ShowSubcommandHelp(cCtx)
|
|
},
|
|
}
|
|
|
|
// Prints help for the App or Command
|
|
type helpPrinter func(w io.Writer, templ string, data interface{})
|
|
|
|
// Prints help for the App or Command with custom template function.
|
|
type helpPrinterCustom func(w io.Writer, templ string, data interface{}, customFunc map[string]interface{})
|
|
|
|
// HelpPrinter is a function that writes the help output. If not set explicitly,
|
|
// this calls HelpPrinterCustom using only the default template functions.
|
|
//
|
|
// If custom logic for printing help is required, this function can be
|
|
// overridden. If the ExtraInfo field is defined on an App, this function
|
|
// should not be modified, as HelpPrinterCustom will be used directly in order
|
|
// to capture the extra information.
|
|
var HelpPrinter helpPrinter = printHelp
|
|
|
|
// HelpPrinterCustom is a function that writes the help output. It is used as
|
|
// the default implementation of HelpPrinter, and may be called directly if
|
|
// the ExtraInfo field is set on an App.
|
|
//
|
|
// In the default implementation, if the customFuncs argument contains a
|
|
// "wrapAt" key, which is a function which takes no arguments and returns
|
|
// an int, this int value will be used to produce a "wrap" function used
|
|
// by the default template to wrap long lines.
|
|
var HelpPrinterCustom helpPrinterCustom = printHelpCustom
|
|
|
|
// VersionPrinter prints the version for the App
|
|
var VersionPrinter = printVersion
|
|
|
|
// ShowAppHelpAndExit - Prints the list of subcommands for the app and exits with exit code.
|
|
func ShowAppHelpAndExit(c *Context, exitCode int) {
|
|
_ = ShowAppHelp(c)
|
|
os.Exit(exitCode)
|
|
}
|
|
|
|
// ShowAppHelp is an action that displays the help.
|
|
func ShowAppHelp(cCtx *Context) error {
|
|
tpl := cCtx.App.CustomAppHelpTemplate
|
|
if tpl == "" {
|
|
tpl = AppHelpTemplate
|
|
}
|
|
|
|
if cCtx.App.ExtraInfo == nil {
|
|
HelpPrinter(cCtx.App.Writer, tpl, cCtx.App)
|
|
return nil
|
|
}
|
|
|
|
customAppData := func() map[string]interface{} {
|
|
return map[string]interface{}{
|
|
"ExtraInfo": cCtx.App.ExtraInfo,
|
|
}
|
|
}
|
|
HelpPrinterCustom(cCtx.App.Writer, tpl, cCtx.App, customAppData())
|
|
|
|
return nil
|
|
}
|
|
|
|
// DefaultAppComplete prints the list of subcommands as the default app completion method
|
|
func DefaultAppComplete(cCtx *Context) {
|
|
DefaultCompleteWithFlags(nil)(cCtx)
|
|
}
|
|
|
|
func printCommandSuggestions(commands []*Command, writer io.Writer) {
|
|
for _, command := range commands {
|
|
if command.Hidden {
|
|
continue
|
|
}
|
|
if strings.HasSuffix(os.Getenv("SHELL"), "zsh") {
|
|
for _, name := range command.Names() {
|
|
_, _ = fmt.Fprintf(writer, "%s:%s\n", name, command.Usage)
|
|
}
|
|
} else {
|
|
for _, name := range command.Names() {
|
|
_, _ = fmt.Fprintf(writer, "%s\n", name)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func cliArgContains(flagName string) bool {
|
|
for _, name := range strings.Split(flagName, ",") {
|
|
name = strings.TrimSpace(name)
|
|
count := utf8.RuneCountInString(name)
|
|
if count > 2 {
|
|
count = 2
|
|
}
|
|
flag := fmt.Sprintf("%s%s", strings.Repeat("-", count), name)
|
|
for _, a := range os.Args {
|
|
if a == flag {
|
|
return true
|
|
}
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func printFlagSuggestions(lastArg string, flags []Flag, writer io.Writer) {
|
|
cur := strings.TrimPrefix(lastArg, "-")
|
|
cur = strings.TrimPrefix(cur, "-")
|
|
for _, flag := range flags {
|
|
if bflag, ok := flag.(*BoolFlag); ok && bflag.Hidden {
|
|
continue
|
|
}
|
|
for _, name := range flag.Names() {
|
|
name = strings.TrimSpace(name)
|
|
// this will get total count utf8 letters in flag name
|
|
count := utf8.RuneCountInString(name)
|
|
if count > 2 {
|
|
count = 2 // resuse this count to generate single - or -- in flag completion
|
|
}
|
|
// if flag name has more than one utf8 letter and last argument in cli has -- prefix then
|
|
// skip flag completion for short flags example -v or -x
|
|
if strings.HasPrefix(lastArg, "--") && count == 1 {
|
|
continue
|
|
}
|
|
// match if last argument matches this flag and it is not repeated
|
|
if strings.HasPrefix(name, cur) && cur != name && !cliArgContains(name) {
|
|
flagCompletion := fmt.Sprintf("%s%s", strings.Repeat("-", count), name)
|
|
_, _ = fmt.Fprintln(writer, flagCompletion)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func DefaultCompleteWithFlags(cmd *Command) func(cCtx *Context) {
|
|
return func(cCtx *Context) {
|
|
if len(os.Args) > 2 {
|
|
lastArg := os.Args[len(os.Args)-2]
|
|
|
|
if strings.HasPrefix(lastArg, "-") {
|
|
if cmd != nil {
|
|
printFlagSuggestions(lastArg, cmd.Flags, cCtx.App.Writer)
|
|
|
|
return
|
|
}
|
|
|
|
printFlagSuggestions(lastArg, cCtx.App.Flags, cCtx.App.Writer)
|
|
|
|
return
|
|
}
|
|
}
|
|
|
|
if cmd != nil {
|
|
printCommandSuggestions(cmd.Subcommands, cCtx.App.Writer)
|
|
return
|
|
}
|
|
|
|
printCommandSuggestions(cCtx.App.Commands, cCtx.App.Writer)
|
|
}
|
|
}
|
|
|
|
// ShowCommandHelpAndExit - exits with code after showing help
|
|
func ShowCommandHelpAndExit(c *Context, command string, code int) {
|
|
_ = ShowCommandHelp(c, command)
|
|
os.Exit(code)
|
|
}
|
|
|
|
// ShowCommandHelp prints help for the given command
|
|
func ShowCommandHelp(ctx *Context, command string) error {
|
|
// show the subcommand help for a command with subcommands
|
|
if command == "" {
|
|
HelpPrinter(ctx.App.Writer, SubcommandHelpTemplate, ctx.App)
|
|
return nil
|
|
}
|
|
|
|
for _, c := range ctx.App.Commands {
|
|
if c.HasName(command) {
|
|
templ := c.CustomHelpTemplate
|
|
if templ == "" {
|
|
templ = CommandHelpTemplate
|
|
}
|
|
|
|
HelpPrinter(ctx.App.Writer, templ, c)
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
if ctx.App.CommandNotFound == nil {
|
|
errMsg := fmt.Sprintf("No help topic for '%v'", command)
|
|
if ctx.App.Suggest {
|
|
if suggestion := SuggestCommand(ctx.App.Commands, command); suggestion != "" {
|
|
errMsg += ". " + suggestion
|
|
}
|
|
}
|
|
return Exit(errMsg, 3)
|
|
}
|
|
|
|
ctx.App.CommandNotFound(ctx, command)
|
|
return nil
|
|
}
|
|
|
|
// ShowSubcommandHelpAndExit - Prints help for the given subcommand and exits with exit code.
|
|
func ShowSubcommandHelpAndExit(c *Context, exitCode int) {
|
|
_ = ShowSubcommandHelp(c)
|
|
os.Exit(exitCode)
|
|
}
|
|
|
|
// ShowSubcommandHelp prints help for the given subcommand
|
|
func ShowSubcommandHelp(cCtx *Context) error {
|
|
if cCtx == nil {
|
|
return nil
|
|
}
|
|
|
|
if cCtx.Command != nil {
|
|
return ShowCommandHelp(cCtx, cCtx.Command.Name)
|
|
}
|
|
|
|
return ShowCommandHelp(cCtx, "")
|
|
}
|
|
|
|
// ShowVersion prints the version number of the App
|
|
func ShowVersion(cCtx *Context) {
|
|
VersionPrinter(cCtx)
|
|
}
|
|
|
|
func printVersion(cCtx *Context) {
|
|
_, _ = fmt.Fprintf(cCtx.App.Writer, "%v version %v\n", cCtx.App.Name, cCtx.App.Version)
|
|
}
|
|
|
|
// ShowCompletions prints the lists of commands within a given context
|
|
func ShowCompletions(cCtx *Context) {
|
|
a := cCtx.App
|
|
if a != nil && a.BashComplete != nil {
|
|
a.BashComplete(cCtx)
|
|
}
|
|
}
|
|
|
|
// ShowCommandCompletions prints the custom completions for a given command
|
|
func ShowCommandCompletions(ctx *Context, command string) {
|
|
c := ctx.App.Command(command)
|
|
if c != nil {
|
|
if c.BashComplete != nil {
|
|
c.BashComplete(ctx)
|
|
} else {
|
|
DefaultCompleteWithFlags(c)(ctx)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
// printHelpCustom is the default implementation of HelpPrinterCustom.
|
|
//
|
|
// The customFuncs map will be combined with a default template.FuncMap to
|
|
// allow using arbitrary functions in template rendering.
|
|
func printHelpCustom(out io.Writer, templ string, data interface{}, customFuncs map[string]interface{}) {
|
|
|
|
const maxLineLength = 10000
|
|
|
|
funcMap := template.FuncMap{
|
|
"join": strings.Join,
|
|
"indent": indent,
|
|
"nindent": nindent,
|
|
"trim": strings.TrimSpace,
|
|
"wrap": func(input string, offset int) string { return wrap(input, offset, maxLineLength) },
|
|
"offset": offset,
|
|
}
|
|
|
|
if customFuncs["wrapAt"] != nil {
|
|
if wa, ok := customFuncs["wrapAt"]; ok {
|
|
if waf, ok := wa.(func() int); ok {
|
|
wrapAt := waf()
|
|
customFuncs["wrap"] = func(input string, offset int) string {
|
|
return wrap(input, offset, wrapAt)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for key, value := range customFuncs {
|
|
funcMap[key] = value
|
|
}
|
|
|
|
w := tabwriter.NewWriter(out, 1, 8, 2, ' ', 0)
|
|
t := template.Must(template.New("help").Funcs(funcMap).Parse(templ))
|
|
|
|
err := t.Execute(w, data)
|
|
if err != nil {
|
|
// If the writer is closed, t.Execute will fail, and there's nothing
|
|
// we can do to recover.
|
|
if os.Getenv("CLI_TEMPLATE_ERROR_DEBUG") != "" {
|
|
_, _ = fmt.Fprintf(ErrWriter, "CLI TEMPLATE ERROR: %#v\n", err)
|
|
}
|
|
return
|
|
}
|
|
_ = w.Flush()
|
|
}
|
|
|
|
func printHelp(out io.Writer, templ string, data interface{}) {
|
|
HelpPrinterCustom(out, templ, data, nil)
|
|
}
|
|
|
|
func checkVersion(cCtx *Context) bool {
|
|
found := false
|
|
for _, name := range VersionFlag.Names() {
|
|
if cCtx.Bool(name) {
|
|
found = true
|
|
}
|
|
}
|
|
return found
|
|
}
|
|
|
|
func checkHelp(cCtx *Context) bool {
|
|
found := false
|
|
for _, name := range HelpFlag.Names() {
|
|
if cCtx.Bool(name) {
|
|
found = true
|
|
}
|
|
}
|
|
return found
|
|
}
|
|
|
|
func checkCommandHelp(c *Context, name string) bool {
|
|
if c.Bool("h") || c.Bool("help") {
|
|
_ = ShowCommandHelp(c, name)
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func checkSubcommandHelp(cCtx *Context) bool {
|
|
if cCtx.Bool("h") || cCtx.Bool("help") {
|
|
_ = ShowSubcommandHelp(cCtx)
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func checkShellCompleteFlag(a *App, arguments []string) (bool, []string) {
|
|
if !a.EnableBashCompletion {
|
|
return false, arguments
|
|
}
|
|
|
|
pos := len(arguments) - 1
|
|
lastArg := arguments[pos]
|
|
|
|
if lastArg != "--generate-bash-completion" {
|
|
return false, arguments
|
|
}
|
|
|
|
return true, arguments[:pos]
|
|
}
|
|
|
|
func checkCompletions(cCtx *Context) bool {
|
|
if !cCtx.shellComplete {
|
|
return false
|
|
}
|
|
|
|
if args := cCtx.Args(); args.Present() {
|
|
name := args.First()
|
|
if cmd := cCtx.App.Command(name); cmd != nil {
|
|
// let the command handle the completion
|
|
return false
|
|
}
|
|
}
|
|
|
|
ShowCompletions(cCtx)
|
|
return true
|
|
}
|
|
|
|
func checkCommandCompletions(c *Context, name string) bool {
|
|
if !c.shellComplete {
|
|
return false
|
|
}
|
|
|
|
ShowCommandCompletions(c, name)
|
|
return true
|
|
}
|
|
|
|
func indent(spaces int, v string) string {
|
|
pad := strings.Repeat(" ", spaces)
|
|
return pad + strings.Replace(v, "\n", "\n"+pad, -1)
|
|
}
|
|
|
|
func nindent(spaces int, v string) string {
|
|
return "\n" + indent(spaces, v)
|
|
}
|
|
|
|
func wrap(input string, offset int, wrapAt int) string {
|
|
var sb strings.Builder
|
|
|
|
lines := strings.Split(input, "\n")
|
|
|
|
padding := strings.Repeat(" ", offset)
|
|
|
|
for i, line := range lines {
|
|
if i != 0 {
|
|
sb.WriteString(padding)
|
|
}
|
|
|
|
sb.WriteString(wrapLine(line, offset, wrapAt, padding))
|
|
|
|
if i != len(lines)-1 {
|
|
sb.WriteString("\n")
|
|
}
|
|
}
|
|
|
|
return sb.String()
|
|
}
|
|
|
|
func wrapLine(input string, offset int, wrapAt int, padding string) string {
|
|
if wrapAt <= offset || len(input) <= wrapAt-offset {
|
|
return input
|
|
}
|
|
|
|
lineWidth := wrapAt - offset
|
|
words := strings.Fields(input)
|
|
if len(words) == 0 {
|
|
return input
|
|
}
|
|
|
|
wrapped := words[0]
|
|
spaceLeft := lineWidth - len(wrapped)
|
|
for _, word := range words[1:] {
|
|
if len(word)+1 > spaceLeft {
|
|
wrapped += "\n" + padding + word
|
|
spaceLeft = lineWidth - len(word)
|
|
} else {
|
|
wrapped += " " + word
|
|
spaceLeft -= 1 + len(word)
|
|
}
|
|
}
|
|
|
|
return wrapped
|
|
}
|
|
|
|
func offset(input string, fixed int) int {
|
|
return len(input) + fixed
|
|
}
|