Files
open-cluster-management/pkg/server/services/work/work_test.go
Wei Liu f1e7905b16
Some checks failed
Scorecard supply-chain security / Scorecard analysis (push) Failing after 32s
Post / coverage (push) Failing after 25s
Post / images (amd64, addon-manager) (push) Failing after 30s
Post / images (amd64, placement) (push) Failing after 15s
Post / images (amd64, registration) (push) Failing after 20s
Post / images (amd64, registration-operator) (push) Failing after 27s
Post / images (amd64, work) (push) Failing after 14s
Post / images (arm64, addon-manager) (push) Failing after 15s
Post / images (arm64, placement) (push) Failing after 19s
Post / images (arm64, registration) (push) Failing after 18s
Post / images (arm64, registration-operator) (push) Failing after 19s
Post / images (arm64, work) (push) Failing after 17s
Post / image manifest (addon-manager) (push) Has been skipped
Post / image manifest (placement) (push) Has been skipped
Post / image manifest (registration) (push) Has been skipped
Post / image manifest (registration-operator) (push) Has been skipped
Post / image manifest (work) (push) Has been skipped
Post / trigger clusteradm e2e (push) Has been skipped
Close stale issues and PRs / stale (push) Failing after 34s
using mw finalizer instead of resource finalizer (#1211)
Signed-off-by: Wei Liu <liuweixa@redhat.com>
2025-10-16 09:30:26 +00:00

402 lines
12 KiB
Go

package work
import (
"context"
"fmt"
"testing"
"time"
cloudevents "github.com/cloudevents/sdk-go/v2"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
clienttesting "k8s.io/client-go/testing"
workfake "open-cluster-management.io/api/client/work/clientset/versioned/fake"
workinformers "open-cluster-management.io/api/client/work/informers/externalversions"
workv1 "open-cluster-management.io/api/work/v1"
"open-cluster-management.io/sdk-go/pkg/cloudevents/clients/common"
"open-cluster-management.io/sdk-go/pkg/cloudevents/clients/work/payload"
"open-cluster-management.io/sdk-go/pkg/cloudevents/generic/types"
testingcommon "open-cluster-management.io/ocm/pkg/common/testing"
)
func TestGet(t *testing.T) {
cases := []struct {
name string
works []runtime.Object
resourceID string
expectedError bool
}{
{
name: "work not found",
works: []runtime.Object{},
resourceID: "test-namespace/test-work",
expectedError: true,
},
{
name: "get work",
resourceID: "test-namespace/test-work",
works: []runtime.Object{&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
Name: "test-work",
Namespace: "test-namespace",
ResourceVersion: "1",
},
}},
},
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
workClient := workfake.NewSimpleClientset(c.works...)
workInformers := workinformers.NewSharedInformerFactory(workClient, 10*time.Minute)
workInformer := workInformers.Work().V1().ManifestWorks()
for _, obj := range c.works {
if err := workInformer.Informer().GetStore().Add(obj); err != nil {
t.Fatal(err)
}
}
service := NewWorkService(workClient, workInformer)
_, err := service.Get(context.Background(), c.resourceID)
if c.expectedError {
if err == nil {
t.Errorf("expected error, got nil")
}
return
}
if err != nil {
t.Errorf("unexpected error: %v", err)
}
})
}
}
func TestList(t *testing.T) {
cases := []struct {
name string
works []runtime.Object
clusterName string
expectedWorks int
}{
{
name: "no works",
works: []runtime.Object{},
clusterName: "test-cluster",
expectedWorks: 0,
},
{
name: "list works",
works: []runtime.Object{
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
Name: "test-work1",
Namespace: "test-cluster1",
ResourceVersion: "1",
},
},
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
Name: "test-work2",
Namespace: "test-cluster2",
ResourceVersion: "1",
},
},
},
clusterName: "test-cluster1",
expectedWorks: 1,
},
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
workClient := workfake.NewSimpleClientset(c.works...)
workInformers := workinformers.NewSharedInformerFactory(workClient, 10*time.Minute)
workInformer := workInformers.Work().V1().ManifestWorks()
for _, obj := range c.works {
if err := workInformer.Informer().GetStore().Add(obj); err != nil {
t.Fatal(err)
}
}
service := NewWorkService(workClient, workInformer)
evts, err := service.List(types.ListOptions{ClusterName: c.clusterName})
if err != nil {
t.Errorf("unexpected error: %v", err)
}
if len(evts) != c.expectedWorks {
t.Errorf("expected %d works, got %d", c.expectedWorks, len(evts))
}
})
}
}
func TestHandleStatusUpdate(t *testing.T) {
cases := []struct {
name string
works []runtime.Object
workEvt *cloudevents.Event
validateActions func(t *testing.T, actions []clienttesting.Action)
expectedError bool
}{
{
name: "invalid event type",
works: []runtime.Object{},
workEvt: func() *cloudevents.Event {
evt := types.NewEventBuilder("test", types.CloudEventsType{}).NewEvent()
return &evt
}(),
expectedError: true,
},
{
name: "invalid action",
works: []runtime.Object{
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
UID: "test-cluster/test-work",
Name: "test-work",
Namespace: "test-cluster",
ResourceVersion: "1",
},
},
},
workEvt: func() *cloudevents.Event {
evt := types.NewEventBuilder("test", types.CloudEventsType{
CloudEventsDataType: payload.ManifestBundleEventDataType,
SubResource: types.SubResourceSpec,
Action: types.CreateRequestAction,
}).WithResourceVersion(1).
WithClusterName("test-cluster").
WithResourceID("test-cluster/test-work").
WithStatusUpdateSequenceID("1").NewEvent()
manifestBundleStatus := &payload.ManifestBundleStatus{
Conditions: []metav1.Condition{
{
Type: "Test",
Status: metav1.ConditionTrue,
},
},
}
evt.SetData(cloudevents.ApplicationJSON, manifestBundleStatus)
return &evt
}(),
expectedError: true,
},
{
name: "update work status",
works: []runtime.Object{
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
UID: "test-cluster/test-work",
Name: "test-work",
Namespace: "test-cluster",
Generation: 1,
},
},
},
workEvt: func() *cloudevents.Event {
evt := types.NewEventBuilder("test", types.CloudEventsType{
CloudEventsDataType: payload.ManifestBundleEventDataType,
SubResource: types.SubResourceStatus,
Action: types.UpdateRequestAction,
}).WithResourceVersion(1).
WithClusterName("test-cluster").
WithResourceID("test-cluster/test-work").
WithStatusUpdateSequenceID("1").NewEvent()
manifestBundleStatus := &payload.ManifestBundleStatus{
Conditions: []metav1.Condition{
{
Type: "Test",
Status: metav1.ConditionTrue,
},
},
}
evt.SetData(cloudevents.ApplicationJSON, manifestBundleStatus)
return &evt
}(),
validateActions: func(t *testing.T, actions []clienttesting.Action) {
testingcommon.AssertActions(t, actions, "patch", "patch")
if len(actions[0].GetSubresource()) != 0 {
// add finalizer
t.Errorf("unexpected subresource %s", actions[0].GetSubresource())
}
if actions[1].GetSubresource() != "status" {
// update status
t.Errorf("expected subresource %s, got %s", "status", actions[1].GetSubresource())
}
},
},
{
name: "update work status (resource version mismatch)",
works: []runtime.Object{
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
UID: "test-cluster/test-work",
Name: "test-work",
Namespace: "test-cluster",
Generation: 2,
},
},
},
workEvt: func() *cloudevents.Event {
evt := types.NewEventBuilder("test", types.CloudEventsType{
CloudEventsDataType: payload.ManifestBundleEventDataType,
SubResource: types.SubResourceStatus,
Action: types.UpdateRequestAction,
}).WithResourceVersion(1).
WithClusterName("test-cluster").
WithResourceID("test-cluster/test-work").
WithStatusUpdateSequenceID("1").NewEvent()
manifestBundleStatus := &payload.ManifestBundleStatus{
Conditions: []metav1.Condition{
{
Type: "Test",
Status: metav1.ConditionTrue,
},
},
}
evt.SetData(cloudevents.ApplicationJSON, manifestBundleStatus)
return &evt
}(),
validateActions: func(t *testing.T, actions []clienttesting.Action) {
if len(actions) != 0 {
t.Errorf("expected 0 actions, got %d", len(actions))
}
},
},
{
name: "update work status (deleted)",
works: []runtime.Object{
&workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{
UID: "test-cluster/test-work",
Name: "test-work",
Namespace: "test-cluster",
Generation: 1,
Finalizers: []string{workv1.ManifestWorkFinalizer},
},
},
},
workEvt: func() *cloudevents.Event {
evt := types.NewEventBuilder("test", types.CloudEventsType{
CloudEventsDataType: payload.ManifestBundleEventDataType,
SubResource: types.SubResourceStatus,
Action: types.UpdateRequestAction,
}).WithResourceVersion(1).
WithClusterName("test-cluster").
WithResourceID("test-cluster/test-work").
WithStatusUpdateSequenceID("1").NewEvent()
manifestBundleStatus := &payload.ManifestBundleStatus{
Conditions: []metav1.Condition{
{
Type: common.ResourceDeleted,
Status: metav1.ConditionTrue,
},
},
}
evt.SetData(cloudevents.ApplicationJSON, manifestBundleStatus)
return &evt
}(),
validateActions: func(t *testing.T, actions []clienttesting.Action) {
testingcommon.AssertActions(t, actions, "patch")
if len(actions[0].GetSubresource()) != 0 {
// remove finalizer
t.Errorf("unexpected subresource %s", actions[0].GetSubresource())
}
},
},
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
workClient := workfake.NewSimpleClientset(c.works...)
workInformers := workinformers.NewSharedInformerFactory(workClient, 10*time.Minute)
workInformer := workInformers.Work().V1().ManifestWorks()
for _, obj := range c.works {
if err := workInformer.Informer().GetStore().Add(obj); err != nil {
t.Fatal(err)
}
}
service := NewWorkService(workClient, workInformer)
err := service.HandleStatusUpdate(context.Background(), c.workEvt)
if c.expectedError {
if err == nil {
t.Errorf("expected error, got nil")
}
return
}
if err != nil {
t.Errorf("unexpected error: %v", err)
}
c.validateActions(t, workClient.Actions())
})
}
}
func TestEventHandlerFuncs(t *testing.T) {
handler := &workHandler{}
service := &WorkService{}
eventHandlerFuncs := service.EventHandlerFuncs(handler)
work := &workv1.ManifestWork{
ObjectMeta: metav1.ObjectMeta{Name: "test-work", Namespace: "test-namespace"},
}
eventHandlerFuncs.AddFunc(work)
if !handler.onCreateCalled {
t.Errorf("onCreate not called")
}
eventHandlerFuncs.UpdateFunc(nil, work)
if !handler.onUpdateCalled {
t.Errorf("onUpdate not called")
}
eventHandlerFuncs.DeleteFunc(work)
if !handler.onDeleteCalled {
t.Errorf("onDelete not called")
}
}
type workHandler struct {
onCreateCalled bool
onUpdateCalled bool
onDeleteCalled bool
}
func (m *workHandler) OnCreate(ctx context.Context, t types.CloudEventsDataType, resourceID string) error {
if t != payload.ManifestBundleEventDataType {
return fmt.Errorf("expected %v, got %v", payload.ManifestBundleEventDataType, t)
}
if resourceID != "test-namespace/test-work" {
return fmt.Errorf("expected %v, got %v", "test-namespace/test-work", resourceID)
}
m.onCreateCalled = true
return nil
}
func (m *workHandler) OnUpdate(ctx context.Context, t types.CloudEventsDataType, resourceID string) error {
if t != payload.ManifestBundleEventDataType {
return fmt.Errorf("expected %v, got %v", payload.ManifestBundleEventDataType, t)
}
if resourceID != "test-namespace/test-work" {
return fmt.Errorf("expected %v, got %v", "test-namespace/test-work", resourceID)
}
m.onUpdateCalled = true
return nil
}
func (m *workHandler) OnDelete(ctx context.Context, t types.CloudEventsDataType, resourceID string) error {
if t != payload.ManifestBundleEventDataType {
return fmt.Errorf("expected %v, got %v", payload.ManifestBundleEventDataType, t)
}
if resourceID != "test-namespace/test-work" {
return fmt.Errorf("expected %v, got %v", "test-namespace/test-work", resourceID)
}
m.onDeleteCalled = true
return nil
}