From bd28908b66ab6b6cad422f0fde2be7de68002697 Mon Sep 17 00:00:00 2001 From: Karthik K N Date: Wed, 29 Jun 2022 17:49:11 +0530 Subject: [PATCH 1/3] Fetch internal IP from dhcp server --- cloud/scope/powervs_machine.go | 95 ++++++ cloud/scope/powervs_machine_test.go | 308 ++++++++++++++++++ controllers/ibmpowervsmachine_controller.go | 10 +- .../ibmpowervsmachine_controller_test.go | 8 +- main.go | 5 + .../powervs/mock/powervs_generated.go | 30 ++ pkg/cloud/services/powervs/powervs.go | 2 + pkg/cloud/services/powervs/service.go | 12 + pkg/options/caching.go | 36 ++ 9 files changed, 504 insertions(+), 2 deletions(-) create mode 100644 pkg/options/caching.go diff --git a/cloud/scope/powervs_machine.go b/cloud/scope/powervs_machine.go index c33ee42d1..f8439a6ed 100644 --- a/cloud/scope/powervs_machine.go +++ b/cloud/scope/powervs_machine.go @@ -33,6 +33,7 @@ import ( powerVSUtils "github.com/ppc64le-cloud/powervs-utils" corev1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/types" + "k8s.io/client-go/tools/cache" "k8s.io/klog/v2/klogr" "k8s.io/utils/pointer" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" @@ -60,6 +61,7 @@ type PowerVSMachineScopeParams struct { IBMPowerVSMachine *infrav1beta1.IBMPowerVSMachine IBMPowerVSImage *infrav1beta1.IBMPowerVSImage ServiceEndpoint []endpoints.ServiceEndpoint + DHCPIPCacheStore cache.Store } // PowerVSMachineScope defines a scope defined around a Power VS Machine. @@ -75,6 +77,7 @@ type PowerVSMachineScope struct { IBMPowerVSMachine *infrav1beta1.IBMPowerVSMachine IBMPowerVSImage *infrav1beta1.IBMPowerVSImage ServiceEndpoint []endpoints.ServiceEndpoint + DHCPIPCacheStore cache.Store } // NewPowerVSMachineScope creates a new PowerVSMachineScope from the supplied parameters. @@ -185,6 +188,7 @@ func NewPowerVSMachineScope(params PowerVSMachineScopeParams) (scope *PowerVSMac return } scope.IBMPowerVSClient = c + scope.DHCPIPCacheStore = params.DHCPIPCacheStore return scope, nil } @@ -435,6 +439,97 @@ func (m *PowerVSMachineScope) SetAddresses(instance *models.PVMInstance) { } } m.IBMPowerVSMachine.Status.Addresses = addresses + if len(addresses) > 2 { + // If the address length is more than 2 means either NodeInternalIP or NodeExternalIP is updated so return + return + } + // In this case there is no IP found under instance.Networks, So try to fetch the IP from cache or DHCP server + // Look for DHCP IP from the cache + obj, exists, err := m.DHCPIPCacheStore.GetByKey(*instance.ServerName) + if err != nil { + m.Error(err, "failed to fetch the DHCP IP address from cache store", "VM", *instance.ServerName) + } + if exists { + m.Info("found IP for VM from DHCP cache", "IP", obj.(options.VMip).IP, "VM", *instance.ServerName) + addresses = append(addresses, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: obj.(options.VMip).IP, + }) + m.IBMPowerVSMachine.Status.Addresses = addresses + return + } + // Fetch the VM network ID + networkID, err := getNetworkID(m.IBMPowerVSMachine.Spec.Network, m) + if err != nil { + m.Error(err, "failed to fetch network id from network resource", "VM", *instance.ServerName) + return + } + // Fetch the details of the network attached to the VM + var pvmNetwork *models.PVMInstanceNetwork + for _, network := range instance.Networks { + if network.NetworkID == *networkID { + pvmNetwork = network + m.Info("found network attached to VM", "Network ID", network.NetworkID, "VM", *instance.ServerName) + } + } + if pvmNetwork == nil { + m.Info("failed to get network attached to VM", "VM", *instance.ServerName, "Network ID", *networkID) + return + } + // Get all the DHCP servers + dhcpServer, err := m.IBMPowerVSClient.GetDHCPServers() + if err != nil { + m.Error(err, "failed to get DHCP server") + return + } + // Get the Details of DHCP server associated with the network + var dhcpServerDetails *models.DHCPServerDetail + for _, server := range dhcpServer { + if *server.Network.ID == *networkID { + m.Info("found DHCP server for network", "DHCP server ID", *server.ID, "network ID", *networkID) + dhcpServerDetails, err = m.IBMPowerVSClient.GetDHCPServerByID(*server.ID) + if err != nil { + m.Error(err, "failed to get DHCP server details", "DHCP server ID", *server.ID) + return + } + break + } + } + if dhcpServerDetails == nil { + errStr := fmt.Errorf("DHCP server details is nil") + m.Error(errStr, "DHCP server associated with network is nil", "Network ID", *networkID) + return + } + + // Fetch the VM IP using VM's mac from DHCP server lease + var internalIP *string + for _, lease := range dhcpServerDetails.Leases { + if *lease.InstanceMacAddress == pvmNetwork.MacAddress { + m.Info("found internal ip for VM from DHCP lease", "IP", *lease.InstanceIP, "VM", *instance.ServerName) + internalIP = lease.InstanceIP + break + } + } + if internalIP == nil { + errStr := fmt.Errorf("internal IP is nil") + m.Error(errStr, "failed to get internal IP, DHCP lease not found for VM with MAC in DHCP network", "VM", *instance.ServerName, + "MAC", pvmNetwork.MacAddress, "DHCP server ID", *dhcpServerDetails.ID) + return + } + m.Info("found internal IP for VM from DHCP lease", "IP", *internalIP, "VM", *instance.ServerName) + addresses = append(addresses, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: *internalIP, + }) + // Update the cache with the ip and VM name + err = m.DHCPIPCacheStore.Add(options.VMip{ + Name: *instance.ServerName, + IP: *internalIP, + }) + if err != nil { + m.Error(err, "failed to update the DHCP cache store with the IP", "VM", *instance.ServerName, "IP", *internalIP) + } + m.IBMPowerVSMachine.Status.Addresses = addresses } // SetInstanceState will set the state for the machine. diff --git a/cloud/scope/powervs_machine_test.go b/cloud/scope/powervs_machine_test.go index 69a60ce00..7a9957f75 100644 --- a/cloud/scope/powervs_machine_test.go +++ b/cloud/scope/powervs_machine_test.go @@ -18,7 +18,9 @@ package scope import ( "errors" + "fmt" "testing" + "time" "github.com/IBM-Cloud/power-go-client/power/models" "github.com/IBM/go-sdk-core/v5/core" @@ -27,7 +29,10 @@ import ( corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/client-go/kubernetes/scheme" + "k8s.io/client-go/tools/cache" "k8s.io/klog/v2/klogr" + "k8s.io/utils/pointer" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client/fake" @@ -88,6 +93,42 @@ func setupPowerVSMachineScope(clusterName string, machineName string, imageID *s Machine: machine, IBMPowerVSCluster: powervsCluster, IBMPowerVSMachine: powervsMachine, + DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), + } +} + +func newPowerVSInstance(name, networkID, mac string) *models.PVMInstance { + return &models.PVMInstance{ + ServerName: pointer.StringPtr(name), + Networks: []*models.PVMInstanceNetwork{ + { + NetworkID: networkID, + MacAddress: mac, + }, + }, + } +} + +func newDHCPServer(serverID, networkID string) models.DHCPServers { + return models.DHCPServers{ + &models.DHCPServer{ + ID: pointer.StringPtr(serverID), + Network: &models.DHCPServerNetwork{ + ID: pointer.StringPtr(networkID), + }, + }, + } +} + +func newDHCPServerDetails(serverID, leaseIP, instanceMac string) *models.DHCPServerDetail { + return &models.DHCPServerDetail{ + ID: pointer.StringPtr(serverID), + Leases: []*models.DHCPServerLeases{ + { + InstanceIP: pointer.StringPtr(leaseIP), + InstanceMacAddress: pointer.StringPtr(instanceMac), + }, + }, } } @@ -348,3 +389,270 @@ func TestDeleteMachinePVS(t *testing.T) { }) }) } + +func TestSetAddresses(t *testing.T) { + instanceName := "test_vm" + networkID := "test-net-ID" + leaseIP := "192.168.0.10" + instanceMac := "ff:11:33:dd:00:22" + dhcpServerID := "test-server-id" + defaultExpectedMachineAddress := []corev1.NodeAddress{ + { + Type: corev1.NodeInternalDNS, + Address: instanceName, + }, + { + Type: corev1.NodeHostName, + Address: instanceName, + }, + } + + defaultDhcpCacheStoreFunc := func() cache.Store { + return cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) + } + + testCases := []struct { + testcase string + powerVSClientFunc func(*gomock.Controller) *mock.MockPowerVS + pvmInstance *models.PVMInstance + expectedNodeAddress []corev1.NodeAddress + expectedError error + dhcpCacheStoreFunc func() cache.Store + setNetworkID bool + }{ + { + testcase: "should set external IP address from instance network", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + return mockPowerVSClient + }, + pvmInstance: &models.PVMInstance{ + Networks: []*models.PVMInstanceNetwork{ + { + ExternalIP: "10.11.2.3", + }, + }, + ServerName: pointer.StringPtr(instanceName), + }, + expectedNodeAddress: append(defaultExpectedMachineAddress, corev1.NodeAddress{ + Type: corev1.NodeExternalIP, + Address: "10.11.2.3", + }), + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + }, + { + testcase: "should set internal IP address from instance network", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + return mockPowerVSClient + }, + pvmInstance: &models.PVMInstance{ + Networks: []*models.PVMInstanceNetwork{ + { + IPAddress: "192.168.10.3", + }, + }, + ServerName: pointer.StringPtr(instanceName), + }, + expectedNodeAddress: append(defaultExpectedMachineAddress, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: "192.168.10.3", + }), + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + }, + { + testcase: "should set both internal and external IP address from instance network", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + return mockPowerVSClient + }, + pvmInstance: &models.PVMInstance{ + Networks: []*models.PVMInstanceNetwork{ + { + IPAddress: "192.168.10.3", + ExternalIP: "10.11.2.3", + }, + }, + ServerName: pointer.StringPtr(instanceName), + }, + expectedNodeAddress: append(defaultExpectedMachineAddress, []corev1.NodeAddress{ + { + Type: corev1.NodeInternalIP, + Address: "192.168.10.3", + }, + { + Type: corev1.NodeExternalIP, + Address: "10.11.2.3", + }, + }...), + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + }, + { + testcase: "error while getting network id", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetAllNetwork().Return(nil, fmt.Errorf("intentional error")) + return mockPowerVSClient + }, + pvmInstance: &models.PVMInstance{ + ServerName: pointer.StringPtr(instanceName), + }, + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + }, + { + testcase: "no network id associated with network name", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + networks := &models.Networks{ + Networks: []*models.NetworkReference{ + { + NetworkID: pointer.StringPtr("test-ID"), + Name: pointer.StringPtr("test-name"), + }, + }, + } + mockPowerVSClient.EXPECT().GetAllNetwork().Return(networks, nil) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + }, + { + testcase: "provided network id not attached to vm", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, "test-net", instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "error while getting DHCP servers", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(nil, fmt.Errorf("intentional error")) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "dhcp server details not found associated to network id", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, "test-network"), nil) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "error on getting DHCP server details", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(nil, fmt.Errorf("intentnional error")) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "dhcp server lease does not have lease for instance", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, "ff:11:33:dd:00:33"), nil) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: defaultExpectedMachineAddress, + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "success in getting ip address from dhcp server", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: append(defaultExpectedMachineAddress, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: leaseIP, + }), + dhcpCacheStoreFunc: defaultDhcpCacheStoreFunc, + setNetworkID: true, + }, + { + testcase: "ip stored in cache expired, fetch from dhcp server", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: append(defaultExpectedMachineAddress, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: leaseIP, + }), + dhcpCacheStoreFunc: func() cache.Store { + cacheStore := cache.NewTTLStore(options.CacheKeyFunc, time.Millisecond) + _ = cacheStore.Add(options.VMip{ + Name: instanceName, + IP: "192.168.99.98", + }) + time.Sleep(time.Millisecond) + return cacheStore + }, + setNetworkID: true, + }, + { + testcase: "success in fetching DHCP IP from cache", + powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { + mockPowerVSClient := mock.NewMockPowerVS(ctrl) + return mockPowerVSClient + }, + pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), + expectedNodeAddress: append(defaultExpectedMachineAddress, corev1.NodeAddress{ + Type: corev1.NodeInternalIP, + Address: leaseIP, + }), + dhcpCacheStoreFunc: func() cache.Store { + cacheStore := cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) + _ = cacheStore.Add(options.VMip{ + Name: instanceName, + IP: leaseIP, + }) + return cacheStore + }, + setNetworkID: true, + }, + } + for _, tc := range testCases { + t.Run(tc.testcase, func(t *testing.T) { + g := NewWithT(t) + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + mockPowerVSClient := tc.powerVSClientFunc(ctrl) + scope := setupPowerVSMachineScope("test-cluster", "test-machine-0", pointer.StringPtr("test-image-ID"), &networkID, tc.setNetworkID, mockPowerVSClient) + scope.DHCPIPCacheStore = tc.dhcpCacheStoreFunc() + scope.SetAddresses(tc.pvmInstance) + g.Expect(scope.IBMPowerVSMachine.Status.Addresses).To(Equal(tc.expectedNodeAddress)) + }) + } +} diff --git a/controllers/ibmpowervsmachine_controller.go b/controllers/ibmpowervsmachine_controller.go index 528e06542..9d702dca1 100644 --- a/controllers/ibmpowervsmachine_controller.go +++ b/controllers/ibmpowervsmachine_controller.go @@ -25,6 +25,7 @@ import ( "github.com/pkg/errors" apierrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/runtime" + "k8s.io/client-go/tools/cache" "k8s.io/client-go/tools/record" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" capierrors "sigs.k8s.io/cluster-api/errors" @@ -37,6 +38,7 @@ import ( infrav1beta1 "sigs.k8s.io/cluster-api-provider-ibmcloud/api/v1beta1" "sigs.k8s.io/cluster-api-provider-ibmcloud/cloud/scope" "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/endpoints" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capibmrecord "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/record" ) @@ -47,6 +49,7 @@ type IBMPowerVSMachineReconciler struct { Recorder record.EventRecorder ServiceEndpoint []endpoints.ServiceEndpoint Scheme *runtime.Scheme + CacheStore cache.Store } // +kubebuilder:rbac:groups=infrastructure.cluster.x-k8s.io,resources=ibmpowervsmachines,verbs=get;list;watch;create;update;patch;delete @@ -117,6 +120,7 @@ func (r *IBMPowerVSMachineReconciler) Reconcile(ctx context.Context, req ctrl.Re IBMPowerVSMachine: ibmPowerVSMachine, IBMPowerVSImage: ibmPowerVSImage, ServiceEndpoint: r.ServiceEndpoint, + DHCPIPCacheStore: r.CacheStore, }) if err != nil { return ctrl.Result{}, errors.Errorf("failed to create scope: %+v", err) @@ -163,7 +167,11 @@ func (r *IBMPowerVSMachineReconciler) reconcileDelete(scope *scope.PowerVSMachin scope.Info("error deleting IBMPowerVSMachine") return ctrl.Result{}, errors.Wrapf(err, "error deleting IBMPowerVSMachine %s/%s", scope.IBMPowerVSMachine.Namespace, scope.IBMPowerVSMachine.Name) } - + // Remove the cached VM IP + err := scope.DHCPIPCacheStore.Delete(options.VMip{Name: scope.IBMPowerVSMachine.Name}) + if err != nil { + scope.Error(err, "failed to delete the VM entry from DHCP cache store", "VM", scope.IBMPowerVSMachine.Name) + } return ctrl.Result{}, nil } diff --git a/controllers/ibmpowervsmachine_controller_test.go b/controllers/ibmpowervsmachine_controller_test.go index bb020e9a8..5d2ed96a7 100644 --- a/controllers/ibmpowervsmachine_controller_test.go +++ b/controllers/ibmpowervsmachine_controller_test.go @@ -24,13 +24,15 @@ import ( "github.com/IBM-Cloud/power-go-client/power/models" "github.com/golang/mock/gomock" - . "github.com/onsi/gomega" + corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/client-go/kubernetes/scheme" + "k8s.io/client-go/tools/cache" "k8s.io/client-go/tools/record" "k8s.io/klog/v2/klogr" "k8s.io/utils/pointer" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" "sigs.k8s.io/cluster-api/util" "sigs.k8s.io/cluster-api/util/conditions" @@ -41,6 +43,8 @@ import ( infrav1beta1 "sigs.k8s.io/cluster-api-provider-ibmcloud/api/v1beta1" "sigs.k8s.io/cluster-api-provider-ibmcloud/cloud/scope" "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs/mock" + + . "github.com/onsi/gomega" ) func TestIBMPowerVSMachineReconciler_Reconcile(t *testing.T) { @@ -297,6 +301,7 @@ func TestIBMPowerVSMachineReconciler_Delete(t *testing.T) { InstanceID: "powervs-instance-id", }, }, + DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), } mockpowervs.EXPECT().DeleteInstance(machineScope.IBMPowerVSMachine.Status.InstanceID).Return(nil) _, err := reconciler.reconcileDelete(machineScope) @@ -475,6 +480,7 @@ func TestIBMPowerVSMachineReconciler_ReconcileOperations(t *testing.T) { }, }, IBMPowerVSClient: mockpowervs, + DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), } instanceReferences := &models.PVMInstances{ diff --git a/main.go b/main.go index 0e36edcb5..9dfbb1701 100644 --- a/main.go +++ b/main.go @@ -27,6 +27,7 @@ import ( "k8s.io/apimachinery/pkg/runtime" clientgoscheme "k8s.io/client-go/kubernetes/scheme" _ "k8s.io/client-go/plugin/pkg/client/auth/gcp" + "k8s.io/client-go/tools/cache" cgrecord "k8s.io/client-go/tools/record" "k8s.io/klog/v2" "k8s.io/klog/v2/klogr" @@ -84,6 +85,9 @@ func main() { os.Exit(1) } + // Create the cache store to hold the DHCP IP + dhcpIPCacheStore := cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) + if watchNamespace != "" { setupLog.Info("Watching cluster-api objects only in namespace for reconciliation", "namespace", watchNamespace) } @@ -149,6 +153,7 @@ func main() { Recorder: mgr.GetEventRecorderFor("ibmpowervsmachine-controller"), ServiceEndpoint: serviceEndpoint, Scheme: mgr.GetScheme(), + CacheStore: dhcpIPCacheStore, }).SetupWithManager(mgr); err != nil { setupLog.Error(err, "unable to create controller", "controller", "IBMPowerVSMachine") os.Exit(1) diff --git a/pkg/cloud/services/powervs/mock/powervs_generated.go b/pkg/cloud/services/powervs/mock/powervs_generated.go index dfcbc3c09..2690c791c 100644 --- a/pkg/cloud/services/powervs/mock/powervs_generated.go +++ b/pkg/cloud/services/powervs/mock/powervs_generated.go @@ -181,6 +181,36 @@ func (mr *MockPowerVSMockRecorder) GetCosImages(id interface{}) *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetCosImages", reflect.TypeOf((*MockPowerVS)(nil).GetCosImages), id) } +// GetDHCPServerByID mocks base method. +func (m *MockPowerVS) GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetDHCPServerByID", id) + ret0, _ := ret[0].(*models.DHCPServerDetail) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetDHCPServerByID indicates an expected call of GetDHCPServerByID. +func (mr *MockPowerVSMockRecorder) GetDHCPServerByID(id interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDHCPServerByID", reflect.TypeOf((*MockPowerVS)(nil).GetDHCPServerByID), id) +} + +// GetDHCPServers mocks base method. +func (m *MockPowerVS) GetDHCPServers() (models.DHCPServers, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetDHCPServers") + ret0, _ := ret[0].(models.DHCPServers) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetDHCPServers indicates an expected call of GetDHCPServers. +func (mr *MockPowerVSMockRecorder) GetDHCPServers() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDHCPServers", reflect.TypeOf((*MockPowerVS)(nil).GetDHCPServers)) +} + // GetImage mocks base method. func (m *MockPowerVS) GetImage(id string) (*models.Image, error) { m.ctrl.T.Helper() diff --git a/pkg/cloud/services/powervs/powervs.go b/pkg/cloud/services/powervs/powervs.go index 04a800416..f8bf86854 100644 --- a/pkg/cloud/services/powervs/powervs.go +++ b/pkg/cloud/services/powervs/powervs.go @@ -37,4 +37,6 @@ type PowerVS interface { GetCosImages(id string) (*models.Job, error) GetJob(id string) (*models.Job, error) DeleteJob(id string) error + GetDHCPServers() (models.DHCPServers, error) + GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) } diff --git a/pkg/cloud/services/powervs/service.go b/pkg/cloud/services/powervs/service.go index 306b7d3e6..5092a38d6 100644 --- a/pkg/cloud/services/powervs/service.go +++ b/pkg/cloud/services/powervs/service.go @@ -36,6 +36,7 @@ type Service struct { networkClient *instance.IBMPINetworkClient imageClient *instance.IBMPIImageClient jobClient *instance.IBMPIJobClient + dhcpClient *instance.IBMPIDhcpClient } // ServiceOptions holds the PowerVS Service Options specific information. @@ -110,6 +111,16 @@ func (s *Service) GetAllNetwork() (*models.Networks, error) { return s.networkClient.GetAll() } +// GetDHCPServers returns all the DHCP servers in the Power VS service instance. +func (s *Service) GetDHCPServers() (models.DHCPServers, error) { + return s.dhcpClient.GetAll() +} + +// GetDHCPServerByID returns the details for DHCP server associated with id. +func (s *Service) GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) { + return s.dhcpClient.Get(id) +} + // NewService returns a new service for the Power VS api client. func NewService(options ServiceOptions) (PowerVS, error) { auth, err := authenticator.GetAuthenticator() @@ -130,5 +141,6 @@ func NewService(options ServiceOptions) (PowerVS, error) { networkClient: instance.NewIBMPINetworkClient(ctx, session, options.CloudInstanceID), imageClient: instance.NewIBMPIImageClient(ctx, session, options.CloudInstanceID), jobClient: instance.NewIBMPIJobClient(ctx, session, options.CloudInstanceID), + dhcpClient: instance.NewIBMPIDhcpClient(ctx, session, options.CloudInstanceID), }, nil } diff --git a/pkg/options/caching.go b/pkg/options/caching.go new file mode 100644 index 000000000..8f643e3df --- /dev/null +++ b/pkg/options/caching.go @@ -0,0 +1,36 @@ +/* +Copyright 2022 The Kubernetes 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 options + +import "time" + +// CacheTTL is duration of time to store the vm ip in cache +// Currently the default sync period is 10 minutes that means every 10 minutes +// there will be a reconciliation, So setting cache timeout to 20 minutes so the cache updates will happen +// once in 2 reconciliations. +const CacheTTL = time.Duration(20) * time.Minute + +// VMip holds the vm name and corresponding dhcp ip used to cache the dhcp ip. +type VMip struct { + Name string + IP string +} + +// CacheKeyFunc defines the key function required in TTLStore. +func CacheKeyFunc(obj interface{}) (string, error) { + return obj.(VMip).Name, nil +} From 878de0c101ef71ae43330d17906cdfe8c6510f41 Mon Sep 17 00:00:00 2001 From: Karthik K N Date: Fri, 8 Jul 2022 17:14:58 +0530 Subject: [PATCH 2/3] Localised inialising the dhcp cache store --- cloud/scope/powervs_machine.go | 15 ++++++++++----- controllers/ibmpowervsmachine_controller.go | 3 --- main.go | 5 ----- 3 files changed, 10 insertions(+), 13 deletions(-) diff --git a/cloud/scope/powervs_machine.go b/cloud/scope/powervs_machine.go index f8439a6ed..e119fe976 100644 --- a/cloud/scope/powervs_machine.go +++ b/cloud/scope/powervs_machine.go @@ -22,6 +22,7 @@ import ( "fmt" "strconv" "strings" + "sync" "github.com/IBM-Cloud/power-go-client/ibmpisession" "github.com/IBM-Cloud/power-go-client/power/client/p_cloud_p_vm_instances" @@ -51,6 +52,9 @@ import ( "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/record" ) +// used to initialise the dhcpCacheStore. +var dhcpIPCacheOnce sync.Once + // PowerVSMachineScopeParams defines the input parameters used to create a new PowerVSMachineScope. type PowerVSMachineScopeParams struct { Logger logr.Logger @@ -61,7 +65,6 @@ type PowerVSMachineScopeParams struct { IBMPowerVSMachine *infrav1beta1.IBMPowerVSMachine IBMPowerVSImage *infrav1beta1.IBMPowerVSImage ServiceEndpoint []endpoints.ServiceEndpoint - DHCPIPCacheStore cache.Store } // PowerVSMachineScope defines a scope defined around a Power VS Machine. @@ -166,7 +169,7 @@ func NewPowerVSMachineScope(params PowerVSMachineScopeParams) (scope *PowerVSMac scope.SetRegion(region) scope.SetZone(*res.RegionID) - options := powervs.ServiceOptions{ + serviceOptions := powervs.ServiceOptions{ IBMPIOptions: &ibmpisession.IBMPIOptions{ Debug: params.Logger.V(DEBUGLEVEL).Enabled(), UserAccount: account, @@ -178,18 +181,20 @@ func NewPowerVSMachineScope(params PowerVSMachineScopeParams) (scope *PowerVSMac // Fetch the service endpoint. if svcEndpoint := endpoints.FetchPVSEndpoint(region, params.ServiceEndpoint); svcEndpoint != "" { - options.IBMPIOptions.URL = svcEndpoint + serviceOptions.IBMPIOptions.URL = svcEndpoint scope.Logger.V(3).Info("overriding the default powervs service endpoint") } - c, err := powervs.NewService(options) + c, err := powervs.NewService(serviceOptions) if err != nil { err = fmt.Errorf("failed to create PowerVS service") return } scope.IBMPowerVSClient = c - scope.DHCPIPCacheStore = params.DHCPIPCacheStore + dhcpIPCacheOnce.Do(func() { + scope.DHCPIPCacheStore = cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) + }) return scope, nil } diff --git a/controllers/ibmpowervsmachine_controller.go b/controllers/ibmpowervsmachine_controller.go index 9d702dca1..dc3102bd8 100644 --- a/controllers/ibmpowervsmachine_controller.go +++ b/controllers/ibmpowervsmachine_controller.go @@ -25,7 +25,6 @@ import ( "github.com/pkg/errors" apierrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/runtime" - "k8s.io/client-go/tools/cache" "k8s.io/client-go/tools/record" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" capierrors "sigs.k8s.io/cluster-api/errors" @@ -49,7 +48,6 @@ type IBMPowerVSMachineReconciler struct { Recorder record.EventRecorder ServiceEndpoint []endpoints.ServiceEndpoint Scheme *runtime.Scheme - CacheStore cache.Store } // +kubebuilder:rbac:groups=infrastructure.cluster.x-k8s.io,resources=ibmpowervsmachines,verbs=get;list;watch;create;update;patch;delete @@ -120,7 +118,6 @@ func (r *IBMPowerVSMachineReconciler) Reconcile(ctx context.Context, req ctrl.Re IBMPowerVSMachine: ibmPowerVSMachine, IBMPowerVSImage: ibmPowerVSImage, ServiceEndpoint: r.ServiceEndpoint, - DHCPIPCacheStore: r.CacheStore, }) if err != nil { return ctrl.Result{}, errors.Errorf("failed to create scope: %+v", err) diff --git a/main.go b/main.go index 9dfbb1701..0e36edcb5 100644 --- a/main.go +++ b/main.go @@ -27,7 +27,6 @@ import ( "k8s.io/apimachinery/pkg/runtime" clientgoscheme "k8s.io/client-go/kubernetes/scheme" _ "k8s.io/client-go/plugin/pkg/client/auth/gcp" - "k8s.io/client-go/tools/cache" cgrecord "k8s.io/client-go/tools/record" "k8s.io/klog/v2" "k8s.io/klog/v2/klogr" @@ -85,9 +84,6 @@ func main() { os.Exit(1) } - // Create the cache store to hold the DHCP IP - dhcpIPCacheStore := cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) - if watchNamespace != "" { setupLog.Info("Watching cluster-api objects only in namespace for reconciliation", "namespace", watchNamespace) } @@ -153,7 +149,6 @@ func main() { Recorder: mgr.GetEventRecorderFor("ibmpowervsmachine-controller"), ServiceEndpoint: serviceEndpoint, Scheme: mgr.GetScheme(), - CacheStore: dhcpIPCacheStore, }).SetupWithManager(mgr); err != nil { setupLog.Error(err, "unable to create controller", "controller", "IBMPowerVSMachine") os.Exit(1) From d274a545c3a1a29e3cc13b9d5b311d49339dc671 Mon Sep 17 00:00:00 2001 From: Karthik K N Date: Mon, 11 Jul 2022 18:26:15 +0530 Subject: [PATCH 3/3] Localise caching to power vs controller --- cloud/scope/powervs_machine.go | 20 ++++----- cloud/scope/powervs_machine_test.go | 35 ++++++++-------- controllers/ibmpowervsmachine_controller.go | 13 +++++- .../ibmpowervsmachine_controller_test.go | 6 +-- .../services/powervs}/caching.go | 13 +++++- .../powervs/mock/powervs_generated.go | 42 +++++++++---------- pkg/cloud/services/powervs/powervs.go | 4 +- pkg/cloud/services/powervs/service.go | 8 ++-- 8 files changed, 77 insertions(+), 64 deletions(-) rename pkg/{options => cloud/services/powervs}/caching.go (84%) diff --git a/cloud/scope/powervs_machine.go b/cloud/scope/powervs_machine.go index e119fe976..689f2374c 100644 --- a/cloud/scope/powervs_machine.go +++ b/cloud/scope/powervs_machine.go @@ -22,7 +22,6 @@ import ( "fmt" "strconv" "strings" - "sync" "github.com/IBM-Cloud/power-go-client/ibmpisession" "github.com/IBM-Cloud/power-go-client/power/client/p_cloud_p_vm_instances" @@ -52,9 +51,6 @@ import ( "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/record" ) -// used to initialise the dhcpCacheStore. -var dhcpIPCacheOnce sync.Once - // PowerVSMachineScopeParams defines the input parameters used to create a new PowerVSMachineScope. type PowerVSMachineScopeParams struct { Logger logr.Logger @@ -65,6 +61,7 @@ type PowerVSMachineScopeParams struct { IBMPowerVSMachine *infrav1beta1.IBMPowerVSMachine IBMPowerVSImage *infrav1beta1.IBMPowerVSImage ServiceEndpoint []endpoints.ServiceEndpoint + DHCPIPCacheStore cache.Store } // PowerVSMachineScope defines a scope defined around a Power VS Machine. @@ -191,10 +188,7 @@ func NewPowerVSMachineScope(params PowerVSMachineScopeParams) (scope *PowerVSMac return } scope.IBMPowerVSClient = c - - dhcpIPCacheOnce.Do(func() { - scope.DHCPIPCacheStore = cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) - }) + scope.DHCPIPCacheStore = params.DHCPIPCacheStore return scope, nil } @@ -455,10 +449,10 @@ func (m *PowerVSMachineScope) SetAddresses(instance *models.PVMInstance) { m.Error(err, "failed to fetch the DHCP IP address from cache store", "VM", *instance.ServerName) } if exists { - m.Info("found IP for VM from DHCP cache", "IP", obj.(options.VMip).IP, "VM", *instance.ServerName) + m.Info("found IP for VM from DHCP cache", "IP", obj.(powervs.VMip).IP, "VM", *instance.ServerName) addresses = append(addresses, corev1.NodeAddress{ Type: corev1.NodeInternalIP, - Address: obj.(options.VMip).IP, + Address: obj.(powervs.VMip).IP, }) m.IBMPowerVSMachine.Status.Addresses = addresses return @@ -482,7 +476,7 @@ func (m *PowerVSMachineScope) SetAddresses(instance *models.PVMInstance) { return } // Get all the DHCP servers - dhcpServer, err := m.IBMPowerVSClient.GetDHCPServers() + dhcpServer, err := m.IBMPowerVSClient.GetAllDHCPServers() if err != nil { m.Error(err, "failed to get DHCP server") return @@ -492,7 +486,7 @@ func (m *PowerVSMachineScope) SetAddresses(instance *models.PVMInstance) { for _, server := range dhcpServer { if *server.Network.ID == *networkID { m.Info("found DHCP server for network", "DHCP server ID", *server.ID, "network ID", *networkID) - dhcpServerDetails, err = m.IBMPowerVSClient.GetDHCPServerByID(*server.ID) + dhcpServerDetails, err = m.IBMPowerVSClient.GetDHCPServer(*server.ID) if err != nil { m.Error(err, "failed to get DHCP server details", "DHCP server ID", *server.ID) return @@ -527,7 +521,7 @@ func (m *PowerVSMachineScope) SetAddresses(instance *models.PVMInstance) { Address: *internalIP, }) // Update the cache with the ip and VM name - err = m.DHCPIPCacheStore.Add(options.VMip{ + err = m.DHCPIPCacheStore.Add(powervs.VMip{ Name: *instance.ServerName, IP: *internalIP, }) diff --git a/cloud/scope/powervs_machine_test.go b/cloud/scope/powervs_machine_test.go index 7a9957f75..b52d9f6de 100644 --- a/cloud/scope/powervs_machine_test.go +++ b/cloud/scope/powervs_machine_test.go @@ -26,18 +26,19 @@ import ( "github.com/IBM/go-sdk-core/v5/core" "github.com/golang/mock/gomock" "github.com/stretchr/testify/require" + corev1 "k8s.io/api/core/v1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/client-go/kubernetes/scheme" "k8s.io/client-go/tools/cache" "k8s.io/klog/v2/klogr" "k8s.io/utils/pointer" - "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" "sigs.k8s.io/controller-runtime/pkg/client" "sigs.k8s.io/controller-runtime/pkg/client/fake" infrav1beta1 "sigs.k8s.io/cluster-api-provider-ibmcloud/api/v1beta1" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs" "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs/mock" . "github.com/onsi/ginkgo" @@ -93,7 +94,7 @@ func setupPowerVSMachineScope(clusterName string, machineName string, imageID *s Machine: machine, IBMPowerVSCluster: powervsCluster, IBMPowerVSMachine: powervsMachine, - DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), + DHCPIPCacheStore: cache.NewTTLStore(powervs.CacheKeyFunc, powervs.CacheTTL), } } @@ -408,7 +409,7 @@ func TestSetAddresses(t *testing.T) { } defaultDhcpCacheStoreFunc := func() cache.Store { - return cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) + return cache.NewTTLStore(powervs.CacheKeyFunc, powervs.CacheTTL) } testCases := []struct { @@ -534,7 +535,7 @@ func TestSetAddresses(t *testing.T) { testcase: "error while getting DHCP servers", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(nil, fmt.Errorf("intentional error")) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(nil, fmt.Errorf("intentional error")) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -546,7 +547,7 @@ func TestSetAddresses(t *testing.T) { testcase: "dhcp server details not found associated to network id", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, "test-network"), nil) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(newDHCPServer(dhcpServerID, "test-network"), nil) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -558,8 +559,8 @@ func TestSetAddresses(t *testing.T) { testcase: "error on getting DHCP server details", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) - mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(nil, fmt.Errorf("intentnional error")) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServer(dhcpServerID).Return(nil, fmt.Errorf("intentnional error")) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -571,8 +572,8 @@ func TestSetAddresses(t *testing.T) { testcase: "dhcp server lease does not have lease for instance", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) - mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, "ff:11:33:dd:00:33"), nil) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServer(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, "ff:11:33:dd:00:33"), nil) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -584,8 +585,8 @@ func TestSetAddresses(t *testing.T) { testcase: "success in getting ip address from dhcp server", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) - mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServer(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -600,8 +601,8 @@ func TestSetAddresses(t *testing.T) { testcase: "ip stored in cache expired, fetch from dhcp server", powerVSClientFunc: func(ctrl *gomock.Controller) *mock.MockPowerVS { mockPowerVSClient := mock.NewMockPowerVS(ctrl) - mockPowerVSClient.EXPECT().GetDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) - mockPowerVSClient.EXPECT().GetDHCPServerByID(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) + mockPowerVSClient.EXPECT().GetAllDHCPServers().Return(newDHCPServer(dhcpServerID, networkID), nil) + mockPowerVSClient.EXPECT().GetDHCPServer(dhcpServerID).Return(newDHCPServerDetails(dhcpServerID, leaseIP, instanceMac), nil) return mockPowerVSClient }, pvmInstance: newPowerVSInstance(instanceName, networkID, instanceMac), @@ -610,8 +611,8 @@ func TestSetAddresses(t *testing.T) { Address: leaseIP, }), dhcpCacheStoreFunc: func() cache.Store { - cacheStore := cache.NewTTLStore(options.CacheKeyFunc, time.Millisecond) - _ = cacheStore.Add(options.VMip{ + cacheStore := cache.NewTTLStore(powervs.CacheKeyFunc, time.Millisecond) + _ = cacheStore.Add(powervs.VMip{ Name: instanceName, IP: "192.168.99.98", }) @@ -632,8 +633,8 @@ func TestSetAddresses(t *testing.T) { Address: leaseIP, }), dhcpCacheStoreFunc: func() cache.Store { - cacheStore := cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL) - _ = cacheStore.Add(options.VMip{ + cacheStore := cache.NewTTLStore(powervs.CacheKeyFunc, powervs.CacheTTL) + _ = cacheStore.Add(powervs.VMip{ Name: instanceName, IP: leaseIP, }) diff --git a/controllers/ibmpowervsmachine_controller.go b/controllers/ibmpowervsmachine_controller.go index dc3102bd8..243df695d 100644 --- a/controllers/ibmpowervsmachine_controller.go +++ b/controllers/ibmpowervsmachine_controller.go @@ -25,6 +25,7 @@ import ( "github.com/pkg/errors" apierrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/runtime" + "k8s.io/client-go/tools/cache" "k8s.io/client-go/tools/record" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" capierrors "sigs.k8s.io/cluster-api/errors" @@ -36,8 +37,8 @@ import ( infrav1beta1 "sigs.k8s.io/cluster-api-provider-ibmcloud/api/v1beta1" "sigs.k8s.io/cluster-api-provider-ibmcloud/cloud/scope" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs" "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/endpoints" - "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capibmrecord "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/record" ) @@ -50,6 +51,13 @@ type IBMPowerVSMachineReconciler struct { Scheme *runtime.Scheme } +// dhcpCacheStore is a cache store to hold the Power VS VM DHCP IP. +var dhcpCacheStore cache.Store + +func init() { + dhcpCacheStore = powervs.InitialiseDHCPCacheStore() +} + // +kubebuilder:rbac:groups=infrastructure.cluster.x-k8s.io,resources=ibmpowervsmachines,verbs=get;list;watch;create;update;patch;delete // +kubebuilder:rbac:groups=infrastructure.cluster.x-k8s.io,resources=ibmpowervsmachines/status,verbs=get;update;patch @@ -118,6 +126,7 @@ func (r *IBMPowerVSMachineReconciler) Reconcile(ctx context.Context, req ctrl.Re IBMPowerVSMachine: ibmPowerVSMachine, IBMPowerVSImage: ibmPowerVSImage, ServiceEndpoint: r.ServiceEndpoint, + DHCPIPCacheStore: dhcpCacheStore, }) if err != nil { return ctrl.Result{}, errors.Errorf("failed to create scope: %+v", err) @@ -165,7 +174,7 @@ func (r *IBMPowerVSMachineReconciler) reconcileDelete(scope *scope.PowerVSMachin return ctrl.Result{}, errors.Wrapf(err, "error deleting IBMPowerVSMachine %s/%s", scope.IBMPowerVSMachine.Namespace, scope.IBMPowerVSMachine.Name) } // Remove the cached VM IP - err := scope.DHCPIPCacheStore.Delete(options.VMip{Name: scope.IBMPowerVSMachine.Name}) + err := scope.DHCPIPCacheStore.Delete(powervs.VMip{Name: scope.IBMPowerVSMachine.Name}) if err != nil { scope.Error(err, "failed to delete the VM entry from DHCP cache store", "VM", scope.IBMPowerVSMachine.Name) } diff --git a/controllers/ibmpowervsmachine_controller_test.go b/controllers/ibmpowervsmachine_controller_test.go index 5d2ed96a7..f267710d5 100644 --- a/controllers/ibmpowervsmachine_controller_test.go +++ b/controllers/ibmpowervsmachine_controller_test.go @@ -32,7 +32,6 @@ import ( "k8s.io/client-go/tools/record" "k8s.io/klog/v2/klogr" "k8s.io/utils/pointer" - "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/options" capiv1beta1 "sigs.k8s.io/cluster-api/api/v1beta1" "sigs.k8s.io/cluster-api/util" "sigs.k8s.io/cluster-api/util/conditions" @@ -42,6 +41,7 @@ import ( infrav1beta1 "sigs.k8s.io/cluster-api-provider-ibmcloud/api/v1beta1" "sigs.k8s.io/cluster-api-provider-ibmcloud/cloud/scope" + "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs" "sigs.k8s.io/cluster-api-provider-ibmcloud/pkg/cloud/services/powervs/mock" . "github.com/onsi/gomega" @@ -301,7 +301,7 @@ func TestIBMPowerVSMachineReconciler_Delete(t *testing.T) { InstanceID: "powervs-instance-id", }, }, - DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), + DHCPIPCacheStore: cache.NewTTLStore(powervs.CacheKeyFunc, powervs.CacheTTL), } mockpowervs.EXPECT().DeleteInstance(machineScope.IBMPowerVSMachine.Status.InstanceID).Return(nil) _, err := reconciler.reconcileDelete(machineScope) @@ -480,7 +480,7 @@ func TestIBMPowerVSMachineReconciler_ReconcileOperations(t *testing.T) { }, }, IBMPowerVSClient: mockpowervs, - DHCPIPCacheStore: cache.NewTTLStore(options.CacheKeyFunc, options.CacheTTL), + DHCPIPCacheStore: cache.NewTTLStore(powervs.CacheKeyFunc, powervs.CacheTTL), } instanceReferences := &models.PVMInstances{ diff --git a/pkg/options/caching.go b/pkg/cloud/services/powervs/caching.go similarity index 84% rename from pkg/options/caching.go rename to pkg/cloud/services/powervs/caching.go index 8f643e3df..9184abbe8 100644 --- a/pkg/options/caching.go +++ b/pkg/cloud/services/powervs/caching.go @@ -14,9 +14,13 @@ See the License for the specific language governing permissions and limitations under the License. */ -package options +package powervs -import "time" +import ( + "time" + + "k8s.io/client-go/tools/cache" +) // CacheTTL is duration of time to store the vm ip in cache // Currently the default sync period is 10 minutes that means every 10 minutes @@ -34,3 +38,8 @@ type VMip struct { func CacheKeyFunc(obj interface{}) (string, error) { return obj.(VMip).Name, nil } + +// InitialiseDHCPCacheStore returns a new cache store. +func InitialiseDHCPCacheStore() cache.Store { + return cache.NewTTLStore(CacheKeyFunc, CacheTTL) +} diff --git a/pkg/cloud/services/powervs/mock/powervs_generated.go b/pkg/cloud/services/powervs/mock/powervs_generated.go index 2690c791c..2970cf27f 100644 --- a/pkg/cloud/services/powervs/mock/powervs_generated.go +++ b/pkg/cloud/services/powervs/mock/powervs_generated.go @@ -121,6 +121,21 @@ func (mr *MockPowerVSMockRecorder) DeleteJob(id interface{}) *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeleteJob", reflect.TypeOf((*MockPowerVS)(nil).DeleteJob), id) } +// GetAllDHCPServers mocks base method. +func (m *MockPowerVS) GetAllDHCPServers() (models.DHCPServers, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "GetAllDHCPServers") + ret0, _ := ret[0].(models.DHCPServers) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// GetAllDHCPServers indicates an expected call of GetAllDHCPServers. +func (mr *MockPowerVSMockRecorder) GetAllDHCPServers() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetAllDHCPServers", reflect.TypeOf((*MockPowerVS)(nil).GetAllDHCPServers)) +} + // GetAllImage mocks base method. func (m *MockPowerVS) GetAllImage() (*models.Images, error) { m.ctrl.T.Helper() @@ -181,34 +196,19 @@ func (mr *MockPowerVSMockRecorder) GetCosImages(id interface{}) *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetCosImages", reflect.TypeOf((*MockPowerVS)(nil).GetCosImages), id) } -// GetDHCPServerByID mocks base method. -func (m *MockPowerVS) GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) { +// GetDHCPServer mocks base method. +func (m *MockPowerVS) GetDHCPServer(id string) (*models.DHCPServerDetail, error) { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetDHCPServerByID", id) + ret := m.ctrl.Call(m, "GetDHCPServer", id) ret0, _ := ret[0].(*models.DHCPServerDetail) ret1, _ := ret[1].(error) return ret0, ret1 } -// GetDHCPServerByID indicates an expected call of GetDHCPServerByID. -func (mr *MockPowerVSMockRecorder) GetDHCPServerByID(id interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDHCPServerByID", reflect.TypeOf((*MockPowerVS)(nil).GetDHCPServerByID), id) -} - -// GetDHCPServers mocks base method. -func (m *MockPowerVS) GetDHCPServers() (models.DHCPServers, error) { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "GetDHCPServers") - ret0, _ := ret[0].(models.DHCPServers) - ret1, _ := ret[1].(error) - return ret0, ret1 -} - -// GetDHCPServers indicates an expected call of GetDHCPServers. -func (mr *MockPowerVSMockRecorder) GetDHCPServers() *gomock.Call { +// GetDHCPServer indicates an expected call of GetDHCPServer. +func (mr *MockPowerVSMockRecorder) GetDHCPServer(id interface{}) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDHCPServers", reflect.TypeOf((*MockPowerVS)(nil).GetDHCPServers)) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "GetDHCPServer", reflect.TypeOf((*MockPowerVS)(nil).GetDHCPServer), id) } // GetImage mocks base method. diff --git a/pkg/cloud/services/powervs/powervs.go b/pkg/cloud/services/powervs/powervs.go index f8bf86854..c236159d3 100644 --- a/pkg/cloud/services/powervs/powervs.go +++ b/pkg/cloud/services/powervs/powervs.go @@ -37,6 +37,6 @@ type PowerVS interface { GetCosImages(id string) (*models.Job, error) GetJob(id string) (*models.Job, error) DeleteJob(id string) error - GetDHCPServers() (models.DHCPServers, error) - GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) + GetAllDHCPServers() (models.DHCPServers, error) + GetDHCPServer(id string) (*models.DHCPServerDetail, error) } diff --git a/pkg/cloud/services/powervs/service.go b/pkg/cloud/services/powervs/service.go index 5092a38d6..78fe3a3dd 100644 --- a/pkg/cloud/services/powervs/service.go +++ b/pkg/cloud/services/powervs/service.go @@ -111,13 +111,13 @@ func (s *Service) GetAllNetwork() (*models.Networks, error) { return s.networkClient.GetAll() } -// GetDHCPServers returns all the DHCP servers in the Power VS service instance. -func (s *Service) GetDHCPServers() (models.DHCPServers, error) { +// GetAllDHCPServers returns all the DHCP servers in the Power VS service instance. +func (s *Service) GetAllDHCPServers() (models.DHCPServers, error) { return s.dhcpClient.GetAll() } -// GetDHCPServerByID returns the details for DHCP server associated with id. -func (s *Service) GetDHCPServerByID(id string) (*models.DHCPServerDetail, error) { +// GetDHCPServer returns the details for DHCP server associated with id. +func (s *Service) GetDHCPServer(id string) (*models.DHCPServerDetail, error) { return s.dhcpClient.Get(id) }