216 lines
6.3 KiB
Go
216 lines
6.3 KiB
Go
/*
|
|
Copyright 2023 The Karmada 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 implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package status
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
corev1 "k8s.io/api/core/v1"
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
dynamicfake "k8s.io/client-go/dynamic/fake"
|
|
"k8s.io/client-go/kubernetes/scheme"
|
|
"k8s.io/client-go/tools/record"
|
|
controllerruntime "sigs.k8s.io/controller-runtime"
|
|
"sigs.k8s.io/controller-runtime/pkg/client/fake"
|
|
|
|
workv1alpha2 "github.com/karmada-io/karmada/pkg/apis/work/v1alpha2"
|
|
"github.com/karmada-io/karmada/pkg/resourceinterpreter/default/native"
|
|
"github.com/karmada-io/karmada/pkg/util/fedinformer/genericmanager"
|
|
"github.com/karmada-io/karmada/pkg/util/gclient"
|
|
)
|
|
|
|
func generateCRBStatusController() *CRBStatusController {
|
|
stopCh := make(chan struct{})
|
|
defer close(stopCh)
|
|
dynamicClient := dynamicfake.NewSimpleDynamicClient(scheme.Scheme,
|
|
&corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: "ns1"}})
|
|
m := genericmanager.NewSingleClusterInformerManager(dynamicClient, 0, stopCh)
|
|
m.Lister(corev1.SchemeGroupVersion.WithResource("namespaces"))
|
|
m.Start()
|
|
m.WaitForCacheSync()
|
|
|
|
c := &CRBStatusController{
|
|
Client: fake.NewClientBuilder().WithScheme(gclient.NewSchema()).Build(),
|
|
DynamicClient: dynamicClient,
|
|
InformerManager: m,
|
|
RESTMapper: func() meta.RESTMapper {
|
|
m := meta.NewDefaultRESTMapper([]schema.GroupVersion{corev1.SchemeGroupVersion})
|
|
m.Add(corev1.SchemeGroupVersion.WithKind("Namespace"), meta.RESTScopeNamespace)
|
|
return m
|
|
}(),
|
|
EventRecorder: &record.FakeRecorder{},
|
|
}
|
|
return c
|
|
}
|
|
|
|
func TestCRBStatusController_Reconcile(t *testing.T) {
|
|
preTime := metav1.Date(2023, 0, 0, 0, 0, 0, 0, time.UTC)
|
|
tests := []struct {
|
|
name string
|
|
binding *workv1alpha2.ClusterResourceBinding
|
|
expectRes controllerruntime.Result
|
|
expectError bool
|
|
}{
|
|
{
|
|
name: "failed in syncBindingStatus",
|
|
binding: &workv1alpha2.ClusterResourceBinding{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "binding",
|
|
},
|
|
Spec: workv1alpha2.ResourceBindingSpec{
|
|
Resource: workv1alpha2.ObjectReference{
|
|
APIVersion: "v1",
|
|
Kind: "Namespace",
|
|
Name: "ns",
|
|
},
|
|
},
|
|
},
|
|
expectRes: controllerruntime.Result{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "binding not found in client",
|
|
expectRes: controllerruntime.Result{},
|
|
expectError: false,
|
|
},
|
|
{
|
|
name: "failed in syncBindingStatus",
|
|
binding: &workv1alpha2.ClusterResourceBinding{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "binding",
|
|
// finalizers field is required when deletionTimestamp is defined, otherwise will encounter the
|
|
// error: `refusing to create obj binding with metadata.deletionTimestamp but no finalizers`.
|
|
Finalizers: []string{"test"},
|
|
DeletionTimestamp: &preTime,
|
|
},
|
|
Spec: workv1alpha2.ResourceBindingSpec{
|
|
Resource: workv1alpha2.ObjectReference{
|
|
APIVersion: "v1",
|
|
Kind: "Namespace",
|
|
Name: "ns",
|
|
},
|
|
},
|
|
},
|
|
expectRes: controllerruntime.Result{},
|
|
expectError: false,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
c := generateCRBStatusController()
|
|
c.ResourceInterpreter = FakeResourceInterpreter{DefaultInterpreter: native.NewDefaultInterpreter()}
|
|
|
|
// Prepare req
|
|
req := controllerruntime.Request{
|
|
NamespacedName: types.NamespacedName{
|
|
Name: "binding",
|
|
},
|
|
}
|
|
|
|
// Prepare binding and create it in client
|
|
if tt.binding != nil {
|
|
c.Client = fake.NewClientBuilder().WithScheme(gclient.NewSchema()).WithObjects(tt.binding).WithStatusSubresource(tt.binding).Build()
|
|
}
|
|
|
|
res, err := c.Reconcile(context.Background(), req)
|
|
assert.Equal(t, tt.expectRes, res)
|
|
if tt.expectError {
|
|
assert.Error(t, err)
|
|
} else {
|
|
assert.NoError(t, err)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestCRBStatusController_syncBindingStatus(t *testing.T) {
|
|
tests := []struct {
|
|
name string
|
|
resource workv1alpha2.ObjectReference
|
|
nsNameInDynamicClient string
|
|
resourceExistInClient bool
|
|
expectedError bool
|
|
}{
|
|
{
|
|
name: "failed in FetchResourceTemplate, err is NotFound",
|
|
resource: workv1alpha2.ObjectReference{
|
|
APIVersion: "v1",
|
|
Kind: "Namespace",
|
|
Name: "ns",
|
|
},
|
|
nsNameInDynamicClient: "ns1",
|
|
resourceExistInClient: true,
|
|
expectedError: false,
|
|
},
|
|
{
|
|
name: "failed in FetchResourceTemplate, err is not NotFound",
|
|
resource: workv1alpha2.ObjectReference{},
|
|
nsNameInDynamicClient: "ns",
|
|
resourceExistInClient: true,
|
|
expectedError: true,
|
|
},
|
|
{
|
|
name: "failed in AggregateClusterResourceBindingWorkStatus",
|
|
resource: workv1alpha2.ObjectReference{
|
|
APIVersion: "v1",
|
|
Kind: "Namespace",
|
|
Name: "ns",
|
|
},
|
|
nsNameInDynamicClient: "ns",
|
|
resourceExistInClient: false,
|
|
expectedError: true,
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
c := generateCRBStatusController()
|
|
c.DynamicClient = dynamicfake.NewSimpleDynamicClient(scheme.Scheme,
|
|
&corev1.Namespace{ObjectMeta: metav1.ObjectMeta{Name: tt.nsNameInDynamicClient}})
|
|
c.ResourceInterpreter = FakeResourceInterpreter{DefaultInterpreter: native.NewDefaultInterpreter()}
|
|
|
|
binding := &workv1alpha2.ClusterResourceBinding{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: "binding",
|
|
},
|
|
Spec: workv1alpha2.ResourceBindingSpec{
|
|
Resource: tt.resource,
|
|
},
|
|
}
|
|
|
|
if tt.resourceExistInClient {
|
|
c.Client = fake.NewClientBuilder().WithScheme(gclient.NewSchema()).WithObjects(binding).WithStatusSubresource(binding).Build()
|
|
}
|
|
|
|
err := c.syncBindingStatus(context.Background(), binding)
|
|
|
|
if tt.expectedError {
|
|
assert.Error(t, err)
|
|
} else {
|
|
assert.NoError(t, err)
|
|
}
|
|
})
|
|
}
|
|
}
|