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 (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"strings"
|
2017-05-06 14:06:49 -05:00
|
|
|
|
|
|
|
"github.com/posener/complete/cmd"
|
2017-05-05 08:57:22 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
envComplete = "COMP_LINE"
|
|
|
|
envDebug = "COMP_DEBUG"
|
|
|
|
)
|
|
|
|
|
2017-05-06 00:24:17 -05:00
|
|
|
// Run get a command, get the typed arguments from environment
|
|
|
|
// variable, and print out the complete options
|
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".
|
|
|
|
func Run(name string, c Command) {
|
2017-05-06 10:55:54 -05:00
|
|
|
args, ok := getLine()
|
|
|
|
if !ok {
|
2017-05-06 23:59:42 -05:00
|
|
|
cmd.Run(name)
|
2017-05-06 10:55:54 -05:00
|
|
|
return
|
|
|
|
}
|
2017-05-05 16:25:27 -05:00
|
|
|
Log("Completing args: %s", args)
|
2017-05-05 08:57:22 -05:00
|
|
|
|
2017-05-06 00:24:17 -05:00
|
|
|
options := complete(c, args)
|
2017-05-05 08:57:22 -05:00
|
|
|
|
2017-05-05 16:25:27 -05:00
|
|
|
Log("Completion: %s", options)
|
2017-05-05 08:57:22 -05:00
|
|
|
output(options)
|
|
|
|
}
|
|
|
|
|
2017-05-06 00:24:17 -05:00
|
|
|
// complete get a command an command line arguments and returns
|
|
|
|
// matching completion options
|
|
|
|
func complete(c Command, args []string) (matching []string) {
|
|
|
|
options, _ := c.options(args[:len(args)-1])
|
2017-05-05 08:57:22 -05:00
|
|
|
|
2017-05-06 00:24:17 -05:00
|
|
|
// choose only matching options
|
|
|
|
l := last(args)
|
2017-05-05 13:59:10 -05:00
|
|
|
for _, option := range options {
|
2017-05-06 14:06:49 -05:00
|
|
|
if option.Match(l) {
|
2017-05-06 00:24:17 -05:00
|
|
|
matching = append(matching, option.String())
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-05-06 10:55:54 -05:00
|
|
|
func getLine() ([]string, bool) {
|
2017-05-05 08:57:22 -05:00
|
|
|
line := os.Getenv(envComplete)
|
|
|
|
if line == "" {
|
2017-05-06 10:55:54 -05:00
|
|
|
return nil, false
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
2017-05-06 10:55:54 -05:00
|
|
|
return strings.Split(line, " "), true
|
2017-05-05 08:57:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func last(args []string) (last string) {
|
|
|
|
if len(args) > 0 {
|
|
|
|
last = args[len(args)-1]
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func output(options []string) {
|
|
|
|
// stdout of program defines the complete options
|
|
|
|
for _, option := range options {
|
|
|
|
fmt.Println(option)
|
|
|
|
}
|
|
|
|
}
|