mirror of
https://github.com/kubevirt/containerized-data-importer.git
synced 2025-06-03 06:30:22 +00:00

* Run `make deps-update` Signed-off-by: Maya Rashish <mrashish@redhat.com> * Update to ginkgo v2 Avoid using table extension to avoid compilation errors Switch to v2 everywhere Update qe-tools as well (required) Signed-off-by: Maya Rashish <mrashish@redhat.com> * Fix/avoid deprecation warnings Signed-off-by: Maya Rashish <mrashish@redhat.com> * Do not use v1 reporter For unit tests: stop using custom reporter, unnecessary For functional tests: borrow code from kubevirt to keep reporting Avoid deprecated warnings by golangci for using deprecated reporter Signed-off-by: Maya Rashish <mrashish@redhat.com> * Increase ginkgo timeout to 24h (default in ginkgo v1) this may seem excessive, but we have a lower timeout in Prow, let's save ourselves the future trouble of bumping timeouts in two places. Signed-off-by: Maya Rashish <mrashish@redhat.com> * use the ginkgo built-in junit reporter Signed-off-by: Maya Rashish <mrashish@redhat.com> * Avoid using deprecated --ginkgo.noColor, use --ginkgo.no-color instead Signed-off-by: Maya Rashish <mrashish@redhat.com> --------- Signed-off-by: Maya Rashish <mrashish@redhat.com>
164 lines
5.6 KiB
Go
164 lines
5.6 KiB
Go
/*
|
|
Copyright 2022 The CDI 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 controller
|
|
|
|
import (
|
|
"context"
|
|
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
|
|
snapshotv1 "github.com/kubernetes-csi/external-snapshotter/client/v6/apis/volumesnapshot/v1"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/client-go/kubernetes/scheme"
|
|
"k8s.io/utils/pointer"
|
|
cdiv1 "kubevirt.io/containerized-data-importer-api/pkg/apis/core/v1beta1"
|
|
. "kubevirt.io/containerized-data-importer/pkg/controller/common"
|
|
"sigs.k8s.io/controller-runtime/pkg/client/fake"
|
|
"sigs.k8s.io/controller-runtime/pkg/reconcile"
|
|
)
|
|
|
|
const (
|
|
dsName = "test-datasource"
|
|
pvcName = "test-pvc"
|
|
snapshotName = "test-snapshot"
|
|
)
|
|
|
|
var _ = Describe("All DataSource Tests", func() {
|
|
var _ = Describe("DataSource controller reconcile loop", func() {
|
|
var (
|
|
reconciler *DataSourceReconciler
|
|
ds *cdiv1.DataSource
|
|
dsKey = types.NamespacedName{Name: dsName, Namespace: metav1.NamespaceDefault}
|
|
dsReq = reconcile.Request{NamespacedName: dsKey}
|
|
)
|
|
|
|
// verifyConditions reconciles, gets DataSource, and verifies its status conditions
|
|
var verifyConditions = func(step string, isReady bool, reasonReady string) {
|
|
By(step)
|
|
_, err := reconciler.Reconcile(context.TODO(), dsReq)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
err = reconciler.client.Get(context.TODO(), dsKey, ds)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
dsCond := FindDataSourceConditionByType(ds, cdiv1.DataSourceReady)
|
|
Expect(dsCond).ToNot(BeNil())
|
|
verifyConditionState(string(cdiv1.DataSourceReady), dsCond.ConditionState, isReady, reasonReady)
|
|
}
|
|
|
|
It("Should do nothing and return nil when no DataSource exists", func() {
|
|
reconciler = createDataSourceReconciler()
|
|
_, err := reconciler.Reconcile(context.TODO(), dsReq)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
})
|
|
|
|
It("Should update Ready condition when DataSource has no source", func() {
|
|
ds = createDataSource()
|
|
reconciler = createDataSourceReconciler(ds)
|
|
verifyConditions("No source", false, noSource)
|
|
})
|
|
|
|
It("Should update Ready condition when DataSource has source pvc", func() {
|
|
ds = createDataSource()
|
|
ds.Spec.Source.PVC = &cdiv1.DataVolumeSourcePVC{Namespace: metav1.NamespaceDefault, Name: pvcName}
|
|
reconciler = createDataSourceReconciler(ds)
|
|
verifyConditions("Source DV does not exist", false, NotFound)
|
|
|
|
dv := NewImportDataVolume(pvcName)
|
|
err := reconciler.client.Create(context.TODO(), dv)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
|
|
dv.Status.Phase = cdiv1.ImportInProgress
|
|
err = reconciler.client.Update(context.TODO(), dv)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source DV ImportInProgress", false, string(dv.Status.Phase))
|
|
|
|
dv.Status.Phase = cdiv1.Succeeded
|
|
err = reconciler.client.Update(context.TODO(), dv)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source DV Succeeded", true, ready)
|
|
|
|
err = reconciler.client.Delete(context.TODO(), dv)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source DV Deleted", false, NotFound)
|
|
|
|
pvc := CreatePvc(pvcName, metav1.NamespaceDefault, nil, nil)
|
|
err = reconciler.client.Create(context.TODO(), pvc)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source PVC exists, but no DV", true, ready)
|
|
|
|
err = reconciler.client.Delete(context.TODO(), pvc)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source PVC Deleted", false, NotFound)
|
|
})
|
|
|
|
It("Should update Ready condition when DataSource has source snapshot", func() {
|
|
ds = createDataSource()
|
|
ds.Spec.Source.Snapshot = &cdiv1.DataVolumeSourceSnapshot{Namespace: metav1.NamespaceDefault, Name: snapshotName}
|
|
reconciler = createDataSourceReconciler(ds)
|
|
verifyConditions("Source snapshot does not exist", false, NotFound)
|
|
|
|
snap := &snapshotv1.VolumeSnapshot{
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: snapshotName,
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
Spec: snapshotv1.VolumeSnapshotSpec{},
|
|
Status: &snapshotv1.VolumeSnapshotStatus{
|
|
ReadyToUse: pointer.Bool(false),
|
|
},
|
|
}
|
|
err := reconciler.client.Create(context.TODO(), snap)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source snapshot not ready", false, "SnapshotNotReady")
|
|
|
|
snap.Status.ReadyToUse = pointer.Bool(true)
|
|
err = reconciler.client.Update(context.TODO(), snap)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source snapshot ready", true, ready)
|
|
|
|
err = reconciler.client.Delete(context.TODO(), snap)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
verifyConditions("Source snapshot Deleted", false, NotFound)
|
|
})
|
|
})
|
|
})
|
|
|
|
func createDataSourceReconciler(objects ...runtime.Object) *DataSourceReconciler {
|
|
s := scheme.Scheme
|
|
_ = cdiv1.AddToScheme(s)
|
|
_ = snapshotv1.AddToScheme(s)
|
|
cl := fake.NewClientBuilder().WithScheme(s).WithRuntimeObjects(objects...).Build()
|
|
r := &DataSourceReconciler{
|
|
client: cl,
|
|
scheme: s,
|
|
log: cronLog,
|
|
}
|
|
return r
|
|
}
|
|
|
|
func createDataSource() *cdiv1.DataSource {
|
|
return &cdiv1.DataSource{
|
|
TypeMeta: metav1.TypeMeta{APIVersion: cdiv1.SchemeGroupVersion.String()},
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
Name: dsName,
|
|
Namespace: metav1.NamespaceDefault,
|
|
},
|
|
}
|
|
}
|