complete/complete.go

97 lines
2.3 KiB
Go
Raw Normal View History

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"
)
// Complete structs define completion for a command with CLI options
type Complete struct {
Command Command
cmd.CLI
}
// 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".
// command is the struct of the command completion.
func New(name string, command Command) *Complete {
return &Complete{
Command: command,
CLI: cmd.CLI{Name: name},
}
}
// Run get a command, get the typed arguments from environment
// variable, and print out the complete options
// returns success if the completion ran or if the cli matched
// any of the given flags, false otherwise
func (c *Complete) Run() bool {
args, ok := getLine()
if !ok {
// make sure flags parsed,
// in case they were not added in the main program
return c.CLI.Run()
}
2017-05-05 16:25:27 -05:00
Log("Completing args: %s", args)
2017-05-05 08:57:22 -05:00
options := complete(c.Command, 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)
return true
2017-05-05 08:57:22 -05:00
}
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)
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 {
Log("option %T, %s -> %t", option, option, option.Match(l))
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
}
func getLine() ([]string, bool) {
2017-05-05 08:57:22 -05:00
line := os.Getenv(envComplete)
if line == "" {
return nil, false
2017-05-05 08:57:22 -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) {
Log("")
2017-05-05 08:57:22 -05:00
// stdout of program defines the complete options
for _, option := range options {
fmt.Println(option)
}
}