Skip to content

Commit

Permalink
chore: Add unit test for nv-ipam-cni objects rendering and sync
Browse files Browse the repository at this point in the history
Signed-off-by: Soule BA <[email protected]>
  • Loading branch information
souleb committed Jul 16, 2024
1 parent fdcb8a3 commit 40c19aa
Show file tree
Hide file tree
Showing 4 changed files with 443 additions and 36 deletions.
106 changes: 95 additions & 11 deletions pkg/state/common_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -25,18 +25,28 @@ import (

mellanoxv1alpha1 "github.com/Mellanox/network-operator/api/v1alpha1"
clustertype_mocks "github.com/Mellanox/network-operator/pkg/clustertype/mocks"
"github.com/Mellanox/network-operator/pkg/consts"
"github.com/Mellanox/network-operator/pkg/state"
"github.com/Mellanox/network-operator/pkg/staticconfig"
staticconfig_mocks "github.com/Mellanox/network-operator/pkg/staticconfig/mocks"

appsv1 "k8s.io/api/apps/v1"
corev1 "k8s.io/api/core/v1"
k8serrors "k8s.io/apimachinery/pkg/api/errors"
"k8s.io/apimachinery/pkg/api/resource"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/apimachinery/pkg/types"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/log"
)

const (
defaultTestRepository = "myRepo"
defaultTestImage = "myImage"
defaultTestVersion = "myVersion"
)

