2017-05-06 14:06:49 -05:00
|
|
|
// Package complete provides a tool for bash writing bash completion in go.
|
|
|
|
//
|
|
|
|
// Writing bash completion scripts is a hard work. This package provides an easy way
|
|
|
|
// to create bash completion scripts for any command, and also an easy way to install/uninstall
|
|
|
|
// the completion of the command.
|
2017-05-05 08:57:22 -05:00
|
|
|
package complete
|
|
|
|
|
|
|
|
import (
|
2017-05-22 23:33:14 -05:00
|
|
|
"flag"
|
2017-05-05 08:57:22 -05:00
|
|
|
"fmt"
|
2017-11-04 03:21:41 -05:00
|
|
|
"io"
|
2017-05-05 08:57:22 -05:00
|
|
|
"os"
|
2018-10-19 12:10:37 -05:00
|
|
|
"strconv"
|
2017-05-06 14:06:49 -05:00
|
|
|
|
|
|
|
"github.com/posener/complete/cmd"
|
2017-11-04 03:21:41 -05:00
|
|
|
"github.com/posener/complete/match"
|
2017-05-05 08:57:22 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2018-10-19 12:10:37 -05:00
|
|
|
envLine = "COMP_LINE"
|
|
|
|
envPoint = "COMP_POINT"
|
|
|
|
envDebug = "COMP_DEBUG"
|
2017-05-05 08:57:22 -05:00
|
|
|
)
|
|
|
|
|
2017-05-09 23:28:43 -05:00
|
|
|
// Complete structs define completion for a command with CLI options
|
|
|
|
type Complete struct {
|
|
|
|
Command Command
|
|
|
|
cmd.CLI
|
2017-11-04 03:21:41 -05:00
|
|
|
Out io.Writer
|
2017-05-09 23:28:43 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// New creates a new complete command.
|
2017-05-06 23:59:42 -05:00
|
|
|
// name is the name of command we want to auto complete.
|
|
|
|
// IMPORTANT: it must be the same name - if the auto complete
|
|
|
|
// completes the 'go' command, name must be equal to "go".
|
2017-05-09 23:28:43 -05:00
|
|
|
// command is the struct of the command completion.
|
|
|
|
func New(name string, command Command) *Complete {
|
|
|
|
return &Complete{
|
|
|
|
Command: command,
|
|
|
|
CLI: cmd.CLI{Name: name},
|
2017-11-04 03:21:41 -05:00
|
|
|
Out: os.Stdout,
|
2017-05-09 23:28:43 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-22 23:33:14 -05:00
|
|
|
// Run runs the completion and add installation flags beforehand.
|
|
|
|
// The flags are added to the main flag CommandLine variable.
|
|
|
|
func (c *Complete) Run() bool {
|
|
|
|
c.AddFlags(nil)
|
|
|
|
flag.Parse()
|
|
|
|
return c.Complete()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Complete a command from completion line in environment variable,
|
|
|
|
// and print out the complete options.
|
2017-05-09 23:28:43 -05:00
|
|
|
// returns success if the completion ran or if the cli matched
|
|
|
|
// any of the given flags, false otherwise
|
2017-05-22 23:33:14 -05:00
|
|
|
// For installation: it assumes that flags were added and parsed before
|
|
|
|
// it was called.
|
|
|
|
func (c *Complete) Complete() bool {
|
2018-10-19 12:10:37 -05:00
|
|
|
line, point, ok := getEnv()
|
2017-05-06 10:55:54 -05:00
|
|
|
if !ok {
|
2017-05-09 23:28:43 -05:00
|
|
|
// make sure flags parsed,
|
|
|
|
// in case they were not added in the main program
|
|
|
|
return c.CLI.Run()
|
2017-05-06 10:55:54 -05:00
|
|
|
}
|
2018-10-19 12:10:37 -05:00
|
|
|
|
2018-10-25 11:38:01 -05:00
|
|
|
if point >= 0 && point < len(line) {
|
|
|
|
line = line[:point]
|
|
|
|
}
|
2018-10-19 12:10:37 -05:00
|
|
|
|
2018-10-25 11:38:01 -05:00
|
|
|
Log("Completing phrase: %s", line)
|
|
|
|
a := newArgs(line)
|
2017-11-04 04:20:50 -05:00
|
|
|
Log("Completing last field: %s", a.Last)
|
2017-05-11 12:28:31 -05:00
|
|
|
options := c.Command.Predict(a)
|
2017-11-04 03:21:41 -05:00
|
|
|
Log("Options: %s", options)
|
2017-05-05 08:57:22 -05:00
|
|
|
|
2017-11-04 03:21:41 -05:00
|
|
|
// filter only options that match the last argument
|
|
|
|
matches := []string{}
|
|
|
|
for _, option := range options {
|
|
|
|
if match.Prefix(option, a.Last) {
|
|
|
|
matches = append(matches, option)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Log("Matches: %s", matches)
|
|
|
|
c.output(matches)
|
2017-05-09 23:28:43 -05:00
|
|
|
return true
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
|
|
|
|
2018-10-19 12:10:37 -05:00
|
|
|
func getEnv() (line string, point int, ok bool) {
|
|
|
|
line = os.Getenv(envLine)
|
2017-05-05 08:57:22 -05:00
|
|
|
if line == "" {
|
2018-10-19 12:10:37 -05:00
|
|
|
return
|
|
|
|
}
|
|
|
|
point, err := strconv.Atoi(os.Getenv(envPoint))
|
|
|
|
if err != nil {
|
|
|
|
// If failed parsing point for some reason, set it to point
|
|
|
|
// on the end of the line.
|
|
|
|
Log("Failed parsing point %s: %v", os.Getenv(envPoint), err)
|
|
|
|
point = len(line)
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
2018-10-19 12:10:37 -05:00
|
|
|
return line, point, true
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
|
|
|
|
2017-11-04 03:21:41 -05:00
|
|
|
func (c *Complete) output(options []string) {
|
2017-05-05 08:57:22 -05:00
|
|
|
// stdout of program defines the complete options
|
|
|
|
for _, option := range options {
|
2017-11-04 03:21:41 -05:00
|
|
|
fmt.Fprintln(c.Out, option)
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
|
|
|
}
|