mirror of
https://github.com/StackExchange/dnscontrol.git
synced 2025-01-12 02:17:43 +08:00
169 lines
4.9 KiB
Go
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
|
|
}
|