func getTestCatalog() state.InfoCatalog {
catalog := state.NewInfoCatalog()
clusterTypeProvider := clustertype_mocks.Provider{}
Expand Down Expand Up @@ -90,20 +100,39 @@ func assertCommonPodTemplateFields(template *corev1.PodTemplateSpec, image *mell
Expect(template.Spec.Containers[0].Resources.Requests).To(Equal(image.ContainerResources[0].Requests))
}

func assertCommonDeploymentFields(u *unstructured.Unstructured, image *mellanoxv1alpha1.ImageSpec) {
func assertCommonDeploymentFieldsFromUnstructured(u *unstructured.Unstructured, image *mellanoxv1alpha1.ImageSpec) {
d := &appsv1.Deployment{}
err := runtime.DefaultUnstructuredConverter.FromUnstructured(u.UnstructuredContent(), d)
Expect(err).ToNot(HaveOccurred())
assertCommonDeploymentFields(d, image)
}

func assertCommonDeploymentFields(d *appsv1.Deployment, image *mellanoxv1alpha1.ImageSpec) {
assertCommonPodTemplateFields(&d.Spec.Template, image)

Expect(d.Spec.Template.Spec.Tolerations).To(ContainElements(
corev1.Toleration{
Key: "nvidia.com/gpu",
Operator: "Exists",
Value: "",
Effect: "NoSchedule",
TolerationSeconds: nil,
},
))
}

func assertCommonDaemonSetFields(u *unstructured.Unstructured,
func assertCommonDaemonSetFieldsFromUnstructured(u *unstructured.Unstructured,
image *mellanoxv1alpha1.ImageSpec, policy *mellanoxv1alpha1.NicClusterPolicy) {
ds := &appsv1.DaemonSet{}
err := runtime.DefaultUnstructuredConverter.FromUnstructured(u.UnstructuredContent(), ds)
Expect(err).ToNot(HaveOccurred())
assertCommonDaemonSetFields(ds, image, policy)
}

func assertCommonDaemonSetFields(ds *appsv1.DaemonSet, image *mellanoxv1alpha1.ImageSpec,
policy *mellanoxv1alpha1.NicClusterPolicy) {
assertCommonPodTemplateFields(&ds.Spec.Template, image)
// Tolerations

Expect(ds.Spec.Template.Spec.Tolerations).To(ContainElements(
corev1.Toleration{Key: "first-taint"},
corev1.Toleration{
Expand All @@ -115,15 +144,14 @@ func assertCommonDaemonSetFields(u *unstructured.Unstructured,
},
))

// NodeAffinity
Expect(ds.Spec.Template.Spec.Affinity.NodeAffinity).To(Equal(policy.Spec.NodeAffinity))
}

func getTestImageSpec() *mellanoxv1alpha1.ImageSpec {
return &mellanoxv1alpha1.ImageSpec{
Image: "image-one",
Repository: "repository",
Version: "five",
Image: defaultTestImage,
Repository: defaultTestRepository,
Version: defaultTestVersion,
ImagePullSecrets: []string{"secret-one", "secret-two"},
}
}
Expand All @@ -148,10 +176,14 @@ func isNamespaced(obj *unstructured.Unstructured) bool {
obj.GetKind() != "ValidatingWebhookConfiguration"
}

func assertCNIBinDirForDS(u *unstructured.Unstructured) {
func assertCNIBinDirForDSFromUnstructured(u *unstructured.Unstructured) {
ds := &appsv1.DaemonSet{}
err := runtime.DefaultUnstructuredConverter.FromUnstructured(u.UnstructuredContent(), ds)
Expect(err).ToNot(HaveOccurred())
assertCNIBinDirForDS(ds)
}

func assertCNIBinDirForDS(ds *appsv1.DaemonSet) {
for i := range ds.Spec.Template.Spec.Volumes {
vol := ds.Spec.Template.Spec.Volumes[i]
if vol.Name == "cnibin" {
Expand All @@ -171,10 +203,10 @@ func GetManifestObjectsTest(ctx context.Context, cr *mellanoxv1alpha1.NicCluster
}
switch got[i].GetKind() {
case "DaemonSet":
assertCommonDaemonSetFields(got[i], imageSpec, cr)
assertCNIBinDirForDS(got[i])
assertCommonDaemonSetFieldsFromUnstructured(got[i], imageSpec, cr)
assertCNIBinDirForDSFromUnstructured(got[i])
case "Deployment":
assertCommonDeploymentFields(got[i], imageSpec)
assertCommonDeploymentFieldsFromUnstructured(got[i], imageSpec)
}
}
}
Expand All @@ -201,3 +233,55 @@ func getTestClusterPolicyWithBaseFields() *mellanoxv1alpha1.NicClusterPolicy {
},
}
}

func getKindState(ctx context.Context, c client.Client, objs []*unstructured.Unstructured,
targetKind string) (state.SyncState, error) {
reqLogger := log.FromContext(ctx)
reqLogger.V(consts.LogLevelInfo).Info("Checking related object states")
for _, obj := range objs {
if obj.GetKind() != targetKind {
continue
}
found := obj.DeepCopy()
err := c.Get(
ctx, types.NamespacedName{Name: found.GetName(), Namespace: found.GetNamespace()}, found)
if err != nil {
if k8serrors.IsNotFound(err) {
return state.SyncStateNotReady, nil
}
return state.SyncStateNotReady, fmt.Errorf("failed to get object: %w", err)
}

buf, err := found.MarshalJSON()
if err != nil {
return state.SyncStateNotReady, fmt.Errorf("failed to marshall unstructured daemonset object: %w", err)
}

switch obj.GetKind() {
case "DaemonSet":
ds := &appsv1.DaemonSet{}
if err = json.Unmarshal(buf, ds); err != nil {
return state.SyncStateNotReady, fmt.Errorf("failed to unmarshall to daemonset object: %w", err)
}
if ds.Status.DesiredNumberScheduled != 0 && ds.Status.DesiredNumberScheduled == ds.Status.NumberAvailable &&
ds.Status.UpdatedNumberScheduled == ds.Status.NumberAvailable {
return state.SyncStateReady, nil
}
return state.SyncStateNotReady, nil
case "Deployment":
d := &appsv1.Deployment{}
if err = json.Unmarshal(buf, d); err != nil {
return state.SyncStateNotReady, fmt.Errorf("failed to unmarshall to deployment object: %w", err)
}

if d.Status.ObservedGeneration > 0 && d.Status.Replicas == d.Status.ReadyReplicas &&
d.Status.UpdatedReplicas == d.Status.AvailableReplicas {
return state.SyncStateReady, nil
}
return state.SyncStateNotReady, nil
default:
return state.SyncStateNotReady, fmt.Errorf("unsupported target kind")
}
}
return state.SyncStateNotReady, fmt.Errorf("objects list does not contain the specified target kind")
}
25 changes: 17 additions & 8 deletions pkg/state/info_source.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,31 +45,37 @@ func NewInfoCatalog() InfoCatalog {
// InfoSource represents an object that is a source of information.
type InfoSource interface{}

// InfoCatalog is an information catalog to be used to retrieve infoSources. used for State implementation that require
// additional helping functionality to perfrom the Sync operation. As more States are added,
// more infoSources may be added to aid them. for any infoSource if not present in the catalog, nil will be returned.
// InfoCatalog is an information catalog to be used to retrieve infoSources.
// It is used for State implementations that require additional helping functionality
// to perform the Sync operation. As more States are added, more infoSources may be
// added to aid them. For any infoSource, if not present in the catalog, nil will be returned.
type InfoCatalog interface {
// Add an infoSource of InfoType to the catalog
Add(InfoType, InfoSource)
// GetNodeInfoProvider returns a reference nodeinfo.Provider from catalog or nil if provider does not exist
// GetNodeInfoProvider returns a reference nodeinfo.Provider from catalog or
// nil if provider does not exist
GetNodeInfoProvider() nodeinfo.Provider
// GetClusterTypeProvider returns a reference clustertype.Provider from catalog or nil if provider does not exist
// GetClusterTypeProvider returns a reference clustertype.Provider from catalog
// or nil if provider does not exist
GetClusterTypeProvider() clustertype.Provider
// GetStaticConfigProvider returns a reference staticinfo.Provider from catalog or nil if provider does not exist
GetStaticConfigProvider() staticconfig.Provider
// GetDocaDriverImageProvider returns a reference docadriverimages.Provider from catalog
// GetStaticConfigProvider returns a reference staticinfo.Provider from catalog
// or nil if provider does not exist
GetStaticConfigProvider() staticconfig.Provider
// GetDocaDriverImageProvider returns a reference docadriverimages.Provider
// from catalog or nil if provider does not exist
GetDocaDriverImageProvider() docadriverimages.Provider
}

type infoCatalog struct {
infoSources map[InfoType]InfoSource
}

// Add an infoSource to the catalog.
func (sc *infoCatalog) Add(infoType InfoType, infoSource InfoSource) {
sc.infoSources[infoType] = infoSource
}

// GetNodeInfoProvider returns a nodeinfo.Provider or nil.
func (sc *infoCatalog) GetNodeInfoProvider() nodeinfo.Provider {
infoSource, ok := sc.infoSources[InfoTypeNodeInfo]
if !ok {
Expand All @@ -78,6 +84,7 @@ func (sc *infoCatalog) GetNodeInfoProvider() nodeinfo.Provider {
return infoSource.(nodeinfo.Provider)
}

// GetClusterTypeProvider returns a clustertype.Provider or nil.
func (sc *infoCatalog) GetClusterTypeProvider() clustertype.Provider {
infoSource, ok := sc.infoSources[InfoTypeClusterType]
if !ok {
Expand All @@ -86,6 +93,7 @@ func (sc *infoCatalog) GetClusterTypeProvider() clustertype.Provider {
return infoSource.(clustertype.Provider)
}

// GetStaticConfigProvider returns a staticconfig.Provider or nil.
func (sc *infoCatalog) GetStaticConfigProvider() staticconfig.Provider {
infoSource, ok := sc.infoSources[InfoTypeStaticConfig]
if !ok {
Expand All @@ -94,6 +102,7 @@ func (sc *infoCatalog) GetStaticConfigProvider() staticconfig.Provider {
return infoSource.(staticconfig.Provider)
}

// GetDocaDriverImageProvider returns a docadriverimages.Provider or nil.
func (sc *infoCatalog) GetDocaDriverImageProvider() docadriverimages.Provider {
infoSource, ok := sc.infoSources[InfoTypeDocaDriverImage]
if !ok {
Expand Down
2 changes: 1 addition & 1 deletion pkg/state/state_nv_ipam_cni.go
Original file line number Diff line number Diff line change
Expand Up @@ -71,7 +71,7 @@ type NVIPAMManifestRenderData struct {
//
//nolint:dupl
func (s *stateNVIPAMCNI) Sync(
ctx context.Context, customResource interface{}, infoCatalog InfoCatalog) (SyncState, error) {
ctx context.Context, customResource any, infoCatalog InfoCatalog) (SyncState, error) {
reqLogger := log.FromContext(ctx)
cr := customResource.(*mellanoxv1alpha1.NicClusterPolicy)
reqLogger.V(consts.LogLevelInfo).Info(
Expand Down
Loading

0 comments on commit 40c19aa

Please sign in to comment.