mirror of https://github.com/knative/client.git
Add E2E tests for trigger (#645)
* add integration test for trigger * update doc
This commit is contained in:
parent
dd2dc97f92
commit
e0319177d6
|
|
@ -20,6 +20,11 @@ import (
|
||||||
"knative.dev/client/pkg/kn/commands"
|
"knative.dev/client/pkg/kn/commands"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
// How often to retry in case of an optimistic lock error when replacing a trigger (--force)
|
||||||
|
MaxUpdateRetries = 3
|
||||||
|
)
|
||||||
|
|
||||||
// NewTriggerCommand to create trigger command group
|
// NewTriggerCommand to create trigger command group
|
||||||
func NewTriggerCommand(p *commands.KnParams) *cobra.Command {
|
func NewTriggerCommand(p *commands.KnParams) *cobra.Command {
|
||||||
triggerCmd := &cobra.Command{
|
triggerCmd := &cobra.Command{
|
||||||
|
|
|
||||||
|
|
@ -19,6 +19,7 @@ import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
||||||
|
|
||||||
"knative.dev/eventing/pkg/apis/eventing/v1alpha1"
|
"knative.dev/eventing/pkg/apis/eventing/v1alpha1"
|
||||||
duckv1 "knative.dev/pkg/apis/duck/v1"
|
duckv1 "knative.dev/pkg/apis/duck/v1"
|
||||||
|
|
@ -68,43 +69,52 @@ func NewTriggerUpdateCommand(p *commands.KnParams) *cobra.Command {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
trigger, err := eventingClient.GetTrigger(name)
|
var retries = 0
|
||||||
if err != nil {
|
for {
|
||||||
return err
|
trigger, err := eventingClient.GetTrigger(name)
|
||||||
}
|
|
||||||
|
|
||||||
b := client_v1alpha1.NewTriggerBuilderFromExisting(trigger)
|
|
||||||
|
|
||||||
if cmd.Flags().Changed("broker") {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"cannot update trigger '%s' because broker is immutable", name)
|
|
||||||
}
|
|
||||||
if cmd.Flags().Changed("filter") {
|
|
||||||
updated, removed, err := triggerUpdateFlags.GetUpdateFilters()
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf(
|
|
||||||
"cannot update trigger '%s' because %s", name, err)
|
|
||||||
}
|
|
||||||
existing := extractFilters(trigger)
|
|
||||||
b.Filters(existing.Merge(updated).Remove(removed))
|
|
||||||
}
|
|
||||||
if cmd.Flags().Changed("sink") {
|
|
||||||
destination, err := sinkFlags.ResolveSink(dynamicClient, namespace)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
b.Subscriber(&duckv1.Destination{
|
|
||||||
Ref: destination.Ref,
|
b := client_v1alpha1.NewTriggerBuilderFromExisting(trigger)
|
||||||
URI: destination.URI,
|
|
||||||
})
|
if cmd.Flags().Changed("broker") {
|
||||||
}
|
return fmt.Errorf(
|
||||||
err = eventingClient.UpdateTrigger(b.Build())
|
"cannot update trigger '%s' because broker is immutable", name)
|
||||||
if err == nil {
|
}
|
||||||
|
if cmd.Flags().Changed("filter") {
|
||||||
|
updated, removed, err := triggerUpdateFlags.GetUpdateFilters()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf(
|
||||||
|
"cannot update trigger '%s' because %s", name, err)
|
||||||
|
}
|
||||||
|
existing := extractFilters(trigger)
|
||||||
|
b.Filters(existing.Merge(updated).Remove(removed))
|
||||||
|
}
|
||||||
|
if cmd.Flags().Changed("sink") {
|
||||||
|
destination, err := sinkFlags.ResolveSink(dynamicClient, namespace)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
b.Subscriber(&duckv1.Destination{
|
||||||
|
Ref: destination.Ref,
|
||||||
|
URI: destination.URI,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
err = eventingClient.UpdateTrigger(b.Build())
|
||||||
|
if err != nil {
|
||||||
|
if apierrors.IsConflict(err) && retries < MaxUpdateRetries {
|
||||||
|
retries++
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
fmt.Fprintf(cmd.OutOrStdout(), "Trigger '%s' updated in namespace '%s'.\n", name, namespace)
|
fmt.Fprintf(cmd.OutOrStdout(), "Trigger '%s' updated in namespace '%s'.\n", name, namespace)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
return err
|
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
commands.AddNamespaceFlags(cmd.Flags(), false)
|
commands.AddNamespaceFlags(cmd.Flags(), false)
|
||||||
triggerUpdateFlags.Add(cmd)
|
triggerUpdateFlags.Add(cmd)
|
||||||
sinkFlags.Add(cmd)
|
sinkFlags.Add(cmd)
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,126 @@
|
||||||
|
// Copyright 2020 The Knative Authors
|
||||||
|
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or im
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
// +build e2e
|
||||||
|
// +build !serving
|
||||||
|
|
||||||
|
package e2e
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"gotest.tools/assert"
|
||||||
|
"k8s.io/apimachinery/pkg/util/wait"
|
||||||
|
"knative.dev/client/pkg/util"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestBrokerTrigger(t *testing.T) {
|
||||||
|
t.Parallel()
|
||||||
|
test := NewE2eTest(t)
|
||||||
|
test.Setup(t)
|
||||||
|
defer test.Teardown(t)
|
||||||
|
|
||||||
|
err := test.lableNamespaceForDefaultBroker(t)
|
||||||
|
assert.NilError(t, err)
|
||||||
|
test.serviceCreate(t, "sinksvc0")
|
||||||
|
test.serviceCreate(t, "sinksvc1")
|
||||||
|
|
||||||
|
t.Run("create triggers and list them", func(t *testing.T) {
|
||||||
|
test.triggerCreate(t, "trigger1", []string{"a=b"}, "sinksvc0")
|
||||||
|
test.triggerCreate(t, "trigger2", []string{"type=knative.dev.bar", "source=cronjob"}, "sinksvc1")
|
||||||
|
test.verifyTriggerList(t, []string{"trigger1", "trigger2"})
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("create a trigger and delete it", func(t *testing.T) {
|
||||||
|
test.triggerCreate(t, "deltrigger", []string{"a=b"}, "sinksvc0")
|
||||||
|
test.triggerDelete(t, "deltrigger")
|
||||||
|
test.verifyTriggerNotfound(t, "deltrigger")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("create a trigger, describe and update it", func(t *testing.T) {
|
||||||
|
test.triggerCreate(t, "updtrigger", []string{"a=b"}, "sinksvc0")
|
||||||
|
test.verifyTriggerDescribe(t, "updtrigger", []string{"a", "b"}, "default", "sinksvc0")
|
||||||
|
test.triggerUpdate(t, "updtrigger", "type=knative.dev.bar", "sinksvc1")
|
||||||
|
test.verifyTriggerDescribe(t, "updtrigger", []string{"a", "b", "type", "knative.dev.bar"}, "default", "sinksvc1")
|
||||||
|
})
|
||||||
|
|
||||||
|
t.Run("create trigger with error return", func(t *testing.T) {
|
||||||
|
test.triggerCreateMissingSink(t, "errtrigger", "notfound")
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) lableNamespaceForDefaultBroker(t *testing.T) error {
|
||||||
|
kubectl := kubectl{t, Logger{}}
|
||||||
|
|
||||||
|
_, err := kubectl.RunWithOpts([]string{"label", "namespace", test.kn.namespace, "knative-eventing-injection=enabled"}, runOpts{})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Error executing 'kubectl label namespace %s knative-eventing-injection=enabled'. Error: %s", test.kn.namespace, err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
return wait.PollImmediate(10*time.Second, 5*time.Minute, func() (bool, error) {
|
||||||
|
out, err := kubectl.RunWithOpts([]string{"get", "broker", "-n", test.kn.namespace, "-o=jsonpath='{.items[0].status.conditions[?(@.type==\"Ready\")].status}'"}, runOpts{AllowError: true})
|
||||||
|
if err != nil {
|
||||||
|
return false, nil
|
||||||
|
} else {
|
||||||
|
return strings.Contains(out, "True"), nil
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) triggerCreate(t *testing.T, name string, filters []string, sinksvc string) {
|
||||||
|
args := []string{"trigger", "create", name, "--broker", "default", "--sink", "svc:" + sinksvc}
|
||||||
|
for _, v := range filters {
|
||||||
|
args = append(args, "--filter", v)
|
||||||
|
}
|
||||||
|
out, err := test.kn.RunWithOpts(args, runOpts{NoNamespace: false})
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, "Trigger", name, "created", "namespace", test.kn.namespace))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) triggerCreateMissingSink(t *testing.T, name string, sinksvc string) {
|
||||||
|
_, err := test.kn.RunWithOpts([]string{"trigger", "create", name, "--broker", "default", "--sink", "svc:" + sinksvc}, runOpts{NoNamespace: false, AllowError: true})
|
||||||
|
assert.ErrorContains(t, err, "services.serving.knative.dev", "not found")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) triggerDelete(t *testing.T, name string) {
|
||||||
|
out, err := test.kn.RunWithOpts([]string{"trigger", "delete", name}, runOpts{NoNamespace: false})
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, "Trigger", name, "deleted", "namespace", test.kn.namespace))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) triggerUpdate(t *testing.T, name string, filter string, sinksvc string) {
|
||||||
|
out, err := test.kn.RunWithOpts([]string{"trigger", "update", name, "--filter", filter, "--sink", "svc:" + sinksvc}, runOpts{NoNamespace: false})
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, "Trigger", name, "updated", "namespace", test.kn.namespace))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) verifyTriggerList(t *testing.T, triggers []string) {
|
||||||
|
out, err := test.kn.RunWithOpts([]string{"trigger", "list"}, runOpts{NoNamespace: false})
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, triggers...))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) verifyTriggerDescribe(t *testing.T, name string, filters []string, broker string, sink string) {
|
||||||
|
out, err := test.kn.RunWithOpts([]string{"trigger", "describe", name}, runOpts{NoNamespace: false})
|
||||||
|
assert.NilError(t, err)
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, filters...))
|
||||||
|
assert.Check(t, util.ContainsAllIgnoreCase(out, name, broker, sink))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (test *e2eTest) verifyTriggerNotfound(t *testing.T, name string) {
|
||||||
|
_, err := test.kn.RunWithOpts([]string{"trigger", "describe", name}, runOpts{NoNamespace: false, AllowError: true})
|
||||||
|
assert.ErrorContains(t, err, name, "not found")
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue