Skip to content

Commit

Permalink
add check for tenant existence
Browse files Browse the repository at this point in the history
  • Loading branch information
Hoanganh.Mai committed Sep 30, 2024
1 parent 110cbb7 commit 21bd119
Show file tree
Hide file tree
Showing 12 changed files with 235 additions and 31 deletions.
2 changes: 1 addition & 1 deletion internal/controller/expected_netboxmock_calls_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -234,7 +234,7 @@ func mockTenancyTenancyTenantsList(tenancyMock *mock_interfaces.MockTenancyInter
}

// -----------------------------
// Restet Mock Functions
// Reset Mock Functions
// -----------------------------

func resetMockFunctions(ipamMockA *mock_interfaces.MockIpamInterface, ipamMockB *mock_interfaces.MockIpamInterface, tenancyMock *mock_interfaces.MockTenancyInterface) {
Expand Down
20 changes: 10 additions & 10 deletions internal/controller/ipaddress_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -78,7 +78,7 @@ func (r *IpAddressReconciler) Reconcile(ctx context.Context, req ctrl.Request) (
// if being deleted
if !o.ObjectMeta.DeletionTimestamp.IsZero() {
if controllerutil.ContainsFinalizer(o, IpAddressFinalizerName) {
if !o.Spec.PreserveInNetbox {
if o.Status.IpAddressId != 0 {
err := r.NetboxClient.DeleteIpAddress(o.Status.IpAddressId)
if err != nil {
return ctrl.Result{}, err
Expand All @@ -101,6 +101,15 @@ func (r *IpAddressReconciler) Reconcile(ctx context.Context, req ctrl.Request) (
return ctrl.Result{}, nil
}

// if PreserveIpInNetbox flag is false then register finalizer if not yet registered
if !o.Spec.PreserveInNetbox && !controllerutil.ContainsFinalizer(o, IpAddressFinalizerName) {
debugLogger.Info("adding the finalizer")
controllerutil.AddFinalizer(o, IpAddressFinalizerName)
if err := r.Update(ctx, o); err != nil {
return ctrl.Result{}, err
}
}

// 1. try to lock lease of parent prefix if IpAddress status condition is not true
// and IpAddress is owned by an IpAddressClaim
or := o.ObjectMeta.OwnerReferences
Expand Down Expand Up @@ -206,15 +215,6 @@ func (r *IpAddressReconciler) Reconcile(ctx context.Context, req ctrl.Request) (
r.Recorder.Event(o, corev1.EventTypeWarning, "IpDescriptionTruncated", "ip address was created with truncated description")
}

// if PreserveIpInNetbox flag is false then register finalizer if not yet registered
if !o.Spec.PreserveInNetbox && !controllerutil.ContainsFinalizer(o, IpAddressFinalizerName) {
debugLogger.Info("adding the finalizer")
controllerutil.AddFinalizer(o, IpAddressFinalizerName)
if err := r.Update(ctx, o); err != nil {
return ctrl.Result{}, err
}
}

debugLogger.Info(fmt.Sprintf("reserved ip address in netbox, ip: %s", o.Spec.IpAddress))

err = r.SetConditionAndCreateEvent(ctx, o, netboxv1.ConditionIpaddressReadyTrue, corev1.EventTypeNormal, "")
Expand Down
9 changes: 7 additions & 2 deletions internal/controller/ipaddressclaim_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -137,9 +137,14 @@ func (r *IpAddressClaimReconciler) Reconcile(ctx context.Context, req ctrl.Reque
},
})
if err != nil {
return ctrl.Result{}, err
setConditionErr := r.SetConditionAndCreateEvent(ctx, o, netboxv1.ConditionIpAssignedFalse, corev1.EventTypeWarning, err.Error())
if setConditionErr != nil {
return ctrl.Result{}, fmt.Errorf("error updating status: %w, when getting an available IP address failed: %w", setConditionErr, err)
}

return ctrl.Result{Requeue: true}, nil
}
debugLogger.Info(fmt.Sprintf("ip address is not reserved in netbox, assignined new ip address: %s", ipAddressModel.IpAddress))
debugLogger.Info(fmt.Sprintf("ip address is not reserved in netbox, assigned new ip address: %s", ipAddressModel.IpAddress))
} else {
// 5.b reassign reserved ip address from netbox
// do nothing, ip address restored
Expand Down
20 changes: 10 additions & 10 deletions internal/controller/prefix_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -79,7 +79,7 @@ func (r *PrefixReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctr
// if being deleted
if !prefix.ObjectMeta.DeletionTimestamp.IsZero() {
if controllerutil.ContainsFinalizer(prefix, PrefixFinalizerName) {
if !prefix.Spec.PreserveInNetbox {
if prefix.Status.PrefixId != 0 {
if err := r.NetboxClient.DeletePrefix(prefix.Status.PrefixId); err != nil {
return ctrl.Result{}, err
}
Expand All @@ -99,6 +99,15 @@ func (r *PrefixReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctr
return ctrl.Result{}, nil
}

// register finalizer if not yet registered
if !prefix.Spec.PreserveInNetbox && !controllerutil.ContainsFinalizer(prefix, PrefixFinalizerName) {
debugLogger.Info("adding the finalizer")
controllerutil.AddFinalizer(prefix, PrefixFinalizerName)
if err := r.Update(ctx, prefix); err != nil {
return ctrl.Result{}, err
}
}

/*
1. try to lock the lease of the parent prefix if all of the following conditions are met
- the prefix is owned by at least 1 prefixClaim
Expand Down Expand Up @@ -201,15 +210,6 @@ func (r *PrefixReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctr
r.Recorder.Event(prefix, corev1.EventTypeWarning, "PrefixDescriptionTruncated", "prefix was created with truncated description")
}

// register finalizer if not yet registered
if !prefix.Spec.PreserveInNetbox && !controllerutil.ContainsFinalizer(prefix, PrefixFinalizerName) {
debugLogger.Info("adding the finalizer")
controllerutil.AddFinalizer(prefix, PrefixFinalizerName)
if err := r.Update(ctx, prefix); err != nil {
return ctrl.Result{}, err
}
}

debugLogger.Info(fmt.Sprintf("reserved prefix in netbox, prefix: %s", prefix.Spec.Prefix))

if err = r.SetConditionAndCreateEvent(ctx, prefix, netboxv1.ConditionPrefixReadyTrue, corev1.EventTypeNormal, ""); err != nil {
Expand Down
7 changes: 6 additions & 1 deletion internal/controller/prefixclaim_controller.go
Original file line number Diff line number Diff line change
Expand Up @@ -135,7 +135,12 @@ func (r *PrefixClaimReconciler) Reconcile(ctx context.Context, req ctrl.Request)
},
})
if err != nil {
return ctrl.Result{}, err
setConditionErr := r.SetConditionAndCreateEvent(ctx, prefixClaim, netboxv1.ConditionPrefixAssignedFalse, corev1.EventTypeWarning, err.Error())
if setConditionErr != nil {
return ctrl.Result{}, fmt.Errorf("error updating status: %w, when getting an available Prefix failed: %w", setConditionErr, err)
}

return ctrl.Result{Requeue: true}, nil
}
debugLogger.Info(fmt.Sprintf("prefix is not reserved in netbox, assignined new prefix: %s", prefixModel.Prefix))
} else {
Expand Down
7 changes: 6 additions & 1 deletion pkg/netbox/api/ip_address_claim.go
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ import (

"github.com/netbox-community/go-netbox/v3/netbox/client/ipam"
"github.com/netbox-community/netbox-operator/pkg/netbox/models"
"github.com/netbox-community/netbox-operator/pkg/netbox/utils"
)

const (
Expand Down Expand Up @@ -57,6 +58,10 @@ func (r *NetboxClient) RestoreExistingIpByHash(customFieldName string, hash stri

// GetAvailableIpAddressByClaim searches an available IpAddress in Netbox matching IpAddressClaim requirements
func (r *NetboxClient) GetAvailableIpAddressByClaim(ipAddressClaim *models.IPAddressClaim) (*models.IPAddress, error) {
_, err := r.GetTenantDetails(ipAddressClaim.Metadata.Tenant)
if err != nil {
return nil, err
}

responseParentPrefix, err := r.GetPrefix(&models.Prefix{
Prefix: ipAddressClaim.ParentPrefix,
Expand All @@ -66,7 +71,7 @@ func (r *NetboxClient) GetAvailableIpAddressByClaim(ipAddressClaim *models.IPAdd
return nil, err
}
if len(responseParentPrefix.Payload.Results) == 0 {
return nil, errors.New("parent prefix not found")
return nil, utils.NetboxNotFoundError("parent prefix")
}

parentPrefixId := responseParentPrefix.Payload.Results[0].ID
Expand Down
80 changes: 79 additions & 1 deletion pkg/netbox/api/ip_address_claim_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ limitations under the License.
package api

import (
"errors"
"testing"

"github.com/netbox-community/go-netbox/v3/netbox/client/ipam"
Expand Down Expand Up @@ -166,8 +167,10 @@ func TestIPAddressClaim(t *testing.T) {
},
})

expectedErrMsg := "failed to fetch parent prefix: not found"

// assert error
AssertError(t, err, "parent prefix not found")
AssertError(t, err, expectedErrMsg)
// assert nil output
assert.Nil(t, actual)
})
Expand Down Expand Up @@ -231,3 +234,78 @@ func TestIPAddressClaim(t *testing.T) {
assert.Equal(t, ipAddressRestore, actual.IpAddress)
})
}

func TestIPAddressClaim_GetNoAvailableIPAddressWithTenancyChecks(t *testing.T) {
ctrl := gomock.NewController(t)
defer ctrl.Finish()

mockTenancy := mock_interfaces.NewMockTenancyInterface(ctrl)

parentPrefix := "10.112.140.0/24"
t.Run("No IP address asigned with an error when getting the tenant list", func(t *testing.T) {

tenantName := "Tenant1"

inputTenant := tenancy.NewTenancyTenantsListParams().WithName(&tenantName)

// expected error
expectedErrorMsg := "cannot get the list" // testcase-defined error

mockTenancy.EXPECT().TenancyTenantsList(inputTenant, nil).Return(nil, errors.New(expectedErrorMsg)).AnyTimes()

// init client
client := &NetboxClient{
Tenancy: mockTenancy,
}

actual, err := client.GetAvailableIpAddressByClaim(&models.IPAddressClaim{
ParentPrefix: parentPrefix,
Metadata: &models.NetboxMetadata{
Tenant: tenantName,
},
})

// assert error
assert.Containsf(t, err.Error(), expectedErrorMsg, "Error should contain: %v, got: %v", expectedErrorMsg, err.Error())
// assert nil output
assert.Equal(t, actual, (*models.IPAddress)(nil))
})

t.Run("No IP address asigned with non-existing tenant", func(t *testing.T) {

// non existing tenant
nonExistingTenant := "non-existing-tenant"

inputTenant := tenancy.NewTenancyTenantsListParams().WithName(&nonExistingTenant)

// empty tenant list
emptyTenantList := &tenancy.TenancyTenantsListOK{
Payload: &tenancy.TenancyTenantsListOKBody{
Results: []*netboxModels.Tenant{},
},
}

// expected error
expectedErrorMsg := "failed to fetch tenant: not found"

// mock empty list call
mockTenancy.EXPECT().TenancyTenantsList(inputTenant, nil).Return(emptyTenantList, nil).AnyTimes()

// init client
client := &NetboxClient{
Tenancy: mockTenancy,
}

actual, err := client.GetAvailableIpAddressByClaim(&models.IPAddressClaim{
ParentPrefix: parentPrefix,
Metadata: &models.NetboxMetadata{
Tenant: nonExistingTenant,
},
})

// assert error
assert.EqualErrorf(t, err, expectedErrorMsg, "Error should be: %v, got: %v", expectedErrorMsg, err)
// assert nil output
assert.Equal(t, actual, (*models.IPAddress)(nil))
})
}
5 changes: 5 additions & 0 deletions pkg/netbox/api/prefix_claim.go
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,11 @@ func isRequestingTheEntireParentPrefix(prefixClaim *models.PrefixClaim) (bool, e

// GetAvailablePrefixByClaim searches an available Prefix in Netbox matching PrefixClaim requirements
func (r *NetboxClient) GetAvailablePrefixByClaim(prefixClaim *models.PrefixClaim) (*models.Prefix, error) {
_, err := r.GetTenantDetails(prefixClaim.Metadata.Tenant)
if err != nil {
return nil, err
}

responseParentPrefix, err := r.GetPrefix(&models.Prefix{
Prefix: prefixClaim.ParentPrefix,
Metadata: prefixClaim.Metadata,
Expand Down
Loading

0 comments on commit 21bd119

Please sign in to comment.