2020-04-03 19:43:43 +00:00
|
|
|
package cobra
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"strings"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
2020-04-06 17:28:44 +00:00
|
|
|
func validArgsFunc(cmd *Command, args []string, toComplete string) ([]string, ShellCompDirective) {
|
2020-04-03 19:43:43 +00:00
|
|
|
if len(args) != 0 {
|
2020-04-06 17:28:44 +00:00
|
|
|
return nil, ShellCompDirectiveNoFileComp
|
2020-04-03 19:43:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var completions []string
|
|
|
|
for _, comp := range []string{"one", "two"} {
|
|
|
|
if strings.HasPrefix(comp, toComplete) {
|
|
|
|
completions = append(completions, comp)
|
|
|
|
}
|
|
|
|
}
|
2020-04-06 17:28:44 +00:00
|
|
|
return completions, ShellCompDirectiveDefault
|
2020-04-03 19:43:43 +00:00
|
|
|
}
|
|
|
|
|
2020-04-06 17:28:44 +00:00
|
|
|
func validArgsFunc2(cmd *Command, args []string, toComplete string) ([]string, ShellCompDirective) {
|
2020-04-03 19:43:43 +00:00
|
|
|
if len(args) != 0 {
|
2020-04-06 17:28:44 +00:00
|
|
|
return nil, ShellCompDirectiveNoFileComp
|
2020-04-03 19:43:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var completions []string
|
|
|
|
for _, comp := range []string{"three", "four"} {
|
|
|
|
if strings.HasPrefix(comp, toComplete) {
|
|
|
|
completions = append(completions, comp)
|
|
|
|
}
|
|
|
|
}
|
2020-04-06 17:28:44 +00:00
|
|
|
return completions, ShellCompDirectiveDefault
|
2020-04-03 19:43:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestValidArgsFuncSingleCmd(t *testing.T) {
|
|
|
|
rootCmd := &Command{
|
|
|
|
Use: "root",
|
|
|
|
ValidArgsFunction: validArgsFunc,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test completing an empty string
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err := executeCommand(rootCmd, ShellCompRequestCmd, "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := strings.Join([]string{
|
|
|
|
"one",
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with a prefix
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestValidArgsFuncSingleCmdInvalidArg(t *testing.T) {
|
|
|
|
rootCmd := &Command{
|
|
|
|
Use: "root",
|
|
|
|
// If we don't specify a value for Args, this test fails.
|
|
|
|
// This is only true for a root command without any subcommands, and is caused
|
|
|
|
// by the fact that the __complete command becomes a subcommand when there should not be one.
|
|
|
|
// The problem is in the implementation of legacyArgs().
|
|
|
|
Args: MinimumNArgs(1),
|
|
|
|
ValidArgsFunction: validArgsFunc,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with wrong number of args
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err := executeCommand(rootCmd, ShellCompRequestCmd, "unexpectedArg", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := strings.Join([]string{
|
|
|
|
":4",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestValidArgsFuncChildCmds(t *testing.T) {
|
|
|
|
rootCmd := &Command{Use: "root", Args: NoArgs, Run: emptyRun}
|
|
|
|
child1Cmd := &Command{
|
|
|
|
Use: "child1",
|
|
|
|
ValidArgsFunction: validArgsFunc,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
child2Cmd := &Command{
|
|
|
|
Use: "child2",
|
|
|
|
ValidArgsFunction: validArgsFunc2,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
rootCmd.AddCommand(child1Cmd, child2Cmd)
|
|
|
|
|
|
|
|
// Test completion of first sub-command with empty argument
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err := executeCommand(rootCmd, ShellCompRequestCmd, "child1", "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := strings.Join([]string{
|
|
|
|
"one",
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test completion of first sub-command with a prefix to complete
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "child1", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with wrong number of args
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "child1", "unexpectedArg", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
":4",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test completion of second sub-command with empty argument
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "child2", "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"three",
|
|
|
|
"four",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "child2", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"three",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with wrong number of args
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "child2", "unexpectedArg", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
":4",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestValidArgsFuncAliases(t *testing.T) {
|
|
|
|
rootCmd := &Command{Use: "root", Args: NoArgs, Run: emptyRun}
|
|
|
|
child := &Command{
|
|
|
|
Use: "child",
|
|
|
|
Aliases: []string{"son", "daughter"},
|
|
|
|
ValidArgsFunction: validArgsFunc,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
rootCmd.AddCommand(child)
|
|
|
|
|
|
|
|
// Test completion of first sub-command with empty argument
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err := executeCommand(rootCmd, ShellCompRequestCmd, "son", "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := strings.Join([]string{
|
|
|
|
"one",
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test completion of first sub-command with a prefix to complete
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "daughter", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"two",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with wrong number of args
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "son", "unexpectedArg", "t")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
":4",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestValidArgsFuncInScript(t *testing.T) {
|
|
|
|
rootCmd := &Command{Use: "root", Args: NoArgs, Run: emptyRun}
|
|
|
|
child := &Command{
|
|
|
|
Use: "child",
|
|
|
|
ValidArgsFunction: validArgsFunc,
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
rootCmd.AddCommand(child)
|
|
|
|
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
rootCmd.GenBashCompletion(buf)
|
|
|
|
output := buf.String()
|
|
|
|
|
|
|
|
check(t, output, "has_completion_function=1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNoValidArgsFuncInScript(t *testing.T) {
|
|
|
|
rootCmd := &Command{Use: "root", Args: NoArgs, Run: emptyRun}
|
|
|
|
child := &Command{
|
|
|
|
Use: "child",
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
rootCmd.AddCommand(child)
|
|
|
|
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
rootCmd.GenBashCompletion(buf)
|
|
|
|
output := buf.String()
|
|
|
|
|
|
|
|
checkOmit(t, output, "has_completion_function=1")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestFlagCompletionInGo(t *testing.T) {
|
|
|
|
rootCmd := &Command{
|
|
|
|
Use: "root",
|
|
|
|
Run: emptyRun,
|
|
|
|
}
|
|
|
|
rootCmd.Flags().IntP("introot", "i", -1, "help message for flag introot")
|
2020-04-06 17:28:44 +00:00
|
|
|
rootCmd.RegisterFlagCompletionFunc("introot", func(cmd *Command, args []string, toComplete string) ([]string, ShellCompDirective) {
|
2020-04-03 19:43:43 +00:00
|
|
|
completions := []string{}
|
|
|
|
for _, comp := range []string{"1", "2", "10"} {
|
|
|
|
if strings.HasPrefix(comp, toComplete) {
|
|
|
|
completions = append(completions, comp)
|
|
|
|
}
|
|
|
|
}
|
2020-04-06 17:28:44 +00:00
|
|
|
return completions, ShellCompDirectiveDefault
|
2020-04-03 19:43:43 +00:00
|
|
|
})
|
|
|
|
rootCmd.Flags().String("filename", "", "Enter a filename")
|
2020-04-06 17:28:44 +00:00
|
|
|
rootCmd.RegisterFlagCompletionFunc("filename", func(cmd *Command, args []string, toComplete string) ([]string, ShellCompDirective) {
|
2020-04-03 19:43:43 +00:00
|
|
|
completions := []string{}
|
|
|
|
for _, comp := range []string{"file.yaml", "myfile.json", "file.xml"} {
|
|
|
|
if strings.HasPrefix(comp, toComplete) {
|
|
|
|
completions = append(completions, comp)
|
|
|
|
}
|
|
|
|
}
|
2020-04-06 17:28:44 +00:00
|
|
|
return completions, ShellCompDirectiveNoSpace | ShellCompDirectiveNoFileComp
|
2020-04-03 19:43:43 +00:00
|
|
|
})
|
|
|
|
|
|
|
|
// Test completing an empty string
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err := executeCommand(rootCmd, ShellCompRequestCmd, "--introot", "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected := strings.Join([]string{
|
|
|
|
"1",
|
|
|
|
"2",
|
|
|
|
"10",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with a prefix
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "--introot", "1")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"1",
|
|
|
|
"10",
|
|
|
|
":0",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveDefault", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test completing an empty string
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "--filename", "")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"file.yaml",
|
|
|
|
"myfile.json",
|
|
|
|
"file.xml",
|
|
|
|
":6",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoSpace, ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check completing with a prefix
|
2020-04-06 17:28:44 +00:00
|
|
|
output, err = executeCommand(rootCmd, ShellCompRequestCmd, "--filename", "f")
|
2020-04-03 19:43:43 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Errorf("Unexpected error: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
expected = strings.Join([]string{
|
|
|
|
"file.yaml",
|
|
|
|
"file.xml",
|
|
|
|
":6",
|
2020-04-06 17:28:44 +00:00
|
|
|
"Completion ended with directive: ShellCompDirectiveNoSpace, ShellCompDirectiveNoFileComp", ""}, "\n")
|
2020-04-03 19:43:43 +00:00
|
|
|
|
|
|
|
if output != expected {
|
|
|
|
t.Errorf("expected: %q, got: %q", expected, output)
|
|
|
|
}
|
|
|
|
}
|