mirror of https://github.com/knative/client.git
Added autocompletion for broker (#1559)
* Added autocompletion for broker * Added broker completion tests * Renamed imports
This commit is contained in:
parent
99c69a9cc1
commit
c6997da944
|
|
@ -38,9 +38,10 @@ func NewBrokerDeleteCommand(p *commands.KnParams) *cobra.Command {
|
||||||
var waitFlags commands.WaitFlags
|
var waitFlags commands.WaitFlags
|
||||||
|
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "delete NAME",
|
Use: "delete NAME",
|
||||||
Short: "Delete a broker",
|
Short: "Delete a broker",
|
||||||
Example: deleteExample,
|
Example: deleteExample,
|
||||||
|
ValidArgsFunction: commands.ResourceNameCompletionFunc(p),
|
||||||
RunE: func(cmd *cobra.Command, args []string) (err error) {
|
RunE: func(cmd *cobra.Command, args []string) (err error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return errors.New("'broker delete' requires the broker name given as single argument")
|
return errors.New("'broker delete' requires the broker name given as single argument")
|
||||||
|
|
|
||||||
|
|
@ -52,9 +52,10 @@ func NewBrokerDescribeCommand(p *commands.KnParams) *cobra.Command {
|
||||||
machineReadablePrintFlags := genericclioptions.NewPrintFlags("")
|
machineReadablePrintFlags := genericclioptions.NewPrintFlags("")
|
||||||
|
|
||||||
cmd := &cobra.Command{
|
cmd := &cobra.Command{
|
||||||
Use: "describe NAME",
|
Use: "describe NAME",
|
||||||
Short: "Describe broker",
|
Short: "Describe broker",
|
||||||
Example: describeExample,
|
Example: describeExample,
|
||||||
|
ValidArgsFunction: commands.ResourceNameCompletionFunc(p),
|
||||||
RunE: func(cmd *cobra.Command, args []string) (err error) {
|
RunE: func(cmd *cobra.Command, args []string) (err error) {
|
||||||
if len(args) != 1 {
|
if len(args) != 1 {
|
||||||
return errors.New("'broker describe' requires the broker name given as single argument")
|
return errors.New("'broker describe' requires the broker name given as single argument")
|
||||||
|
|
|
||||||
|
|
@ -29,6 +29,7 @@ type completionConfig struct {
|
||||||
|
|
||||||
var (
|
var (
|
||||||
resourceToFuncMap = map[string]func(config *completionConfig) []string{
|
resourceToFuncMap = map[string]func(config *completionConfig) []string{
|
||||||
|
"broker": completeBroker,
|
||||||
"service": completeService,
|
"service": completeService,
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
|
@ -123,3 +124,29 @@ func completeService(config *completionConfig) (suggestions []string) {
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func completeBroker(config *completionConfig) (suggestions []string) {
|
||||||
|
suggestions = make([]string, 0)
|
||||||
|
if len(config.args) != 0 {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
namespace, err := config.params.GetNamespace(config.command)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
client, err := config.params.NewEventingClient(namespace)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
brokerList, err := client.ListBrokers(config.command.Context())
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
for _, sug := range brokerList.Items {
|
||||||
|
if !strings.HasPrefix(sug.Name, config.toComplete) {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
suggestions = append(suggestions, sug.Name)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -28,8 +28,11 @@ import (
|
||||||
"k8s.io/apimachinery/pkg/runtime"
|
"k8s.io/apimachinery/pkg/runtime"
|
||||||
"k8s.io/cli-runtime/pkg/genericclioptions"
|
"k8s.io/cli-runtime/pkg/genericclioptions"
|
||||||
clienttesting "k8s.io/client-go/testing"
|
clienttesting "k8s.io/client-go/testing"
|
||||||
|
clienteventingv1 "knative.dev/client/pkg/eventing/v1"
|
||||||
v1 "knative.dev/client/pkg/serving/v1"
|
v1 "knative.dev/client/pkg/serving/v1"
|
||||||
v12 "knative.dev/serving/pkg/apis/serving/v1"
|
eventingv1 "knative.dev/eventing/pkg/apis/eventing/v1"
|
||||||
|
"knative.dev/eventing/pkg/client/clientset/versioned/typed/eventing/v1/fake"
|
||||||
|
servingv1 "knative.dev/serving/pkg/apis/serving/v1"
|
||||||
servingv1fake "knative.dev/serving/pkg/client/clientset/versioned/typed/serving/v1/fake"
|
servingv1fake "knative.dev/serving/pkg/client/clientset/versioned/typed/serving/v1/fake"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -48,39 +51,74 @@ const (
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
testSvc1 = v12.Service{
|
testSvc1 = servingv1.Service{
|
||||||
TypeMeta: metav1.TypeMeta{
|
TypeMeta: metav1.TypeMeta{
|
||||||
Kind: "Service",
|
Kind: "Service",
|
||||||
APIVersion: "serving.knative.dev/v1",
|
APIVersion: "serving.knative.dev/v1",
|
||||||
},
|
},
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-1", Namespace: testNs},
|
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-1", Namespace: testNs},
|
||||||
}
|
}
|
||||||
testSvc2 = v12.Service{
|
testSvc2 = servingv1.Service{
|
||||||
TypeMeta: metav1.TypeMeta{
|
TypeMeta: metav1.TypeMeta{
|
||||||
Kind: "Service",
|
Kind: "Service",
|
||||||
APIVersion: "serving.knative.dev/v1",
|
APIVersion: "serving.knative.dev/v1",
|
||||||
},
|
},
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-2", Namespace: testNs},
|
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-2", Namespace: testNs},
|
||||||
}
|
}
|
||||||
testSvc3 = v12.Service{
|
testSvc3 = servingv1.Service{
|
||||||
TypeMeta: metav1.TypeMeta{
|
TypeMeta: metav1.TypeMeta{
|
||||||
Kind: "Service",
|
Kind: "Service",
|
||||||
APIVersion: "serving.knative.dev/v1",
|
APIVersion: "serving.knative.dev/v1",
|
||||||
},
|
},
|
||||||
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-3", Namespace: testNs},
|
ObjectMeta: metav1.ObjectMeta{Name: "test-svc-3", Namespace: testNs},
|
||||||
}
|
}
|
||||||
testNsServices = []v12.Service{testSvc1, testSvc2, testSvc3}
|
testNsServices = []servingv1.Service{testSvc1, testSvc2, testSvc3}
|
||||||
|
|
||||||
fakeServing = &servingv1fake.FakeServingV1{Fake: &clienttesting.Fake{}}
|
fakeServing = &servingv1fake.FakeServingV1{Fake: &clienttesting.Fake{}}
|
||||||
knParams = &KnParams{
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
testBroker1 = eventingv1.Broker{
|
||||||
|
TypeMeta: metav1.TypeMeta{
|
||||||
|
Kind: "Broker",
|
||||||
|
APIVersion: "eventing.knative.dev/v1",
|
||||||
|
},
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "test-broker-1", Namespace: testNs},
|
||||||
|
}
|
||||||
|
testBroker2 = eventingv1.Broker{
|
||||||
|
TypeMeta: metav1.TypeMeta{
|
||||||
|
Kind: "Broker",
|
||||||
|
APIVersion: "eventing.knative.dev/v1",
|
||||||
|
},
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "test-broker-2", Namespace: testNs},
|
||||||
|
}
|
||||||
|
testBroker3 = eventingv1.Broker{
|
||||||
|
TypeMeta: metav1.TypeMeta{
|
||||||
|
Kind: "Broker",
|
||||||
|
APIVersion: "eventing.knative.dev/v1",
|
||||||
|
},
|
||||||
|
ObjectMeta: metav1.ObjectMeta{Name: "test-broker-3", Namespace: testNs},
|
||||||
|
}
|
||||||
|
testNsBrokers = []eventingv1.Broker{testBroker1, testBroker2, testBroker3}
|
||||||
|
|
||||||
|
fakeEventing = &fake.FakeEventingV1{Fake: &clienttesting.Fake{}}
|
||||||
|
)
|
||||||
|
|
||||||
|
var knParams = initialiseKnParams()
|
||||||
|
|
||||||
|
func initialiseKnParams() *KnParams {
|
||||||
|
return &KnParams{
|
||||||
NewServingClient: func(namespace string) (v1.KnServingClient, error) {
|
NewServingClient: func(namespace string) (v1.KnServingClient, error) {
|
||||||
return v1.NewKnServingClient(fakeServing, namespace), nil
|
return v1.NewKnServingClient(fakeServing, namespace), nil
|
||||||
},
|
},
|
||||||
NewGitopsServingClient: func(namespace string, dir string) (v1.KnServingClient, error) {
|
NewGitopsServingClient: func(namespace string, dir string) (v1.KnServingClient, error) {
|
||||||
return v1.NewKnServingGitOpsClient(namespace, dir), nil
|
return v1.NewKnServingGitOpsClient(namespace, dir), nil
|
||||||
},
|
},
|
||||||
|
NewEventingClient: func(namespace string) (clienteventingv1.KnEventingClient, error) {
|
||||||
|
return clienteventingv1.NewKnEventingClient(fakeEventing, namespace), nil
|
||||||
|
},
|
||||||
}
|
}
|
||||||
)
|
}
|
||||||
|
|
||||||
func TestResourceNameCompletionFuncService(t *testing.T) {
|
func TestResourceNameCompletionFuncService(t *testing.T) {
|
||||||
completionFunc := ResourceNameCompletionFunc(knParams)
|
completionFunc := ResourceNameCompletionFunc(knParams)
|
||||||
|
|
@ -90,7 +128,7 @@ func TestResourceNameCompletionFuncService(t *testing.T) {
|
||||||
if a.GetNamespace() == errorNs {
|
if a.GetNamespace() == errorNs {
|
||||||
return true, nil, errors.NewInternalError(fmt.Errorf("unable to list services"))
|
return true, nil, errors.NewInternalError(fmt.Errorf("unable to list services"))
|
||||||
}
|
}
|
||||||
return true, &v12.ServiceList{Items: testNsServices}, nil
|
return true, &servingv1.ServiceList{Items: testNsServices}, nil
|
||||||
})
|
})
|
||||||
|
|
||||||
tests := []testType{
|
tests := []testType{
|
||||||
|
|
@ -143,6 +181,83 @@ func TestResourceNameCompletionFuncService(t *testing.T) {
|
||||||
"service",
|
"service",
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, tt := range tests {
|
||||||
|
cmd := getResourceCommandWithTestSubcommand(tt.resource, tt.namespace != "", tt.resource != "no-parent")
|
||||||
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
config := &completionConfig{
|
||||||
|
params: tt.p,
|
||||||
|
command: cmd,
|
||||||
|
args: tt.args,
|
||||||
|
toComplete: tt.toComplete,
|
||||||
|
}
|
||||||
|
expectedFunc := resourceToFuncMap[tt.resource]
|
||||||
|
if expectedFunc == nil {
|
||||||
|
expectedFunc = func(config *completionConfig) []string {
|
||||||
|
return []string{}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
cmd.Flags().Set("namespace", tt.namespace)
|
||||||
|
actualSuggestions, actualDirective := completionFunc(cmd, tt.args, tt.toComplete)
|
||||||
|
expectedSuggestions := expectedFunc(config)
|
||||||
|
expectedDirective := cobra.ShellCompDirectiveNoFileComp
|
||||||
|
assert.DeepEqual(t, actualSuggestions, expectedSuggestions)
|
||||||
|
assert.Equal(t, actualDirective, expectedDirective)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestResourceNameCompletionFuncBroker(t *testing.T) {
|
||||||
|
completionFunc := ResourceNameCompletionFunc(knParams)
|
||||||
|
|
||||||
|
fakeEventing.AddReactor("list", "brokers", func(action clienttesting.Action) (bool, runtime.Object, error) {
|
||||||
|
if action.GetNamespace() == errorNs {
|
||||||
|
return true, nil, errors.NewInternalError(fmt.Errorf("unable to list services"))
|
||||||
|
}
|
||||||
|
return true, &eventingv1.BrokerList{Items: testNsBrokers}, nil
|
||||||
|
})
|
||||||
|
tests := []testType{
|
||||||
|
{
|
||||||
|
"Empty suggestions when non-zero args",
|
||||||
|
testNs,
|
||||||
|
knParams,
|
||||||
|
[]string{"xyz"},
|
||||||
|
"",
|
||||||
|
"broker",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Empty suggestions when no namespace flag",
|
||||||
|
"",
|
||||||
|
knParams,
|
||||||
|
nil,
|
||||||
|
"",
|
||||||
|
"broker",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Suggestions when test-ns namespace set",
|
||||||
|
testNs,
|
||||||
|
knParams,
|
||||||
|
nil,
|
||||||
|
"",
|
||||||
|
"broker",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Empty suggestions when toComplete is not a prefix",
|
||||||
|
testNs,
|
||||||
|
knParams,
|
||||||
|
nil,
|
||||||
|
"xyz",
|
||||||
|
"broker",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Empty suggestions when error during list operation",
|
||||||
|
errorNs,
|
||||||
|
knParams,
|
||||||
|
nil,
|
||||||
|
"",
|
||||||
|
"broker",
|
||||||
|
},
|
||||||
|
}
|
||||||
for _, tt := range tests {
|
for _, tt := range tests {
|
||||||
cmd := getResourceCommandWithTestSubcommand(tt.resource, tt.namespace != "", tt.resource != "no-parent")
|
cmd := getResourceCommandWithTestSubcommand(tt.resource, tt.namespace != "", tt.resource != "no-parent")
|
||||||
t.Run(tt.name, func(t *testing.T) {
|
t.Run(tt.name, func(t *testing.T) {
|
||||||
|
|
@ -273,7 +388,7 @@ func setupTempDir(t *testing.T) string {
|
||||||
err = os.MkdirAll(svcPath, 0700)
|
err = os.MkdirAll(svcPath, 0700)
|
||||||
assert.NilError(t, err)
|
assert.NilError(t, err)
|
||||||
|
|
||||||
for i, testSvc := range []v12.Service{testSvc1, testSvc2, testSvc3} {
|
for i, testSvc := range []servingv1.Service{testSvc1, testSvc2, testSvc3} {
|
||||||
tempFile, err := os.Create(path.Join(svcPath, fmt.Sprintf("test-svc-%d.yaml", i+1)))
|
tempFile, err := os.Create(path.Join(svcPath, fmt.Sprintf("test-svc-%d.yaml", i+1)))
|
||||||
assert.NilError(t, err)
|
assert.NilError(t, err)
|
||||||
writeToFile(t, testSvc, tempFile)
|
writeToFile(t, testSvc, tempFile)
|
||||||
|
|
@ -282,7 +397,7 @@ func setupTempDir(t *testing.T) string {
|
||||||
return tempDir
|
return tempDir
|
||||||
}
|
}
|
||||||
|
|
||||||
func writeToFile(t *testing.T, testSvc v12.Service, tempFile *os.File) {
|
func writeToFile(t *testing.T, testSvc servingv1.Service, tempFile *os.File) {
|
||||||
yamlPrinter, err := genericclioptions.NewJSONYamlPrintFlags().ToPrinter("yaml")
|
yamlPrinter, err := genericclioptions.NewJSONYamlPrintFlags().ToPrinter("yaml")
|
||||||
assert.NilError(t, err)
|
assert.NilError(t, err)
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue