dnscontrol/commands/completion.go
2023-11-22 08:53:04 -05:00

169 lines
4.9 KiB
Go

package commands
import (
"embed"
"errors"
"fmt"
"io"
"os"
"path"
"strings"
"text/template"
"unicode/utf8"
"github.com/urfave/cli/v2"
)
//go:embed completion-scripts/completion.*.gotmpl
var completionScripts embed.FS
func shellCompletionCommand() *cli.Command {
supportedShells, templates, err := getCompletionSupportedShells()
if err != nil {
panic(err)
}
return &cli.Command{
Name: "shell-completion",
Usage: "generate shell completion scripts",
ArgsUsage: fmt.Sprintf("[ %s ]", strings.Join(supportedShells, " | ")),
Description: fmt.Sprintf("Generate shell completion script for [ %s ]", strings.Join(supportedShells, " | ")),
BashComplete: func(ctx *cli.Context) {
for _, shell := range supportedShells {
if strings.HasPrefix(shell, ctx.Args().First()) {
ctx.App.Writer.Write([]byte(shell + "\n"))
}
}
},
Action: func(ctx *cli.Context) error {
var inputShell string
if inputShell = ctx.Args().First(); inputShell == "" {
if inputShell = os.Getenv("SHELL"); inputShell == "" {
return cli.Exit(errors.New("shell not specified"), 1)
}
}
shellName := path.Base(inputShell) // necessary if using $SHELL, noop otherwise
template := templates[shellName]
if template == nil {
return cli.Exit(fmt.Errorf("unknown shell: %s", inputShell), 1)
}
err = template.Execute(ctx.App.Writer, struct {
App *cli.App
}{ctx.App})
if err != nil {
return cli.Exit(fmt.Errorf("failed to print completion script: %w", err), 1)
}
return nil
},
}
}
var _ = cmd(catUtils, shellCompletionCommand())
// getCompletionSupportedShells returns a list of shells with available completions.
// The list is generated from the embedded completion scripts.
func getCompletionSupportedShells() (shells []string, shellCompletionScripts map[string]*template.Template, err error) {
scripts, err := completionScripts.ReadDir("completion-scripts")
if err != nil {
return nil, nil, fmt.Errorf("failed to read completion scripts: %w", err)
}
shellCompletionScripts = make(map[string]*template.Template)
for _, f := range scripts {
fNameWithoutExtension := strings.TrimSuffix(f.Name(), ".gotmpl")
shellName := strings.TrimPrefix(path.Ext(fNameWithoutExtension), ".")
content, err := completionScripts.ReadFile(path.Join("completion-scripts", f.Name()))
if err != nil {
return nil, nil, fmt.Errorf("failed to read completion script %s", f.Name())
}
t := template.New(shellName)
t, err = t.Parse(string(content))
if err != nil {
return nil, nil, fmt.Errorf("failed to parse template %s", f.Name())
}
shells = append(shells, shellName)
shellCompletionScripts[shellName] = t
}
return shells, shellCompletionScripts, nil
}
func dnscontrolPrintCommandSuggestions(commands []*cli.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 dnscontrolCliArgContains(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 dnscontrolPrintFlagSuggestions(lastArg string, flags []cli.Flag, writer io.Writer) {
cur := strings.TrimPrefix(lastArg, "-")
cur = strings.TrimPrefix(cur, "-")
for _, flag := range flags {
if bflag, ok := flag.(*cli.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 // reuse 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 && !dnscontrolCliArgContains(name) {
flagCompletion := fmt.Sprintf("%s%s", strings.Repeat("-", count), name)
_, _ = fmt.Fprintln(writer, flagCompletion)
}
}
}
}
func islastFlagComplete(lastArg string, flags []cli.Flag) bool {
cur := strings.TrimPrefix(lastArg, "-")
cur = strings.TrimPrefix(cur, "-")
for _, flag := range flags {
for _, name := range flag.Names() {
name = strings.TrimSpace(name)
if strings.HasPrefix(name, cur) && cur != name && !dnscontrolCliArgContains(name) {
return false
}
}
}
return true
}