package completion
import (
"fmt"
"sort"
"strings"
"github.com/spf13/cobra"
"github.com/git-bug/git-bug/bridge"
"github.com/git-bug/git-bug/bridge/core/auth"
"github.com/git-bug/git-bug/commands/execenv"
)
type ValidArgsFunction func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective)
func HandleError(err error) (completions []string, directives cobra.ShellCompDirective) {
return nil, cobra.ShellCompDirectiveError
}
func Bridge(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
bridges, err := bridge.ConfiguredBridges(env.Backend)
if err != nil {
return HandleError(err)
}
completions = make([]string, len(bridges))
for i, bridge := range bridges {
completions[i] = bridge + "\t" + "Bridge"
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
}
func BridgeAuth(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
creds, err := auth.List(env.Backend)
if err != nil {
return HandleError(err)
}
completions = make([]string, len(creds))
for i, cred := range creds {
meta := make([]string, 0, len(cred.Metadata()))
for k, v := range cred.Metadata() {
meta = append(meta, k+":"+v)
}
sort.Strings(meta)
metaFmt := strings.Join(meta, ",")
completions[i] = cred.ID().Human() + "\t" + cred.Target() + " " + string(cred.Kind()) + " " + metaFmt
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
}
func From(choices []string) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) ([]string, cobra.ShellCompDirective) {
return choices, cobra.ShellCompDirectiveNoFileComp
}
}
func GitRemote(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
remoteMap, err := env.Backend.GetRemotes()
if err != nil {
return HandleError(err)
}
completions = make([]string, 0, len(remoteMap))
for remote, url := range remoteMap {
completions = append(completions, remote+"\t"+"Remote: "+url)
}
sort.Strings(completions)
return completions, cobra.ShellCompDirectiveNoFileComp
}
}
func Label(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
labels := env.Backend.Bugs().ValidLabels()
completions = make([]string, len(labels))
for i, label := range labels {
if strings.Contains(label.String(), " ") {
completions[i] = fmt.Sprintf("\"%s\"\tLabel", label.String())
} else {
completions[i] = fmt.Sprintf("%s\tLabel", label.String())
}
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
}
func Ls(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if strings.HasPrefix(toComplete, "status:") {
completions = append(completions, "status:open\tOpen bugs")
completions = append(completions, "status:closed\tClosed bugs")
return completions, cobra.ShellCompDirectiveDefault
}
byPerson := []string{"author:", "participant:", "actor:"}
byLabel := []string{"label:", "no:"}
needBackend := false
for _, key := range append(byPerson, byLabel...) {
if strings.HasPrefix(toComplete, key) {
needBackend = true
}
}
if needBackend {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
}
for _, key := range byPerson {
if !strings.HasPrefix(toComplete, key) {
continue
}
ids := env.Backend.Identities().AllIds()
completions = make([]string, len(ids))
for i, id := range ids {
user, err := env.Backend.Identities().ResolveExcerpt(id)
if err != nil {
return HandleError(err)
}
var handle string
if user.Login != "" {
handle = user.Login
} else {
// "author:John Doe" does not work yet, so use the first name.
handle = strings.Split(user.Name, " ")[0]
}
completions[i] = key + handle + "\t" + user.DisplayName()
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
for _, key := range byLabel {
if !strings.HasPrefix(toComplete, key) {
continue
}
labels := env.Backend.Bugs().ValidLabels()
completions = make([]string, len(labels))
for i, label := range labels {
if strings.Contains(label.String(), " ") {
completions[i] = key + "\"" + string(label) + "\""
} else {
completions[i] = key + string(label)
}
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
completions = []string{
"actor:\tFilter by actor",
"author:\tFilter by author",
"label:\tFilter by label",
"no:\tExclude bugs by label",
"participant:\tFilter by participant",
"status:\tFilter by open/close status",
"title:\tFilter by title",
}
return completions, cobra.ShellCompDirectiveNoSpace
}
}
func User(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
ids := env.Backend.Identities().AllIds()
completions = make([]string, len(ids))
for i, id := range ids {
user, err := env.Backend.Identities().ResolveExcerpt(id)
if err != nil {
return HandleError(err)
}
completions[i] = user.Id().Human() + "\t" + user.DisplayName()
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
}
func UserForQuery(env *execenv.Env) ValidArgsFunction {
return func(cmd *cobra.Command, args []string, toComplete string) (completions []string, directives cobra.ShellCompDirective) {
if err := execenv.LoadBackend(env)(cmd, args); err != nil {
return HandleError(err)
}
defer func() {
_ = env.Backend.Close()
}()
ids := env.Backend.Identities().AllIds()
completions = make([]string, len(ids))
for i, id := range ids {
user, err := env.Backend.Identities().ResolveExcerpt(id)
if err != nil {
return HandleError(err)
}
var handle string
if user.Login != "" {
handle = user.Login
} else {
// "author:John Doe" does not work yet, so use the first name.
handle = strings.Split(user.Name, " ")[0]
}
completions[i] = handle + "\t" + user.DisplayName()
}
return completions, cobra.ShellCompDirectiveNoFileComp
}
}