diff --git a/pkg/consts/consts.go b/pkg/consts/consts.go index 9d0adab92c..0b40e0c594 100644 --- a/pkg/consts/consts.go +++ b/pkg/consts/consts.go @@ -213,6 +213,9 @@ var ( // load balancer const ( + // TODO: After dual-stack is supported, all references should be updated and this variable is not needed. + DualstackSupported = false + // PreConfiguredBackendPoolLoadBalancerTypesInternal means that the `internal` load balancers are pre-configured PreConfiguredBackendPoolLoadBalancerTypesInternal = "internal" // PreConfiguredBackendPoolLoadBalancerTypesExternal means that the `external` load balancers are pre-configured diff --git a/pkg/provider/azure_loadbalancer.go b/pkg/provider/azure_loadbalancer.go index 361fa47cf8..2ea301a693 100644 --- a/pkg/provider/azure_loadbalancer.go +++ b/pkg/provider/azure_loadbalancer.go @@ -22,7 +22,6 @@ import ( "errors" "fmt" "math" - "net" "net/netip" "reflect" "sort" @@ -48,36 +47,6 @@ import ( "sigs.k8s.io/cloud-provider-azure/pkg/retry" ) -// getServiceLoadBalancerIP retrieves LB IP from IPv4 annotation, then IPv6 annotation, then service.Spec.LoadBalancerIP. -// TODO: Dual-stack support is not implemented. -func getServiceLoadBalancerIP(service *v1.Service) string { - if service == nil { - return "" - } - - if ip, ok := service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[false]]; ok && ip != "" { - return ip - } - if ip, ok := service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[true]]; ok && ip != "" { - return ip - } - - // Retrieve LB IP from service.Spec.LoadBalancerIP (will be deprecated) - return service.Spec.LoadBalancerIP -} - -// setServiceLoadBalancerIP sets LB IP to a Service -func setServiceLoadBalancerIP(service *v1.Service, ip string) { - if service.Annotations == nil { - service.Annotations = map[string]string{} - } - if net.ParseIP(ip).To4() != nil { - service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[false]] = ip - return - } - service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[true]] = ip -} - // GetLoadBalancer returns whether the specified load balancer and its components exist, and // if so, what its status is. func (az *Cloud) GetLoadBalancer(ctx context.Context, clusterName string, service *v1.Service) (status *v1.LoadBalancerStatus, exists bool, err error) { @@ -452,7 +421,8 @@ func (az *Cloud) cleanOrphanedLoadBalancer(lb *network.LoadBalancer, existingLBs serviceName := getServiceName(service) isBackendPoolPreConfigured := az.isBackendPoolPreConfigured(service) lbResourceGroup := az.getLoadBalancerResourceGroup() - lbBackendPoolName := getBackendPoolName(clusterName, service) + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + lbBackendPoolName := getBackendPoolName(clusterName, isIPv6) lbBackendPoolID := az.getBackendPoolID(lbName, lbResourceGroup, lbBackendPoolName) if isBackendPoolPreConfigured { klog.V(2).Infof("cleanOrphanedLoadBalancer(%s, %s, %s): ignore cleanup of dirty lb because the lb is pre-configured", lbName, serviceName, clusterName) @@ -505,7 +475,7 @@ func (az *Cloud) cleanOrphanedLoadBalancer(lb *network.LoadBalancer, existingLBs } vmssNamesMap := map[string]bool{vmssName: true} - err := az.VMSet.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, lbBackendPoolID) + err := az.VMSet.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, []string{lbBackendPoolID}) if err != nil { klog.Errorf("cleanOrphanedLoadBalancer(%s, %s, %s): failed to EnsureBackendPoolDeletedFromVMSets: %v", lbName, serviceName, clusterName, err) return err @@ -524,8 +494,9 @@ func (az *Cloud) cleanOrphanedLoadBalancer(lb *network.LoadBalancer, existingLBs // safeDeleteLoadBalancer deletes the load balancer after decoupling it from the vmSet func (az *Cloud) safeDeleteLoadBalancer(lb network.LoadBalancer, clusterName, vmSetName string, service *v1.Service) *retry.Error { - lbBackendPoolID := az.getBackendPoolID(pointer.StringDeref(lb.Name, ""), az.getLoadBalancerResourceGroup(), getBackendPoolName(clusterName, service)) - _, err := az.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolID, vmSetName, lb.BackendAddressPools, true) + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + lbBackendPoolID := az.getBackendPoolID(pointer.StringDeref(lb.Name, ""), az.getLoadBalancerResourceGroup(), getBackendPoolName(clusterName, isIPv6)) + _, err := az.VMSet.EnsureBackendPoolDeleted(service, []string{lbBackendPoolID}, vmSetName, lb.BackendAddressPools, true) if err != nil { return retry.NewError(false, fmt.Errorf("safeDeleteLoadBalancer: failed to EnsureBackendPoolDeleted: %w", err)) } @@ -890,17 +861,18 @@ func (az *Cloud) determinePublicIPName(clusterName string, service *v1.Service, if name, found := service.Annotations[consts.ServiceAnnotationPIPName]; found && name != "" { return name, true, nil } + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) if ipPrefix, ok := service.Annotations[consts.ServiceAnnotationPIPPrefixID]; ok && ipPrefix != "" { - return az.getPublicIPName(clusterName, service), false, nil + return az.getPublicIPName(clusterName, service, isIPv6), false, nil } pipResourceGroup := az.getPublicIPAddressResourceGroup(service) - loadBalancerIP := getServiceLoadBalancerIP(service) + loadBalancerIP := getServiceLoadBalancerIP(service, isIPv6) // Assume that the service without loadBalancerIP set is a primary service. // If a secondary service doesn't set the loadBalancerIP, it is not allowed to share the IP. if len(loadBalancerIP) == 0 { - return az.getPublicIPName(clusterName, service), false, nil + return az.getPublicIPName(clusterName, service, isIPv6), false, nil } // For the services with loadBalancerIP set, an existing public IP is required, primary @@ -917,18 +889,25 @@ func (az *Cloud) determinePublicIPName(clusterName string, service *v1.Service, return "", false, fmt.Errorf("user supplied IP Address %s was not found in resource group %s", loadBalancerIP, pipResourceGroup) } -// pips: a non-nil pointer to a slice of existing PIPs, if the slice being pointed to is nil, listPIP would be called when needed and the slice would be filled -func (az *Cloud) findMatchedPIPByLoadBalancerIP(service *v1.Service, loadBalancerIP, pipResourceGroup string, pips *[]network.PublicIPAddress) (*network.PublicIPAddress, error) { +func (az *Cloud) safeListPIP(pipResourceGroup string, pips *[]network.PublicIPAddress) error { if pips == nil { // this should not happen - return nil, fmt.Errorf("findMatchedPIPByLoadBalancerIP: nil pip list passed") + return fmt.Errorf("nil pip list passed") } else if *pips == nil { pipList, err := az.listPIP(pipResourceGroup) if err != nil { - return nil, err + return err } *pips = pipList } + return nil +} + +// pips: a non-nil pointer to a slice of existing PIPs, if the slice being pointed to is nil, listPIP would be called when needed and the slice would be filled +func (az *Cloud) findMatchedPIPByLoadBalancerIP(service *v1.Service, loadBalancerIP, pipResourceGroup string, pips *[]network.PublicIPAddress) (*network.PublicIPAddress, error) { + if err := az.safeListPIP(pipResourceGroup, pips); err != nil { + return nil, fmt.Errorf("findMatchedPIPByLoadBalancerIP: failed to ensurePIP: %w", err) + } for _, pip := range *pips { if pip.PublicIPAddressPropertiesFormat.IPAddress != nil && *pip.PublicIPAddressPropertiesFormat.IPAddress == loadBalancerIP { @@ -963,7 +942,8 @@ func updateServiceLoadBalancerIP(service *v1.Service, serviceIP string) *v1.Serv } func (az *Cloud) findServiceIPAddress(ctx context.Context, clusterName string, service *v1.Service) (string, error) { - lbIP := getServiceLoadBalancerIP(service) + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + lbIP := getServiceLoadBalancerIP(service, isIPv6) if len(lbIP) > 0 { return lbIP, nil } @@ -1360,7 +1340,8 @@ func (az *Cloud) isFrontendIPChanged(clusterName string, config network.Frontend if !strings.EqualFold(pointer.StringDeref(config.Name, ""), lbFrontendIPConfigName) { return false, nil } - loadBalancerIP := getServiceLoadBalancerIP(service) + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + loadBalancerIP := getServiceLoadBalancerIP(service, isIPv6) isInternal := requiresInternalLoadBalancer(service) if isInternal { // Judge subnet @@ -1546,7 +1527,8 @@ func (az *Cloud) reconcileLoadBalancer(clusterName string, service *v1.Service, lbName := *lb.Name lbResourceGroup := az.getLoadBalancerResourceGroup() - lbBackendPoolID := az.getBackendPoolID(lbName, az.getLoadBalancerResourceGroup(), getBackendPoolName(clusterName, service)) + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + lbBackendPoolID := az.getBackendPoolID(lbName, az.getLoadBalancerResourceGroup(), getBackendPoolName(clusterName, isIPv6)) klog.V(2).Infof("reconcileLoadBalancer for service(%s): lb(%s/%s) wantLb(%t) resolved load balancer name", serviceName, lbResourceGroup, lbName, wantLb) defaultLBFrontendIPConfigName := az.getDefaultFrontendIPConfigName(service) defaultLBFrontendIPConfigID := az.getFrontendIPConfigID(lbName, lbResourceGroup, defaultLBFrontendIPConfigName) @@ -1668,7 +1650,8 @@ func (az *Cloud) reconcileLoadBalancer(clusterName string, service *v1.Service, if lb.LoadBalancerPropertiesFormat != nil && lb.BackendAddressPools != nil { backendPools := *lb.BackendAddressPools for _, backendPool := range backendPools { - if strings.EqualFold(pointer.StringDeref(backendPool.Name, ""), getBackendPoolName(clusterName, service)) { + isIPv6 := isBackendPoolIPv6(pointer.StringDeref(backendPool.Name, "")) + if strings.EqualFold(pointer.StringDeref(backendPool.Name, ""), getBackendPoolName(clusterName, isIPv6)) { if err := az.LoadBalancerBackendPool.EnsureHostsInPool(service, nodes, lbBackendPoolID, vmSetName, clusterName, lbName, backendPool); err != nil { return nil, err } @@ -1875,11 +1858,12 @@ func (az *Cloud) reconcileFrontendIPConfigs(clusterName string, service *v1.Serv Subnet: &subnet, } - if utilnet.IsIPv6String(service.Spec.ClusterIP) { + isIPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + if isIPv6 { configProperties.PrivateIPAddressVersion = network.IPv6 } - loadBalancerIP := getServiceLoadBalancerIP(service) + loadBalancerIP := getServiceLoadBalancerIP(service, isIPv6) if loadBalancerIP != "" { configProperties.PrivateIPAllocationMethod = network.Static configProperties.PrivateIPAddress = &loadBalancerIP @@ -2285,7 +2269,7 @@ func (az *Cloud) getExpectedLBRules( var nodeEndpointHealthprobe *network.Probe if servicehelpers.NeedsHealthCheck(service) { podPresencePath, podPresencePort := servicehelpers.GetServiceHealthCheckPathPort(service) - lbRuleName := az.getLoadBalancerRuleName(service, v1.ProtocolTCP, podPresencePort) + lbRuleName := az.getLoadBalancerRuleName(service, v1.ProtocolTCP, podPresencePort, utilnet.IsIPv6String(service.Spec.ClusterIP)) nodeEndpointHealthprobe = &network.Probe{ Name: &lbRuleName, @@ -2306,7 +2290,7 @@ func (az *Cloud) getExpectedLBRules( az.useStandardLoadBalancer() && consts.IsK8sServiceHasHAModeEnabled(service) { - lbRuleName := az.getloadbalancerHAmodeRuleName(service) + lbRuleName := az.getloadbalancerHAmodeRuleName(service, utilnet.IsIPv6String(service.Spec.ClusterIP)) klog.V(2).Infof("getExpectedLBRules lb name (%s) rule name (%s)", lbName, lbRuleName) props, err := az.getExpectedHAModeLoadBalancingRuleProperties(service, lbFrontendIPConfigID, lbBackendPoolID) @@ -2346,7 +2330,7 @@ func (az *Cloud) getExpectedLBRules( // generate lb rule for each port defined in svc object for _, port := range service.Spec.Ports { - lbRuleName := az.getLoadBalancerRuleName(service, port.Protocol, port.Port) + lbRuleName := az.getLoadBalancerRuleName(service, port.Protocol, port.Port, utilnet.IsIPv6String(service.Spec.ClusterIP)) klog.V(2).Infof("getExpectedLBRules lb name (%s) rule name (%s)", lbName, lbRuleName) isNoLBRuleRequired, err := consts.IsLBRuleOnK8sServicePortDisabled(service.Annotations, port.Port) if err != nil { @@ -2630,7 +2614,7 @@ func (az *Cloud) reconcileSecurityRules(sg network.SecurityGroup, service *v1.Se if useSharedSecurityRule(service) && !wantLb { for _, port := range ports { for _, sourceAddressPrefix := range sourceAddressPrefixes { - sharedRuleName := az.getSecurityRuleName(service, port, sourceAddressPrefix) + sharedRuleName := az.getSecurityRuleName(service, port, sourceAddressPrefix, utilnet.IsIPv6String(service.Spec.ClusterIP)) sharedIndex, sharedRule, sharedRuleFound := findSecurityRuleByName(updatedRules, sharedRuleName) if !sharedRuleFound { klog.V(4).Infof("Didn't find shared rule %s for service %s", sharedRuleName, service.Name) @@ -2723,7 +2707,7 @@ func (az *Cloud) getExpectedSecurityRules(wantLb bool, ports []v1.ServicePort, s } for j := range sourceAddressPrefixes { ix := i*len(sourceAddressPrefixes) + j - securityRuleName := az.getSecurityRuleName(service, port, sourceAddressPrefixes[j]) + securityRuleName := az.getSecurityRuleName(service, port, sourceAddressPrefixes[j], utilnet.IsIPv6String(service.Spec.ClusterIP)) nsgRule := network.SecurityRule{ Name: pointer.String(securityRuleName), SecurityRulePropertiesFormat: &network.SecurityRulePropertiesFormat{ @@ -2760,7 +2744,7 @@ func (az *Cloud) getExpectedSecurityRules(wantLb bool, ports []v1.ServicePort, s if err != nil { return nil, err } - securityRuleName := az.getSecurityRuleName(service, port, "deny_all") + securityRuleName := az.getSecurityRuleName(service, port, "deny_all", utilnet.IsIPv6String(service.Spec.ClusterIP)) nsgRule := network.SecurityRule{ Name: pointer.String(securityRuleName), SecurityRulePropertiesFormat: &network.SecurityRulePropertiesFormat{ @@ -3494,7 +3478,7 @@ func serviceOwnsPublicIP(service *v1.Service, pip *network.PublicIPAddress, clus // if there is no service tag on the pip, it is user-created pip if serviceTag == "" { - return strings.EqualFold(pointer.StringDeref(pip.IPAddress, ""), getServiceLoadBalancerIP(service)), true + return strings.EqualFold(pointer.StringDeref(pip.IPAddress, ""), getServiceLoadBalancerIP(service, utilnet.IsIPv6String(service.Spec.ClusterIP))), true } // if there is service tag on the pip, it is system-created pip @@ -3512,7 +3496,7 @@ func serviceOwnsPublicIP(service *v1.Service, pip *network.PublicIPAddress, clus } else { // if the service is not included in the tags of the system-created pip, check the ip address // this could happen for secondary services - return strings.EqualFold(pointer.StringDeref(pip.IPAddress, ""), getServiceLoadBalancerIP(service)), false + return strings.EqualFold(pointer.StringDeref(pip.IPAddress, ""), getServiceLoadBalancerIP(service, utilnet.IsIPv6String(service.Spec.ClusterIP))), false } } diff --git a/pkg/provider/azure_loadbalancer_backendpool.go b/pkg/provider/azure_loadbalancer_backendpool.go index f12536a114..74f50d5de6 100644 --- a/pkg/provider/azure_loadbalancer_backendpool.go +++ b/pkg/provider/azure_loadbalancer_backendpool.go @@ -70,10 +70,30 @@ func (bc *backendPoolTypeNodeIPConfig) EnsureHostsInPool(service *v1.Service, no return bc.VMSet.EnsureHostsInPool(service, nodes, backendPoolID, vmSetName) } +func isLBBackendPoolsExisting(lbBackendPoolNames map[bool]string, bpName *string) (found, isIPv6 bool) { + if strings.EqualFold(pointer.StringDeref(bpName, ""), lbBackendPoolNames[false]) { + isIPv6 = false + found = true + } + if strings.EqualFold(pointer.StringDeref(bpName, ""), lbBackendPoolNames[true]) { + isIPv6 = true + found = true + } + return found, isIPv6 +} + func (bc *backendPoolTypeNodeIPConfig) CleanupVMSetFromBackendPoolByCondition(slb *network.LoadBalancer, service *v1.Service, nodes []*v1.Node, clusterName string, shouldRemoveVMSetFromSLB func(string) bool) (*network.LoadBalancer, error) { - lbBackendPoolName := getBackendPoolName(clusterName, service) + v4Enabled, v6Enabled := getIPFamiliesEnabled(service) + + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } lbResourceGroup := bc.getLoadBalancerResourceGroup() - lbBackendPoolID := bc.getBackendPoolID(pointer.StringDeref(slb.Name, ""), lbResourceGroup, lbBackendPoolName) + lbBackendPoolIDs := map[bool]string{ + false: bc.getBackendPoolID(pointer.StringDeref(slb.Name, ""), lbResourceGroup, lbBackendPoolNames[false]), + true: bc.getBackendPoolID(pointer.StringDeref(slb.Name, ""), lbResourceGroup, lbBackendPoolNames[true]), + } newBackendPools := make([]network.BackendAddressPool, 0) if slb.LoadBalancerPropertiesFormat != nil && slb.BackendAddressPools != nil { newBackendPools = *slb.BackendAddressPools @@ -81,7 +101,7 @@ func (bc *backendPoolTypeNodeIPConfig) CleanupVMSetFromBackendPoolByCondition(sl vmSetNameToBackendIPConfigurationsToBeDeleted := make(map[string][]network.InterfaceIPConfiguration) for j, bp := range newBackendPools { - if strings.EqualFold(pointer.StringDeref(bp.Name, ""), lbBackendPoolName) { + if found, _ := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name); found { klog.V(2).Infof("bc.CleanupVMSetFromBackendPoolByCondition: checking the backend pool %s from standard load balancer %s", pointer.StringDeref(bp.Name, ""), pointer.StringDeref(slb.Name, "")) if bp.BackendAddressPoolPropertiesFormat != nil && bp.BackendIPConfigurations != nil { for i := len(*bp.BackendIPConfigurations) - 1; i >= 0; i-- { @@ -105,27 +125,36 @@ func (bc *backendPoolTypeNodeIPConfig) CleanupVMSetFromBackendPoolByCondition(sl } newBackendPools[j] = bp - break } } for vmSetName := range vmSetNameToBackendIPConfigurationsToBeDeleted { + shouldRefreshLB := false backendIPConfigurationsToBeDeleted := vmSetNameToBackendIPConfigurationsToBeDeleted[vmSetName] - backendpoolToBeDeleted := &[]network.BackendAddressPool{ - { - ID: pointer.String(lbBackendPoolID), + backendpoolToBeDeleted := []network.BackendAddressPool{} + lbBackendPoolIDsSlice := []string{} + findBackendpoolToBeDeleted := func(isIPv6 bool) { + lbBackendPoolIDsSlice = append(lbBackendPoolIDsSlice, lbBackendPoolIDs[isIPv6]) + backendpoolToBeDeleted = append(backendpoolToBeDeleted, network.BackendAddressPool{ + ID: pointer.String(lbBackendPoolIDs[isIPv6]), BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ BackendIPConfigurations: &backendIPConfigurationsToBeDeleted, }, - }, + }) + } + if v4Enabled { + findBackendpoolToBeDeleted(false) + } + if v6Enabled { + findBackendpoolToBeDeleted(true) } // decouple the backendPool from the node - shouldRefreshLB, err := bc.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolID, vmSetName, backendpoolToBeDeleted, true) + shouldRefreshLB, err := bc.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolIDsSlice, vmSetName, &backendpoolToBeDeleted, true) if err != nil { return nil, err } if shouldRefreshLB { - slb, _, err = bc.getAzureLoadBalancer(pointer.StringDeref(slb.Name, ""), cache.CacheReadTypeForceRefresh) + slb, _, err := bc.getAzureLoadBalancer(pointer.StringDeref(slb.Name, ""), cache.CacheReadTypeForceRefresh) if err != nil { return nil, fmt.Errorf("bc.CleanupVMSetFromBackendPoolByCondition: failed to get load balancer %s, err: %w", pointer.StringDeref(slb.Name, ""), err) } @@ -142,22 +171,32 @@ func (bc *backendPoolTypeNodeIPConfig) ReconcileBackendPools(clusterName string, newBackendPools = *lb.BackendAddressPools } - var foundBackendPool, changed, shouldRefreshLB, isOperationSucceeded, isMigration bool + var changed, shouldRefreshLB, isOperationSucceeded, isMigration bool + foundBackendPools := map[bool]bool{} lbName := *lb.Name serviceName := getServiceName(service) - lbBackendPoolName := getBackendPoolName(clusterName, service) - lbBackendPoolID := bc.getBackendPoolID(lbName, bc.getLoadBalancerResourceGroup(), lbBackendPoolName) + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } + lbBackendPoolIDs := map[bool]string{ + false: bc.getBackendPoolID(lbName, bc.getLoadBalancerResourceGroup(), lbBackendPoolNames[false]), + true: bc.getBackendPoolID(lbName, bc.getLoadBalancerResourceGroup(), lbBackendPoolNames[true]), + } vmSetName := bc.mapLoadBalancerNameToVMSet(lbName, clusterName) isBackendPoolPreConfigured := bc.isBackendPoolPreConfigured(service) mc := metrics.NewMetricContext("services", "migrate_to_nic_based_backend_pool", bc.ResourceGroup, bc.getNetworkResourceSubscriptionID(), serviceName) + backendpoolToBeDeleted := []network.BackendAddressPool{} + lbBackendPoolIDsSlice := []string{} for i := len(newBackendPools) - 1; i >= 0; i-- { bp := newBackendPools[i] - if strings.EqualFold(*bp.Name, lbBackendPoolName) { + found, isIPv6 := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name) + if found { klog.V(10).Infof("bc.ReconcileBackendPools for service (%s): lb backendpool - found wanted backendpool. not adding anything", serviceName) - foundBackendPool = true + foundBackendPools[isBackendPoolIPv6(pointer.StringDeref(bp.Name, ""))] = true // Don't bother to remove unused nodeIPConfiguration if backend pool is pre configured if isBackendPoolPreConfigured { @@ -174,8 +213,8 @@ func (bc *backendPoolTypeNodeIPConfig) ReconcileBackendPools(clusterName string, isMigration = true bp.VirtualNetwork = nil if err := bc.CreateOrUpdateLBBackendPool(lbName, bp); err != nil { - klog.Errorf("bc.ReconcileBackendPools for service (%s): failed to cleanup IP based backend pool %s: %s", serviceName, lbBackendPoolName, err.Error()) - return false, false, fmt.Errorf("bc.ReconcileBackendPools for service (%s): failed to cleanup IP based backend pool %s: %w", serviceName, lbBackendPoolName, err) + klog.Errorf("bc.ReconcileBackendPools for service (%s): failed to cleanup IP based backend pool %s: %s", serviceName, lbBackendPoolNames[isIPv6], err.Error()) + return false, false, fmt.Errorf("bc.ReconcileBackendPools for service (%s): failed to cleanup IP based backend pool %s: %w", serviceName, lbBackendPoolNames[isIPv6], err) } newBackendPools[i] = bp lb.BackendAddressPools = &newBackendPools @@ -215,26 +254,27 @@ func (bc *backendPoolTypeNodeIPConfig) ReconcileBackendPools(clusterName string, } backendIPConfigurationsToBeDeleted = getBackendIPConfigurationsToBeDeleted(bp, bipConfigNotFound, bipConfigExclude) if len(backendIPConfigurationsToBeDeleted) > 0 { - backendpoolToBeDeleted := &[]network.BackendAddressPool{ - { - ID: pointer.String(lbBackendPoolID), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - BackendIPConfigurations: &backendIPConfigurationsToBeDeleted, - }, + backendpoolToBeDeleted = append(backendpoolToBeDeleted, network.BackendAddressPool{ + ID: pointer.String(lbBackendPoolIDs[isIPv6]), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + BackendIPConfigurations: &backendIPConfigurationsToBeDeleted, }, - } - // decouple the backendPool from the node - updated, err := bc.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolID, vmSetName, backendpoolToBeDeleted, false) - if err != nil { - return false, false, err - } - if updated { - shouldRefreshLB = true - } + }) + lbBackendPoolIDsSlice = append(lbBackendPoolIDsSlice, lbBackendPoolIDs[isIPv6]) + } - break } else { - klog.V(10).Infof("bc.ReconcileBackendPools for service (%s): lb backendpool - found unmanaged backendpool %s", serviceName, *bp.Name) + klog.V(10).Infof("bc.ReconcileBackendPools for service (%s): lb backendpool - found unmanaged backendpool %s", serviceName, pointer.StringDeref(bp.Name, "")) + } + } + if len(backendpoolToBeDeleted) > 0 { + // decouple the backendPool from the node + updated, err := bc.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolIDsSlice, vmSetName, &backendpoolToBeDeleted, false) + if err != nil { + return false, false, err + } + if updated { + shouldRefreshLB = true } } @@ -245,8 +285,13 @@ func (bc *backendPoolTypeNodeIPConfig) ReconcileBackendPools(clusterName string, } } - if !foundBackendPool { - isBackendPoolPreConfigured = newBackendPool(lb, isBackendPoolPreConfigured, bc.PreConfiguredBackendPoolLoadBalancerTypes, getServiceName(service), getBackendPoolName(clusterName, service)) + for _, ipFamily := range service.Spec.IPFamilies { + if foundBackendPools[ipFamily == v1.IPFamily(network.IPv6)] { + continue + } + isBackendPoolPreConfigured = newBackendPool(lb, isBackendPoolPreConfigured, + bc.PreConfiguredBackendPoolLoadBalancerTypes, serviceName, + lbBackendPoolNames[ipFamily == v1.IPv6Protocol]) changed = true } @@ -303,14 +348,18 @@ func getBackendIPConfigurationsToBeDeleted( func (bc *backendPoolTypeNodeIPConfig) GetBackendPrivateIPs(clusterName string, service *v1.Service, lb *network.LoadBalancer) ([]string, []string) { serviceName := getServiceName(service) - lbBackendPoolName := getBackendPoolName(clusterName, service) + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } if lb.LoadBalancerPropertiesFormat == nil || lb.LoadBalancerPropertiesFormat.BackendAddressPools == nil { return nil, nil } backendPrivateIPv4s, backendPrivateIPv6s := sets.New[string](), sets.New[string]() for _, bp := range *lb.BackendAddressPools { - if strings.EqualFold(pointer.StringDeref(bp.Name, ""), lbBackendPoolName) { + found, _ := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name) + if found { klog.V(10).Infof("bc.GetBackendPrivateIPs for service (%s): found wanted backendpool %s", serviceName, pointer.StringDeref(bp.Name, "")) if bp.BackendAddressPoolPropertiesFormat != nil && bp.BackendIPConfigurations != nil { for _, backendIPConfig := range *bp.BackendIPConfigurations { @@ -352,6 +401,7 @@ func newBackendPoolTypeNodeIP(c *Cloud) BackendPool { } func (bi *backendPoolTypeNodeIP) EnsureHostsInPool(service *v1.Service, nodes []*v1.Node, backendPoolID, vmSetName, clusterName, lbName string, backendPool network.BackendAddressPool) error { + isIPv6 := isBackendPoolIPv6(pointer.StringDeref(backendPool.Name, "")) vnetResourceGroup := bi.ResourceGroup if len(bi.VnetResourceGroup) > 0 { vnetResourceGroup = bi.VnetResourceGroup @@ -360,7 +410,7 @@ func (bi *backendPoolTypeNodeIP) EnsureHostsInPool(service *v1.Service, nodes [] changed := false numOfAdd := 0 - lbBackendPoolName := getBackendPoolName(clusterName, service) + lbBackendPoolName := getBackendPoolName(clusterName, isIPv6) if strings.EqualFold(pointer.StringDeref(backendPool.Name, ""), lbBackendPoolName) && backendPool.BackendAddressPoolPropertiesFormat != nil { backendPool.VirtualNetwork = &network.SubResource{ @@ -414,11 +464,11 @@ func (bi *backendPoolTypeNodeIP) EnsureHostsInPool(service *v1.Service, nodes [] continue } - privateIP := getNodePrivateIPAddress(service, node) + privateIP := getNodePrivateIPAddress(node, isIPv6) if !existingIPs.Has(privateIP) { name := node.Name if utilnet.IsIPv6String(privateIP) { - name = fmt.Sprintf("%s-ipv6", name) + name = fmt.Sprintf("%s-%s", name, v6Suffix) } klog.V(6).Infof("bi.EnsureHostsInPool: adding %s with ip address %s", name, privateIP) @@ -444,15 +494,19 @@ func (bi *backendPoolTypeNodeIP) EnsureHostsInPool(service *v1.Service, nodes [] } func (bi *backendPoolTypeNodeIP) CleanupVMSetFromBackendPoolByCondition(slb *network.LoadBalancer, service *v1.Service, nodes []*v1.Node, clusterName string, shouldRemoveVMSetFromSLB func(string) bool) (*network.LoadBalancer, error) { - lbBackendPoolName := getBackendPoolName(clusterName, service) + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } newBackendPools := make([]network.BackendAddressPool, 0) if slb.LoadBalancerPropertiesFormat != nil && slb.BackendAddressPools != nil { newBackendPools = *slb.BackendAddressPools } - var updatedPrivateIPs bool + updatedPrivateIPs := map[bool]bool{} for j, bp := range newBackendPools { - if strings.EqualFold(pointer.StringDeref(bp.Name, ""), lbBackendPoolName) { + found, isIPv6 := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name) + if found { klog.V(2).Infof("bi.CleanupVMSetFromBackendPoolByCondition: checking the backend pool %s from standard load balancer %s", pointer.StringDeref(bp.Name, ""), pointer.StringDeref(slb.Name, "")) vmIPsToBeDeleted := sets.New[string]() for _, node := range nodes { @@ -462,8 +516,8 @@ func (bi *backendPoolTypeNodeIP) CleanupVMSetFromBackendPoolByCondition(slb *net } if shouldRemoveVMSetFromSLB(vmSetName) { - privateIP := getNodePrivateIPAddress(service, node) - klog.V(4).Infof("bi.CleanupVMSetFromBackendPoolByCondition: removing ip %s from the backend pool %s", privateIP, lbBackendPoolName) + privateIP := getNodePrivateIPAddress(node, isIPv6) + klog.V(4).Infof("bi.CleanupVMSetFromBackendPoolByCondition: removing ip %s from the backend pool %s", privateIP, lbBackendPoolNames[isIPv6]) vmIPsToBeDeleted.Insert(privateIP) } } @@ -473,23 +527,26 @@ func (bi *backendPoolTypeNodeIP) CleanupVMSetFromBackendPoolByCondition(slb *net if (*bp.LoadBalancerBackendAddresses)[i].LoadBalancerBackendAddressPropertiesFormat != nil && vmIPsToBeDeleted.Has(pointer.StringDeref((*bp.LoadBalancerBackendAddresses)[i].IPAddress, "")) { *bp.LoadBalancerBackendAddresses = append((*bp.LoadBalancerBackendAddresses)[:i], (*bp.LoadBalancerBackendAddresses)[i+1:]...) - updatedPrivateIPs = true + updatedPrivateIPs[isIPv6] = true } } } newBackendPools[j] = bp - break + } else { + klog.V(10).Infof("bi.CleanupVMSetFromBackendPoolByCondition: found unmanaged backendpool %s from standard load balancer %q", pointer.StringDeref(bp.Name, ""), pointer.StringDeref(slb.Name, "")) } + } - if updatedPrivateIPs { + for isIPv6 := range updatedPrivateIPs { klog.V(2).Infof("bi.CleanupVMSetFromBackendPoolByCondition: updating lb %s since there are private IP updates", pointer.StringDeref(slb.Name, "")) slb.BackendAddressPools = &newBackendPools for _, backendAddressPool := range *slb.BackendAddressPools { - if strings.EqualFold(lbBackendPoolName, pointer.StringDeref(backendAddressPool.Name, "")) { + if strings.EqualFold(lbBackendPoolNames[isIPv6], pointer.StringDeref(backendAddressPool.Name, "")) { if err := bi.CreateOrUpdateLBBackendPool(pointer.StringDeref(slb.Name, ""), backendAddressPool); err != nil { - return nil, fmt.Errorf("bi.CleanupVMSetFromBackendPoolByCondition: failed to create or update backend pool %s: %w", lbBackendPoolName, err) + return nil, fmt.Errorf("bi.CleanupVMSetFromBackendPoolByCondition: "+ + "failed to create or update backend pool %s: %w", lbBackendPoolNames[isIPv6], err) } } } @@ -504,41 +561,55 @@ func (bi *backendPoolTypeNodeIP) ReconcileBackendPools(clusterName string, servi newBackendPools = *lb.BackendAddressPools } - var foundBackendPool, changed, shouldRefreshLB, isOperationSucceeded, isMigration, updated bool + var changed, shouldRefreshLB, isOperationSucceeded, isMigration, updated bool + foundBackendPools := map[bool]bool{} lbName := *lb.Name serviceName := getServiceName(service) - lbBackendPoolName := getBackendPoolName(clusterName, service) + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } vmSetName := bi.mapLoadBalancerNameToVMSet(lbName, clusterName) - lbBackendPoolID := bi.getBackendPoolID(pointer.StringDeref(lb.Name, ""), bi.getLoadBalancerResourceGroup(), getBackendPoolName(clusterName, service)) + lbBackendPoolIDs := map[bool]string{ + false: bi.getBackendPoolID(pointer.StringDeref(lb.Name, ""), bi.getLoadBalancerResourceGroup(), lbBackendPoolNames[false]), + true: bi.getBackendPoolID(pointer.StringDeref(lb.Name, ""), bi.getLoadBalancerResourceGroup(), lbBackendPoolNames[true]), + } isBackendPoolPreConfigured := bi.isBackendPoolPreConfigured(service) mc := metrics.NewMetricContext("services", "migrate_to_ip_based_backend_pool", bi.ResourceGroup, bi.getNetworkResourceSubscriptionID(), serviceName) - var err error + bpIdxes := []int{} + lbBackendPoolIDsSlice := []string{} for i := len(newBackendPools) - 1; i >= 0; i-- { bp := newBackendPools[i] - if strings.EqualFold(*bp.Name, lbBackendPoolName) { + found, isIPv6 := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name) + if found { + bpIdxes = append(bpIdxes, i) klog.V(10).Infof("bi.ReconcileBackendPools for service (%s): found wanted backendpool. not adding anything", serviceName) - foundBackendPool = true - - // Don't bother to remove unused nodeIP if backend pool is pre configured - if isBackendPoolPreConfigured { - break - } + foundBackendPools[isIPv6] = true + lbBackendPoolIDsSlice = append(lbBackendPoolIDsSlice, lbBackendPoolIDs[isIPv6]) + } else { + klog.V(10).Infof("bi.ReconcileBackendPools for service (%s): found unmanaged backendpool %s", serviceName, *bp.Name) + } + } - // If the LB backend pool type is configured from nodeIPConfiguration - // to nodeIP, we need to decouple the VM NICs from the LB - // before attaching nodeIPs/podIPs to the LB backend pool. - klog.V(2).Infof("bi.ReconcileBackendPools for service (%s) and vmSet (%s): ensuring the LB is decoupled from the VMSet", serviceName, vmSetName) - shouldRefreshLB, err = bi.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolID, vmSetName, lb.BackendAddressPools, true) - if err != nil { - klog.Errorf("bi.ReconcileBackendPools for service (%s): failed to EnsureBackendPoolDeleted: %s", serviceName, err.Error()) - return false, false, err - } - if shouldRefreshLB { - isMigration = true - } + // Don't bother to remove unused nodeIP if backend pool is pre configured + if !isBackendPoolPreConfigured { + // If the LB backend pool type is configured from nodeIPConfiguration + // to nodeIP, we need to decouple the VM NICs from the LB + // before attaching nodeIPs/podIPs to the LB backend pool. + klog.V(2).Infof("bi.ReconcileBackendPools for service (%s) and vmSet (%s): ensuring the LB is decoupled from the VMSet", serviceName, vmSetName) + shouldRefreshLB, err = bi.VMSet.EnsureBackendPoolDeleted(service, lbBackendPoolIDsSlice, vmSetName, lb.BackendAddressPools, true) + if err != nil { + klog.Errorf("bi.ReconcileBackendPools for service (%s): failed to EnsureBackendPoolDeleted: %s", serviceName, err.Error()) + return false, false, err + } + if shouldRefreshLB { + isMigration = true + } + for _, i := range bpIdxes { + bp := newBackendPools[i] var nodeIPAddressesToBeDeleted []string for nodeName := range bi.excludeLoadBalancerNodes { for ip := range bi.nodePrivateIPs[nodeName] { @@ -551,7 +622,6 @@ func (bi *backendPoolTypeNodeIP) ReconcileBackendPools(clusterName string, servi updated = true } } - // delete the vnet in LoadBalancerBackendAddresses and ensure it is in the backend pool level var vnet string if bp.BackendAddressPoolPropertiesFormat != nil { @@ -580,13 +650,10 @@ func (bi *backendPoolTypeNodeIP) ReconcileBackendPools(clusterName string, servi if updated { (*lb.BackendAddressPools)[i] = bp if err := bi.CreateOrUpdateLBBackendPool(lbName, bp); err != nil { - return false, false, fmt.Errorf("bi.ReconcileBackendPools for service (%s): lb backendpool - failed to update backend pool %s for load balancer %s: %w", serviceName, lbBackendPoolName, lbName, err) + return false, false, fmt.Errorf("bi.ReconcileBackendPools for service (%s): lb backendpool - failed to update backend pool %s for load balancer %s: %w", serviceName, pointer.StringDeref(bp.Name, ""), lbName, err) } shouldRefreshLB = true } - break - } else { - klog.V(10).Infof("bi.ReconcileBackendPools for service (%s): found unmanaged backendpool %s", serviceName, *bp.Name) } } @@ -597,8 +664,13 @@ func (bi *backendPoolTypeNodeIP) ReconcileBackendPools(clusterName string, servi } } - if !foundBackendPool { - isBackendPoolPreConfigured = newBackendPool(lb, isBackendPoolPreConfigured, bi.PreConfiguredBackendPoolLoadBalancerTypes, getServiceName(service), getBackendPoolName(clusterName, service)) + for _, ipFamily := range service.Spec.IPFamilies { + if foundBackendPools[ipFamily == v1.IPFamily(network.IPv6)] { + continue + } + isBackendPoolPreConfigured = newBackendPool(lb, isBackendPoolPreConfigured, + bi.PreConfiguredBackendPoolLoadBalancerTypes, serviceName, + lbBackendPoolNames[ipFamily == v1.IPv6Protocol]) changed = true } @@ -614,14 +686,18 @@ func (bi *backendPoolTypeNodeIP) ReconcileBackendPools(clusterName string, servi func (bi *backendPoolTypeNodeIP) GetBackendPrivateIPs(clusterName string, service *v1.Service, lb *network.LoadBalancer) ([]string, []string) { serviceName := getServiceName(service) - lbBackendPoolName := getBackendPoolName(clusterName, service) + lbBackendPoolNames := map[bool]string{ + false: getBackendPoolName(clusterName, false), + true: getBackendPoolName(clusterName, true), + } if lb.LoadBalancerPropertiesFormat == nil || lb.LoadBalancerPropertiesFormat.BackendAddressPools == nil { return nil, nil } backendPrivateIPv4s, backendPrivateIPv6s := sets.New[string](), sets.New[string]() for _, bp := range *lb.BackendAddressPools { - if strings.EqualFold(pointer.StringDeref(bp.Name, ""), lbBackendPoolName) { + found, _ := isLBBackendPoolsExisting(lbBackendPoolNames, bp.Name) + if found { klog.V(10).Infof("bi.GetBackendPrivateIPs for service (%s): found wanted backendpool %s", serviceName, pointer.StringDeref(bp.Name, "")) if bp.BackendAddressPoolPropertiesFormat != nil && bp.LoadBalancerBackendAddresses != nil { for _, backendAddress := range *bp.LoadBalancerBackendAddresses { @@ -630,7 +706,7 @@ func (bi *backendPoolTypeNodeIP) GetBackendPrivateIPs(clusterName string, servic klog.V(2).Infof("bi.GetBackendPrivateIPs for service (%s): lb backendpool - found private IP %q", serviceName, *ipAddress) if utilnet.IsIPv4String(*ipAddress) { backendPrivateIPv4s.Insert(*ipAddress) - } else { + } else if utilnet.IsIPv6String(*ipAddress) { backendPrivateIPv6s.Insert(*ipAddress) } } else { @@ -647,9 +723,10 @@ func (bi *backendPoolTypeNodeIP) GetBackendPrivateIPs(clusterName string, servic func newBackendPool(lb *network.LoadBalancer, isBackendPoolPreConfigured bool, preConfiguredBackendPoolLoadBalancerTypes, serviceName, lbBackendPoolName string) bool { if isBackendPoolPreConfigured { - klog.V(2).Infof("newBackendPool for service (%s)(true): lb backendpool - PreConfiguredBackendPoolLoadBalancerTypes %s has been set but can not find corresponding backend pool, ignoring it", + klog.V(2).Infof("newBackendPool for service (%s)(true): lb backendpool - PreConfiguredBackendPoolLoadBalancerTypes %s has been set but can not find corresponding backend pool %q, ignoring it", serviceName, - preConfiguredBackendPoolLoadBalancerTypes) + preConfiguredBackendPoolLoadBalancerTypes, + lbBackendPoolName) isBackendPoolPreConfigured = false } diff --git a/pkg/provider/azure_loadbalancer_backendpool_test.go b/pkg/provider/azure_loadbalancer_backendpool_test.go index 9ff78603e4..eedc0e0f27 100644 --- a/pkg/provider/azure_loadbalancer_backendpool_test.go +++ b/pkg/provider/azure_loadbalancer_backendpool_test.go @@ -47,47 +47,6 @@ func TestEnsureHostsInPoolNodeIP(t *testing.T) { az.EnableMultipleStandardLoadBalancers = true bi := newBackendPoolTypeNodeIP(az) - backendPool := network.BackendAddressPool{ - Name: pointer.String("kubernetes"), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ - { - LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ - IPAddress: pointer.String("10.0.0.1"), - }, - }, - }, - }, - } - expectedBackendPool := network.BackendAddressPool{ - Name: pointer.String("kubernetes"), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - VirtualNetwork: &network.SubResource{ID: pointer.String("/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/virtualNetworks/vnet")}, - LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ - { - LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ - IPAddress: pointer.String("10.0.0.1"), - }, - }, - { - Name: pointer.String("vmss-0"), - LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ - IPAddress: pointer.String("10.0.0.2"), - }, - }, - }, - }, - } - - mockVMSet := NewMockVMSet(ctrl) - mockVMSet.EXPECT().GetNodeVMSetName(gomock.Any()).Return("vmss-0", nil) - mockVMSet.EXPECT().GetPrimaryVMSetName().Return("vmss-0") - az.VMSet = mockVMSet - - lbClient := mockloadbalancerclient.NewMockInterface(ctrl) - lbClient.EXPECT().CreateOrUpdateBackendPools(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) - az.LoadBalancerClient = lbClient - nodes := []*v1.Node{ { ObjectMeta: metav1.ObjectMeta{ @@ -105,15 +64,156 @@ func TestEnsureHostsInPoolNodeIP(t *testing.T) { Type: v1.NodeInternalIP, Address: "10.0.0.2", }, + { + Type: v1.NodeInternalIP, + Address: "2001::2", + }, }, }, }, } - service := getTestService("svc-1", v1.ProtocolTCP, nil, false, 80) - err := bi.EnsureHostsInPool(&service, nodes, "", "", "kubernetes", "kubernetes", backendPool) - assert.NoError(t, err) - assert.Equal(t, expectedBackendPool, backendPool) + testcases := []struct { + desc string + backendPool network.BackendAddressPool + expectedBackendPool network.BackendAddressPool + }{ + { + desc: "IPv4", + backendPool: network.BackendAddressPool{ + Name: pointer.String("kubernetes"), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ + { + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("10.0.0.1"), + }, + }, + }, + }, + }, + expectedBackendPool: network.BackendAddressPool{ + Name: pointer.String("kubernetes"), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + VirtualNetwork: &network.SubResource{ID: pointer.String("/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/virtualNetworks/vnet")}, + LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ + { + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("10.0.0.1"), + }, + }, + { + Name: pointer.String("vmss-0"), + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("10.0.0.2"), + }, + }, + }, + }, + }, + }, + { + desc: "IPv6", + backendPool: network.BackendAddressPool{ + Name: pointer.String("kubernetes-IPv6"), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ + { + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("2001::1"), + }, + }, + }, + }, + }, + expectedBackendPool: network.BackendAddressPool{ + Name: pointer.String("kubernetes-IPv6"), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + VirtualNetwork: &network.SubResource{ID: pointer.String("/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/virtualNetworks/vnet")}, + LoadBalancerBackendAddresses: &[]network.LoadBalancerBackendAddress{ + { + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("2001::1"), + }, + }, + { + Name: pointer.String("vmss-0-IPv6"), + LoadBalancerBackendAddressPropertiesFormat: &network.LoadBalancerBackendAddressPropertiesFormat{ + IPAddress: pointer.String("2001::2"), + }, + }, + }, + }, + }, + }, + } + + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + mockVMSet := NewMockVMSet(ctrl) + mockVMSet.EXPECT().GetNodeVMSetName(gomock.Any()).Return("vmss-0", nil) + mockVMSet.EXPECT().GetPrimaryVMSetName().Return("vmss-0") + az.VMSet = mockVMSet + + lbClient := mockloadbalancerclient.NewMockInterface(ctrl) + lbClient.EXPECT().CreateOrUpdateBackendPools(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) + az.LoadBalancerClient = lbClient + + service := getTestService("svc-1", v1.ProtocolTCP, nil, false, 80) + makeTestServiceDualStack(&service) + err := bi.EnsureHostsInPool(&service, nodes, "", "", "kubernetes", "kubernetes", tc.backendPool) + assert.NoError(t, err) + assert.Equal(t, tc.expectedBackendPool, tc.backendPool) + }) + } +} + +func TestIsLBBackendPoolsExisting(t *testing.T) { + testcases := []struct { + desc string + lbBackendPoolNames map[bool]string + bpName *string + expectedFound bool + expectedIsIPv6 bool + }{ + { + desc: "IPv4 backendpool exists", + lbBackendPoolNames: map[bool]string{ + false: "bp", + true: "bp-IPv6", + }, + bpName: pointer.String("bp"), + expectedFound: true, + expectedIsIPv6: false, + }, + { + desc: "IPv6 backendpool exists", + lbBackendPoolNames: map[bool]string{ + false: "bp", + true: "bp-IPv6", + }, + bpName: pointer.String("bp-IPv6"), + expectedFound: true, + expectedIsIPv6: true, + }, + { + desc: "backendpool not exists", + lbBackendPoolNames: map[bool]string{ + false: "bp", + true: "bp-IPv6", + }, + bpName: pointer.String("bp0"), + expectedFound: false, + expectedIsIPv6: false, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + found, isIPv6 := isLBBackendPoolsExisting(tc.lbBackendPoolNames, tc.bpName) + assert.Equal(t, tc.expectedFound, found) + assert.Equal(t, tc.expectedIsIPv6, isIPv6) + }) + } } func TestCleanupVMSetFromBackendPoolByConditionNodeIPConfig(t *testing.T) { @@ -613,6 +713,48 @@ func TestGetBackendPrivateIPsNodeIPConfig(t *testing.T) { assert.Equal(t, []string{"fe80::1"}, ipv6) } +func TestGetBackendPrivateIPsNodeIP(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + svc := getTestService("svc1", "TCP", nil, false) // isIPv6 doesn't matter. + testcases := []struct { + desc string + lb *network.LoadBalancer + expectedIPv4 []string + expectedIPv6 []string + }{ + { + "normal", + buildLBWithVMIPs(testClusterName, []string{"1.2.3.4", "fe80::1"}), + []string{"1.2.3.4"}, + []string{"fe80::1"}, + }, + { + "some invalid IPs", + buildLBWithVMIPs(testClusterName, []string{"1.2.3.4.5", "fe80::1"}), + []string{}, + []string{"fe80::1"}, + }, + { + "no IPs", + buildLBWithVMIPs(testClusterName, []string{}), + []string{}, + []string{}, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + az := GetTestCloud(ctrl) + az.VMSet = NewMockVMSet(ctrl) + bi := newBackendPoolTypeNodeIP(az) + ipv4, ipv6 := bi.GetBackendPrivateIPs(testClusterName, &svc, tc.lb) + assert.Equal(t, tc.expectedIPv4, ipv4) + assert.Equal(t, tc.expectedIPv6, ipv6) + }) + } +} + func TestGetBackendIPConfigurationsToBeDeleted(t *testing.T) { for _, tc := range []struct { description string diff --git a/pkg/provider/azure_loadbalancer_test.go b/pkg/provider/azure_loadbalancer_test.go index 6fba65f96f..842162d7e2 100644 --- a/pkg/provider/azure_loadbalancer_test.go +++ b/pkg/provider/azure_loadbalancer_test.go @@ -5813,8 +5813,8 @@ func TestReconcileSharedLoadBalancer(t *testing.T) { } mockVMSet := NewMockVMSet(ctrl) - mockVMSet.EXPECT().EnsureBackendPoolDeleted(gomock.Any(), "/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/loadBalancers/vmss1/backendAddressPools/kubernetes", "vmss1", gomock.Any(), gomock.Any()).Return(false, nil).Times(tc.expectedDeleteCount) - mockVMSet.EXPECT().EnsureBackendPoolDeleted(gomock.Any(), "/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/loadBalancers/vmss1-internal/backendAddressPools/kubernetes", "vmss1", gomock.Any(), gomock.Any()).Return(false, nil).Times(tc.expectedDeleteCount) + mockVMSet.EXPECT().EnsureBackendPoolDeleted(gomock.Any(), []string{"/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/loadBalancers/vmss1/backendAddressPools/kubernetes"}, "vmss1", gomock.Any(), gomock.Any()).Return(false, nil).Times(tc.expectedDeleteCount) + mockVMSet.EXPECT().EnsureBackendPoolDeleted(gomock.Any(), []string{"/subscriptions/subscription/resourceGroups/rg/providers/Microsoft.Network/loadBalancers/vmss1-internal/backendAddressPools/kubernetes"}, "vmss1", gomock.Any(), gomock.Any()).Return(false, nil).Times(tc.expectedDeleteCount) mockVMSet.EXPECT().GetAgentPoolVMSetNames(gomock.Any()).Return(&[]string{"vmss1", "vmss2"}, nil).MaxTimes(tc.expectedGetNamesCount) mockVMSet.EXPECT().GetPrimaryVMSetName().Return("vmss2").AnyTimes() cloud.VMSet = mockVMSet diff --git a/pkg/provider/azure_mock_vmsets.go b/pkg/provider/azure_mock_vmsets.go index 42f33ebec7..f461cb8211 100644 --- a/pkg/provider/azure_mock_vmsets.go +++ b/pkg/provider/azure_mock_vmsets.go @@ -98,32 +98,32 @@ func (mr *MockVMSetMockRecorder) DetachDisk(ctx, nodeName, diskMap interface{}) } // EnsureBackendPoolDeleted mocks base method. -func (m *MockVMSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { +func (m *MockVMSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "EnsureBackendPoolDeleted", service, backendPoolID, vmSetName, backendAddressPools, deleteFromVMSet) + ret := m.ctrl.Call(m, "EnsureBackendPoolDeleted", service, backendPoolIDs, vmSetName, backendAddressPools, deleteFromVMSet) ret0, _ := ret[0].(bool) ret1, _ := ret[1].(error) return ret0, ret1 } // EnsureBackendPoolDeleted indicates an expected call of EnsureBackendPoolDeleted. -func (mr *MockVMSetMockRecorder) EnsureBackendPoolDeleted(service, backendPoolID, vmSetName, backendAddressPools, deleteFromVMSet interface{}) *gomock.Call { +func (mr *MockVMSetMockRecorder) EnsureBackendPoolDeleted(service, backendPoolIDs, vmSetName, backendAddressPools, deleteFromVMSet interface{}) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "EnsureBackendPoolDeleted", reflect.TypeOf((*MockVMSet)(nil).EnsureBackendPoolDeleted), service, backendPoolID, vmSetName, backendAddressPools, deleteFromVMSet) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "EnsureBackendPoolDeleted", reflect.TypeOf((*MockVMSet)(nil).EnsureBackendPoolDeleted), service, backendPoolIDs, vmSetName, backendAddressPools, deleteFromVMSet) } // EnsureBackendPoolDeletedFromVMSets mocks base method. -func (m *MockVMSet) EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap map[string]bool, backendPoolID string) error { +func (m *MockVMSet) EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap map[string]bool, backendPoolIDs []string) error { m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "EnsureBackendPoolDeletedFromVMSets", vmSetNamesMap, backendPoolID) + ret := m.ctrl.Call(m, "EnsureBackendPoolDeletedFromVMSets", vmSetNamesMap, backendPoolIDs) ret0, _ := ret[0].(error) return ret0 } // EnsureBackendPoolDeletedFromVMSets indicates an expected call of EnsureBackendPoolDeletedFromVMSets. -func (mr *MockVMSetMockRecorder) EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap, backendPoolID interface{}) *gomock.Call { +func (mr *MockVMSetMockRecorder) EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap, backendPoolIDs interface{}) *gomock.Call { mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "EnsureBackendPoolDeletedFromVMSets", reflect.TypeOf((*MockVMSet)(nil).EnsureBackendPoolDeletedFromVMSets), vmSetNamesMap, backendPoolID) + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "EnsureBackendPoolDeletedFromVMSets", reflect.TypeOf((*MockVMSet)(nil).EnsureBackendPoolDeletedFromVMSets), vmSetNamesMap, backendPoolIDs) } // EnsureHostInPool mocks base method. diff --git a/pkg/provider/azure_standard.go b/pkg/provider/azure_standard.go index f6649a81e7..01b0851419 100644 --- a/pkg/provider/azure_standard.go +++ b/pkg/provider/azure_standard.go @@ -39,7 +39,6 @@ import ( "k8s.io/apimachinery/pkg/util/uuid" cloudprovider "k8s.io/cloud-provider" "k8s.io/klog/v2" - utilnet "k8s.io/utils/net" "k8s.io/utils/pointer" azcache "sigs.k8s.io/cloud-provider-azure/pkg/cache" @@ -54,6 +53,9 @@ var ( nicIDRE = regexp.MustCompile(`(?i)/subscriptions/(?:.*)/resourceGroups/(.+)/providers/Microsoft.Network/networkInterfaces/(.+)/ipConfigurations/(?:.*)`) vmIDRE = regexp.MustCompile(`(?i)/subscriptions/(?:.*)/resourceGroups/(?:.*)/providers/Microsoft.Compute/virtualMachines/(.+)`) vmasIDRE = regexp.MustCompile(`/subscriptions/(?:.*)/resourceGroups/(?:.*)/providers/Microsoft.Compute/availabilitySets/(.+)`) + + v4Suffix = "IPv4" + v6Suffix = "IPv6" ) // returns the full identifier of an availabilitySet @@ -261,44 +263,50 @@ func isInternalLoadBalancer(lb *network.LoadBalancer) bool { // This means: // clusters moving from IPv4 to dualstack will require no changes // clusters moving from IPv6 to dualstack will require no changes as the IPv4 backend pool will created with -func getBackendPoolName(clusterName string, service *v1.Service) string { - IPv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) - if IPv6 { - return fmt.Sprintf("%v-IPv6", clusterName) +func getBackendPoolName(clusterName string, isIPv6 bool) string { + if isIPv6 { + return fmt.Sprintf("%s-%s", clusterName, v6Suffix) } return clusterName } -func (az *Cloud) getLoadBalancerRuleName(service *v1.Service, protocol v1.Protocol, port int32) string { +// ifBackendPoolIPv6 checks if a backend pool is of IPv6 according to name/ID. +func isBackendPoolIPv6(name string) bool { + return strings.HasSuffix(name, "-IPv6") +} + +func (az *Cloud) getLoadBalancerRuleName(service *v1.Service, protocol v1.Protocol, port int32, isIPv6 bool) string { prefix := az.getRulePrefix(service) ruleName := fmt.Sprintf("%s-%s-%d", prefix, protocol, port) subnet := subnet(service) if subnet == nil { - return ruleName + return getResourceByIPFamily(ruleName, isIPv6) } // Load balancer rule name must be less or equal to 80 characters, so excluding the hyphen two segments cannot exceed 79 subnetSegment := *subnet - if len(ruleName)+len(subnetSegment)+1 > consts.LoadBalancerRuleNameMaxLength { - subnetSegment = subnetSegment[:consts.LoadBalancerRuleNameMaxLength-len(ruleName)-1] + maxLength := consts.LoadBalancerRuleNameMaxLength - consts.IPFamilySuffixLength + if len(ruleName)+len(subnetSegment)+1 > maxLength { + subnetSegment = subnetSegment[:maxLength-len(ruleName)-1] } - return fmt.Sprintf("%s-%s-%s-%d", prefix, subnetSegment, protocol, port) + return getResourceByIPFamily(fmt.Sprintf("%s-%s-%s-%d", prefix, subnetSegment, protocol, port), isIPv6) } -func (az *Cloud) getloadbalancerHAmodeRuleName(service *v1.Service) string { - return az.getLoadBalancerRuleName(service, service.Spec.Ports[0].Protocol, service.Spec.Ports[0].Port) +func (az *Cloud) getloadbalancerHAmodeRuleName(service *v1.Service, isIPv6 bool) string { + return az.getLoadBalancerRuleName(service, service.Spec.Ports[0].Protocol, service.Spec.Ports[0].Port, isIPv6) } -func (az *Cloud) getSecurityRuleName(service *v1.Service, port v1.ServicePort, sourceAddrPrefix string) string { +func (az *Cloud) getSecurityRuleName(service *v1.Service, port v1.ServicePort, sourceAddrPrefix string, isIPv6 bool) string { safePrefix := strings.Replace(sourceAddrPrefix, "/", "_", -1) safePrefix = strings.Replace(safePrefix, ":", ".", -1) // Consider IPv6 address if useSharedSecurityRule(service) { return fmt.Sprintf("shared-%s-%d-%s", port.Protocol, port.Port, safePrefix) } rulePrefix := az.getRulePrefix(service) - return fmt.Sprintf("%s-%s-%d-%s", rulePrefix, port.Protocol, port.Port, safePrefix) + name := fmt.Sprintf("%s-%s-%d-%s", rulePrefix, port.Protocol, port.Port, safePrefix) + return getResourceByIPFamily(name, isIPv6) } // This returns a human-readable version of the Service used to tag some resources. @@ -312,14 +320,15 @@ func (az *Cloud) getRulePrefix(service *v1.Service) string { return az.GetLoadBalancerName(context.TODO(), "", service) } -func (az *Cloud) getPublicIPName(clusterName string, service *v1.Service) string { +func (az *Cloud) getPublicIPName(clusterName string, service *v1.Service, isIPv6 bool) string { pipName := fmt.Sprintf("%s-%s", clusterName, az.GetLoadBalancerName(context.TODO(), clusterName, service)) - if prefixID, ok := service.Annotations[consts.ServiceAnnotationPIPPrefixID]; ok && prefixID != "" { - prefixName, err := getLastSegment(prefixID, "/") + pipName = getResourceByIPFamily(pipName, isIPv6) + if id := getServicePIPPrefixID(service, isIPv6); id != "" { + id, err := getLastSegment(id, "/") if err != nil { return pipName } - pipName = fmt.Sprintf("%s-%s", pipName, prefixName) + pipName = fmt.Sprintf("%s-%s", pipName, id) } return pipName } @@ -343,8 +352,8 @@ func (az *Cloud) serviceOwnsFrontendIP(fip network.FrontendIPConfiguration, serv return true, isPrimaryService, nil } - loadBalancerIP := getServiceLoadBalancerIP(service) - if loadBalancerIP == "" { + loadBalancerIPs := getServiceLoadBalancerIPs(service) + if len(loadBalancerIPs) == 0 { // it is a must that the secondary services set the loadBalancer IP return false, isPrimaryService, nil } @@ -352,24 +361,26 @@ func (az *Cloud) serviceOwnsFrontendIP(fip network.FrontendIPConfiguration, serv // for external secondary service the public IP address should be checked if !requiresInternalLoadBalancer(service) { pipResourceGroup := az.getPublicIPAddressResourceGroup(service) - pip, err := az.findMatchedPIPByLoadBalancerIP(service, loadBalancerIP, pipResourceGroup, pips) - if err != nil { - klog.Warningf("serviceOwnsFrontendIP: unexpected error when finding match public IP of the service %s with loadBalancerLP %s: %v", service.Name, loadBalancerIP, err) - return false, isPrimaryService, nil - } - - if pip != nil && - pip.ID != nil && - pip.PublicIPAddressPropertiesFormat != nil && - pip.IPAddress != nil && - fip.FrontendIPConfigurationPropertiesFormat != nil && - fip.FrontendIPConfigurationPropertiesFormat.PublicIPAddress != nil { - if strings.EqualFold(pointer.StringDeref(pip.ID, ""), pointer.StringDeref(fip.PublicIPAddress.ID, "")) { - klog.V(4).Infof("serviceOwnsFrontendIP: found secondary service %s of the frontend IP config %s", service.Name, *fip.Name) + for _, loadBalancerIP := range loadBalancerIPs { + pip, err := az.findMatchedPIPByLoadBalancerIP(service, loadBalancerIP, pipResourceGroup, pips) + if err != nil { + klog.Warningf("serviceOwnsFrontendIP: unexpected error when finding match public IP of the service %s with loadBalancerIP %s: %v", service.Name, loadBalancerIP, err) + return false, isPrimaryService, nil + } - return true, isPrimaryService, nil + if pip != nil && + pip.ID != nil && + pip.PublicIPAddressPropertiesFormat != nil && + pip.PublicIPAddressPropertiesFormat.IPAddress != nil && + fip.FrontendIPConfigurationPropertiesFormat != nil && + fip.FrontendIPConfigurationPropertiesFormat.PublicIPAddress != nil { + if strings.EqualFold(pointer.StringDeref(pip.ID, ""), pointer.StringDeref(fip.PublicIPAddress.ID, "")) { + klog.Infof("serviceOwnsFrontendIP: found secondary service %s of the frontend IP config %s", service.Name, *fip.Name) + return true, isPrimaryService, nil + } } - klog.V(4).Infof("serviceOwnsFrontendIP: the public IP with ID %s is being referenced by other service with public IP address %s", *pip.ID, *pip.IPAddress) + klog.Infof("serviceOwnsFrontendIP: the public IP with ID %s is being referenced by other service with public IP address %s "+ + "OR it is of incorrect IP version", *pip.ID, *pip.IPAddress) } return false, isPrimaryService, nil @@ -380,7 +391,14 @@ func (az *Cloud) serviceOwnsFrontendIP(fip network.FrontendIPConfiguration, serv return false, isPrimaryService, nil } - return strings.EqualFold(*fip.PrivateIPAddress, loadBalancerIP), isPrimaryService, nil + privateIPEquals := false + for _, loadBalancerIP := range loadBalancerIPs { + if strings.EqualFold(*fip.PrivateIPAddress, loadBalancerIP) { + privateIPEquals = true + break + } + } + return privateIPEquals, isPrimaryService, nil } func (az *Cloud) getDefaultFrontendIPConfigName(service *v1.Service) string { @@ -390,8 +408,9 @@ func (az *Cloud) getDefaultFrontendIPConfigName(service *v1.Service) string { ipcName := fmt.Sprintf("%s-%s", baseName, *subnetName) // Azure lb front end configuration name must not exceed 80 characters - if len(ipcName) > consts.FrontendIPConfigNameMaxLength { - ipcName = ipcName[:consts.FrontendIPConfigNameMaxLength] + maxLength := consts.FrontendIPConfigNameMaxLength - consts.IPFamilySuffixLength + if len(ipcName) > maxLength { + ipcName = ipcName[:maxLength] // Cutting the string may result in char like "-" as the string end. // If the last char is not a letter or '_', replace it with "_". if !unicode.IsLetter(rune(ipcName[len(ipcName)-1:][0])) && ipcName[len(ipcName)-1:] != "_" { @@ -929,7 +948,7 @@ func (as *availabilitySet) EnsureHostInPool(service *v1.Service, nodeName types. } var primaryIPConfig *network.InterfaceIPConfiguration - ipv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + ipv6 := isBackendPoolIPv6(backendPoolID) if !as.Cloud.ipv6DualStackEnabled && !ipv6 { primaryIPConfig, err = getPrimaryIPConfig(nic) if err != nil { @@ -1039,7 +1058,8 @@ func (as *availabilitySet) EnsureHostsInPool(service *v1.Service, nodes []*v1.No } // EnsureBackendPoolDeleted ensures the loadBalancer backendAddressPools deleted from the specified nodes. -func (as *availabilitySet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { +// backendPoolIDs are the IDs of the backendpools to be deleted. +func (as *availabilitySet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { // Returns nil if backend address pools already deleted. if backendAddressPools == nil { return false, nil @@ -1053,23 +1073,33 @@ func (as *availabilitySet) EnsureBackendPoolDeleted(service *v1.Service, backend ipConfigurationIDs := []string{} for _, backendPool := range *backendAddressPools { - if strings.EqualFold(pointer.StringDeref(backendPool.ID, ""), backendPoolID) && - backendPool.BackendAddressPoolPropertiesFormat != nil && - backendPool.BackendIPConfigurations != nil { - for _, ipConf := range *backendPool.BackendIPConfigurations { - if ipConf.ID == nil { - continue - } + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(pointer.StringDeref(backendPool.ID, ""), backendPoolID) { + if backendPool.BackendAddressPoolPropertiesFormat != nil && + backendPool.BackendIPConfigurations != nil { + for _, ipConf := range *backendPool.BackendIPConfigurations { + if ipConf.ID == nil { + continue + } - ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) + ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) + } + } } } + } nicUpdaters := make([]func() error, 0) allErrs := make([]error, 0) var nicUpdated bool + + ipconfigPrefixToNicMap := map[string]network.Interface{} // ipconfig prefix -> nic for i := range ipConfigurationIDs { ipConfigurationID := ipConfigurationIDs[i] + ipConfigIDPrefix := getResourceIDPrefix(ipConfigurationID) + if _, ok := ipconfigPrefixToNicMap[ipConfigIDPrefix]; ok { + continue + } nodeName, _, err := as.GetNodeNameByIPConfigurationID(ipConfigurationID) if err != nil && !errors.Is(err, cloudprovider.InstanceNotFound) { klog.Errorf("Failed to GetNodeNameByIPConfigurationID(%s): %v", ipConfigurationID, err) @@ -1107,38 +1137,43 @@ func (as *availabilitySet) EnsureBackendPoolDeleted(service *v1.Service, backend } if nic.InterfacePropertiesFormat != nil && nic.InterfacePropertiesFormat.IPConfigurations != nil { - newIPConfigs := *nic.IPConfigurations - for j, ipConf := range newIPConfigs { - if !pointer.BoolDeref(ipConf.Primary, false) { - continue - } - // found primary ip configuration - if ipConf.LoadBalancerBackendAddressPools != nil { - newLBAddressPools := *ipConf.LoadBalancerBackendAddressPools - for k := len(newLBAddressPools) - 1; k >= 0; k-- { - pool := newLBAddressPools[k] + ipconfigPrefixToNicMap[ipConfigIDPrefix] = nic + } + } + for _, nic := range ipconfigPrefixToNicMap { + newIPConfigs := *nic.IPConfigurations + for j, ipConf := range newIPConfigs { + if !pointer.BoolDeref(ipConf.Primary, false) { + continue + } + // found primary ip configuration + if ipConf.LoadBalancerBackendAddressPools != nil { + newLBAddressPools := *ipConf.LoadBalancerBackendAddressPools + for k := len(newLBAddressPools) - 1; k >= 0; k-- { + pool := newLBAddressPools[k] + for _, backendPoolID := range backendPoolIDs { if strings.EqualFold(pointer.StringDeref(pool.ID, ""), backendPoolID) { newLBAddressPools = append(newLBAddressPools[:k], newLBAddressPools[k+1:]...) break } } - newIPConfigs[j].LoadBalancerBackendAddressPools = &newLBAddressPools } + newIPConfigs[j].LoadBalancerBackendAddressPools = &newLBAddressPools } - nic.IPConfigurations = &newIPConfigs - nicUpdaters = append(nicUpdaters, func() error { - ctx, cancel := getContextWithCancel() - defer cancel() - klog.V(2).Infof("EnsureBackendPoolDeleted begins to CreateOrUpdate for NIC(%s, %s) with backendPoolID %s", as.ResourceGroup, pointer.StringDeref(nic.Name, ""), backendPoolID) - rerr := as.InterfacesClient.CreateOrUpdate(ctx, as.ResourceGroup, pointer.StringDeref(nic.Name, ""), nic) - if rerr != nil { - klog.Errorf("EnsureBackendPoolDeleted CreateOrUpdate for NIC(%s, %s) failed with error %v", as.ResourceGroup, pointer.StringDeref(nic.Name, ""), rerr.Error()) - return rerr.Error() - } - nicUpdated = true - return nil - }) } + nic.IPConfigurations = &newIPConfigs + nicUpdaters = append(nicUpdaters, func() error { + ctx, cancel := getContextWithCancel() + defer cancel() + klog.V(2).Infof("EnsureBackendPoolDeleted begins to CreateOrUpdate for NIC(%s, %s) with backendPoolIDs %q", as.ResourceGroup, pointer.StringDeref(nic.Name, ""), backendPoolIDs) + rerr := as.InterfacesClient.CreateOrUpdate(ctx, as.ResourceGroup, pointer.StringDeref(nic.Name, ""), nic) + if rerr != nil { + klog.Errorf("EnsureBackendPoolDeleted CreateOrUpdate for NIC(%s, %s) failed with error %v", as.ResourceGroup, pointer.StringDeref(nic.Name, ""), rerr.Error()) + return rerr.Error() + } + nicUpdated = true + return nil + }) } errs := utilerrors.AggregateGoroutines(nicUpdaters...) if errs != nil { @@ -1310,7 +1345,7 @@ func (as *availabilitySet) GetNodeCIDRMasksByProviderID(providerID string) (int, } // EnsureBackendPoolDeletedFromVMSets ensures the loadBalancer backendAddressPools deleted from the specified VMAS -func (as *availabilitySet) EnsureBackendPoolDeletedFromVMSets(vmasNamesMap map[string]bool, backendPoolID string) error { +func (as *availabilitySet) EnsureBackendPoolDeletedFromVMSets(vmasNamesMap map[string]bool, backendPoolIDs []string) error { return nil } diff --git a/pkg/provider/azure_standard_test.go b/pkg/provider/azure_standard_test.go index 071c8ecca1..e1719eb259 100644 --- a/pkg/provider/azure_standard_test.go +++ b/pkg/provider/azure_standard_test.go @@ -33,6 +33,7 @@ import ( "k8s.io/apimachinery/pkg/types" "k8s.io/apimachinery/pkg/util/sets" cloudprovider "k8s.io/cloud-provider" + utilnet "k8s.io/utils/net" "k8s.io/utils/pointer" "sigs.k8s.io/cloud-provider-azure/pkg/azureclients/interfaceclient/mockinterfaceclient" @@ -376,22 +377,22 @@ func TestGetLoadBalancingRuleName(t *testing.T) { expected: "a257b965551374ad2b091ef3f07043ad-shortsubnet-TCP-9000", }, { - description: "internal standard lb should have subnet name on the rule name but truncated to 80 characters", + description: "internal standard lb should have subnet name on the rule name but truncated to 80 (-5) characters", subnetName: "averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggggggggggggggggggggggggggggggggggsubet", isInternal: true, useStandardLB: true, protocol: v1.ProtocolTCP, port: 9000, - expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnngg-TCP-9000", + expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnn-TCP-9000", }, { - description: "internal basic lb should have subnet name on the rule name but truncated to 80 characters", + description: "internal basic lb should have subnet name on the rule name but truncated to 80 (-5) characters", subnetName: "averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggggggggggggggggggggggggggggggggggsubet", isInternal: true, useStandardLB: false, protocol: v1.ProtocolTCP, port: 9000, - expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnngg-TCP-9000", + expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnn-TCP-9000", }, { description: "external standard lb should not have subnet name on the rule name", @@ -422,7 +423,8 @@ func TestGetLoadBalancingRuleName(t *testing.T) { svc.Annotations[consts.ServiceAnnotationLoadBalancerInternalSubnet] = c.subnetName svc.Annotations[consts.ServiceAnnotationLoadBalancerInternal] = strconv.FormatBool(c.isInternal) - loadbalancerRuleName := az.getLoadBalancerRuleName(svc, c.protocol, c.port) + isIPv6 := utilnet.IsIPv6String(svc.Spec.ClusterIP) + loadbalancerRuleName := az.getLoadBalancerRuleName(svc, c.protocol, c.port, isIPv6) assert.Equal(t, c.expected, loadbalancerRuleName, c.description) } } @@ -462,21 +464,21 @@ func TestGetFrontendIPConfigName(t *testing.T) { subnetName: "a--------------------------------------------------z", isInternal: true, useStandardLB: true, - expected: "a257b965551374ad2b091ef3f07043ad-a---------------------------------------------_", + expected: "a257b965551374ad2b091ef3f07043ad-a----------------------------------------_", }, { description: "internal standard lb should have subnet name on the frontend ip configuration name but truncated to 80 characters", subnetName: "averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggggggggggggggggggggggggggggggggggsubet", isInternal: true, useStandardLB: true, - expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnnggggggggggg", + expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggg", }, { description: "internal basic lb should have subnet name on the frontend ip configuration name but truncated to 80 characters", subnetName: "averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggggggggggggggggggggggggggggggggggsubet", isInternal: true, useStandardLB: false, - expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnnggggggggggg", + expected: "a257b965551374ad2b091ef3f07043ad-averylonnnngggnnnnnnnnnnnnnnnnnnnnnngggggg", }, { description: "external standard lb should not have subnet name on the frontend ip configuration name", @@ -902,26 +904,45 @@ func TestGetBackendPoolName(t *testing.T) { service v1.Service clusterName string expectedPoolName string + isIPv6 bool }{ { name: "GetBackendPoolName should return -IPv6", service: getTestService("test1", v1.ProtocolTCP, nil, true, 80), clusterName: "azure", expectedPoolName: "azure-IPv6", + isIPv6: true, }, { name: "GetBackendPoolName should return ", service: getTestService("test1", v1.ProtocolTCP, nil, false, 80), clusterName: "azure", expectedPoolName: "azure", + isIPv6: false, }, } for _, test := range testcases { - backPoolName := getBackendPoolName(test.clusterName, &test.service) + backPoolName := getBackendPoolName(test.clusterName, test.isIPv6) assert.Equal(t, test.expectedPoolName, backPoolName, test.name) } } +func TestIsBackendPoolIPv6(t *testing.T) { + testcases := []struct { + name string + expectedIsIPv6 bool + }{ + {"bp-IPv6", true}, + {"bp-IPv4", false}, + {"bp", false}, + {"bp-ipv6", false}, + } + for _, test := range testcases { + isIPv6 := isBackendPoolIPv6(test.name) + assert.Equal(t, test.expectedIsIPv6, isIPv6) + } +} + func TestGetStandardInstanceIDByNodeName(t *testing.T) { ctrl := gomock.NewController(t) defer ctrl.Finish() @@ -1403,15 +1424,6 @@ func TestStandardEnsureHostInPool(t *testing.T) { nicProvisionState: consts.NicFailedState, vmSetName: "myAvailabilitySet", }, - { - name: "EnsureHostInPool should report error if service.Spec.ClusterIP is ipv6 but node don't have IPv6 address", - service: &v1.Service{Spec: v1.ServiceSpec{ClusterIP: "2001:0db8:85a3:0000:0000:8a2e:0370:7334"}}, - nodeName: "vm4", - nicName: "nic4", - nicID: "/subscriptions/sub/resourceGroups/rg/providers/Microsoft.Network/networkInterfaces/nic4", - vmSetName: "myAvailabilitySet", - expectedErrMsg: fmt.Errorf("failed to determine the ipconfig(IPv6=true). nicname=%q", "nic4"), - }, { name: "EnsureHostInPool should return nil if there is matched backend pool", service: &v1.Service{}, @@ -1571,64 +1583,39 @@ func TestStandardEnsureHostsInPool(t *testing.T) { nicID: "/subscriptions/sub/resourceGroups/rg/providers/Microsoft.Network/networkInterfaces/nic4", vmSetName: "availabilityset-1", }, - { - name: "EnsureHostsInPool should report error if service.Spec.ClusterIP is ipv6 but node don't have IPv6 address", - service: &v1.Service{ - ObjectMeta: meta.ObjectMeta{ - Name: "svc", - Namespace: "default", - }, - Spec: v1.ServiceSpec{ - ClusterIP: "2001:0db8:85a3:0000:0000:8a2e:0370:7334", - }, - }, - nodeName: "vm5", - nodes: []*v1.Node{ - { - ObjectMeta: meta.ObjectMeta{ - Name: "vm5", - }, - }, - }, - nicName: "nic5", - backendPoolID: backendAddressPoolID, - nicID: "/subscriptions/sub/resourceGroups/rg/providers/Microsoft.Network/networkInterfaces/nic5", - vmSetName: "myAvailabilitySet", - expectedErr: true, - expectedErrMsg: fmt.Errorf("ensure(default/svc): backendPoolID(%s) - failed to ensure host in pool: %w", backendAddressPoolID, fmt.Errorf("failed to determine the ipconfig(IPv6=true). nicname=%q", "nic5")), - }, } for _, test := range testCases { - cloud.Config.LoadBalancerSku = consts.LoadBalancerSkuStandard - cloud.Config.ExcludeMasterFromStandardLB = pointer.Bool(true) - cloud.excludeLoadBalancerNodes = sets.New(test.excludeLBNodes...) + t.Run(test.name, func(t *testing.T) { + cloud.Config.LoadBalancerSku = consts.LoadBalancerSkuStandard + cloud.Config.ExcludeMasterFromStandardLB = pointer.Bool(true) + cloud.excludeLoadBalancerNodes = sets.New(test.excludeLBNodes...) - testVM := buildDefaultTestVirtualMachine(availabilitySetID, []string{test.nicID}) - testNIC := buildDefaultTestInterface(false, []string{backendAddressPoolID}) - testNIC.Name = pointer.String(test.nicName) - testNIC.ID = pointer.String(test.nicID) + testVM := buildDefaultTestVirtualMachine(availabilitySetID, []string{test.nicID}) + testNIC := buildDefaultTestInterface(false, []string{backendAddressPoolID}) + testNIC.Name = pointer.String(test.nicName) + testNIC.ID = pointer.String(test.nicID) - mockVMClient := cloud.VirtualMachinesClient.(*mockvmclient.MockInterface) - mockVMClient.EXPECT().Get(gomock.Any(), cloud.ResourceGroup, test.nodeName, gomock.Any()).Return(testVM, nil).AnyTimes() + mockVMClient := cloud.VirtualMachinesClient.(*mockvmclient.MockInterface) + mockVMClient.EXPECT().Get(gomock.Any(), cloud.ResourceGroup, test.nodeName, gomock.Any()).Return(testVM, nil).AnyTimes() - mockInterfaceClient := cloud.InterfacesClient.(*mockinterfaceclient.MockInterface) - mockInterfaceClient.EXPECT().Get(gomock.Any(), cloud.ResourceGroup, test.nicName, gomock.Any()).Return(testNIC, nil).AnyTimes() - mockInterfaceClient.EXPECT().CreateOrUpdate(gomock.Any(), cloud.ResourceGroup, gomock.Any(), gomock.Any()).Return(nil).AnyTimes() + mockInterfaceClient := cloud.InterfacesClient.(*mockinterfaceclient.MockInterface) + mockInterfaceClient.EXPECT().Get(gomock.Any(), cloud.ResourceGroup, test.nicName, gomock.Any()).Return(testNIC, nil).AnyTimes() + mockInterfaceClient.EXPECT().CreateOrUpdate(gomock.Any(), cloud.ResourceGroup, gomock.Any(), gomock.Any()).Return(nil).AnyTimes() - err := cloud.VMSet.EnsureHostsInPool(test.service, test.nodes, test.backendPoolID, test.vmSetName) - if test.expectedErr { - assert.EqualError(t, test.expectedErrMsg, err.Error(), test.name) - } else { - assert.Nil(t, err, test.name) - } + err := cloud.VMSet.EnsureHostsInPool(test.service, test.nodes, test.backendPoolID, test.vmSetName) + if test.expectedErr { + assert.EqualError(t, test.expectedErrMsg, err.Error(), test.name) + } else { + assert.Nil(t, err, test.name) + } + }) } } func TestServiceOwnsFrontendIP(t *testing.T) { ctrl := gomock.NewController(t) defer ctrl.Finish() - cloud := GetTestCloud(ctrl) testCases := []struct { desc string @@ -1735,8 +1722,10 @@ func TestServiceOwnsFrontendIP(t *testing.T) { }, service: &v1.Service{ ObjectMeta: meta.ObjectMeta{ - UID: types.UID("secondary"), - Annotations: map[string]string{consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1"}, + UID: types.UID("secondary"), + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1", + }, }, }, }, @@ -1760,12 +1749,56 @@ func TestServiceOwnsFrontendIP(t *testing.T) { }, service: &v1.Service{ ObjectMeta: meta.ObjectMeta{ - UID: types.UID("secondary"), - Annotations: map[string]string{consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1"}, + UID: types.UID("secondary"), + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1", + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "fd00::eef0", + }, }, }, isOwned: true, }, + { + desc: "serviceOwnsFrontendIP should detect the secondary external service dual-stack", + existingPIPs: []network.PublicIPAddress{ + { + ID: pointer.String("pip"), + PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{ + PublicIPAddressVersion: network.IPv4, + IPAddress: pointer.String("4.3.2.1"), + }, + }, + { + ID: pointer.String("pip1"), + PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{ + PublicIPAddressVersion: network.IPv6, + IPAddress: pointer.String("fd00::eef0"), + }, + }, + }, + fip: network.FrontendIPConfiguration{ + Name: pointer.String("auid"), + FrontendIPConfigurationPropertiesFormat: &network.FrontendIPConfigurationPropertiesFormat{ + PublicIPAddress: &network.PublicIPAddress{ + PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{ + PublicIPAddressVersion: network.IPv6, + }, + ID: pointer.String("pip1"), + }, + }, + }, + service: &v1.Service{ + ObjectMeta: meta.ObjectMeta{ + UID: types.UID("secondary"), + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1", + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "fd00::eef0", + }, + }, + }, + isOwned: true, + isPrimary: false, + }, { desc: "serviceOwnsFrontendIP should detect the secondary internal service", fip: network.FrontendIPConfiguration{ @@ -1785,13 +1818,36 @@ func TestServiceOwnsFrontendIP(t *testing.T) { }, isOwned: true, }, + { + desc: "serviceOwnsFrontendIP should detect the secondary internal service - dualstack", + fip: network.FrontendIPConfiguration{ + Name: pointer.String("auid"), + FrontendIPConfigurationPropertiesFormat: &network.FrontendIPConfigurationPropertiesFormat{ + PrivateIPAddress: pointer.String("fd00::eef0"), + }, + }, + service: &v1.Service{ + ObjectMeta: meta.ObjectMeta{ + UID: types.UID("secondary"), + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerInternal: "true", + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "4.3.2.1", + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "fd00::eef0", + }, + }, + }, + isOwned: true, + }, } for _, test := range testCases { - isOwned, isPrimary, err := cloud.serviceOwnsFrontendIP(test.fip, test.service, &test.existingPIPs) - assert.Equal(t, test.expectedErr, err, test.desc) - assert.Equal(t, test.isOwned, isOwned, test.desc) - assert.Equal(t, test.isPrimary, isPrimary, test.desc) + t.Run(test.desc, func(t *testing.T) { + cloud := GetTestCloud(ctrl) + isOwned, isPrimary, err := cloud.serviceOwnsFrontendIP(test.fip, test.service, &test.existingPIPs) + assert.Equal(t, test.expectedErr, err) + assert.Equal(t, test.isOwned, isOwned) + assert.Equal(t, test.isPrimary, isPrimary) + }) } } @@ -1844,7 +1900,7 @@ func TestStandardEnsureBackendPoolDeleted(t *testing.T) { mockNICClient.EXPECT().CreateOrUpdate(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(nil) cloud.InterfacesClient = mockNICClient - nicUpdated, err := cloud.VMSet.EnsureBackendPoolDeleted(&service, backendPoolID, vmSetName, test.backendAddressPools, true) + nicUpdated, err := cloud.VMSet.EnsureBackendPoolDeleted(&service, []string{backendPoolID}, vmSetName, test.backendAddressPools, true) assert.NoError(t, err, test.desc) assert.True(t, nicUpdated) } @@ -2187,6 +2243,7 @@ func TestGetSecurityRuleName(t *testing.T) { svc *v1.Service port v1.ServicePort sourceAddrPrefix string + isIPv6 bool expectedRuleName string }{ { @@ -2201,6 +2258,7 @@ func TestGetSecurityRuleName(t *testing.T) { Port: 80, }, "10.0.0.1/24", + false, "a257b965551374ad2b091ef3f07043ad-TCP-80-10.0.0.1_24", }, { @@ -2216,6 +2274,7 @@ func TestGetSecurityRuleName(t *testing.T) { Port: 80, }, "10.0.0.1/24", + false, "shared-TCP-80-10.0.0.1_24", }, { @@ -2230,6 +2289,7 @@ func TestGetSecurityRuleName(t *testing.T) { Port: 80, }, "2001:0:0::1/64", + true, "a257b965551374ad2b091ef3f07043ad-TCP-80-2001.0.0..1_64", }, } @@ -2239,7 +2299,7 @@ func TestGetSecurityRuleName(t *testing.T) { az := GetTestCloud(ctrl) for _, tc := range testcases { t.Run(tc.desc, func(t *testing.T) { - ruleName := az.getSecurityRuleName(tc.svc, tc.port, tc.sourceAddrPrefix) + ruleName := az.getSecurityRuleName(tc.svc, tc.port, tc.sourceAddrPrefix, tc.isIPv6) assert.Equal(t, tc.expectedRuleName, ruleName) }) } diff --git a/pkg/provider/azure_test.go b/pkg/provider/azure_test.go index 39438ee0cb..29bee80a06 100644 --- a/pkg/provider/azure_test.go +++ b/pkg/provider/azure_test.go @@ -39,6 +39,7 @@ import ( cloudprovider "k8s.io/cloud-provider" cloudproviderapi "k8s.io/cloud-provider/api" servicehelpers "k8s.io/cloud-provider/service/helpers" + utilnet "k8s.io/utils/net" "k8s.io/utils/pointer" "sigs.k8s.io/cloud-provider-azure/pkg/azureclients/interfaceclient/mockinterfaceclient" @@ -727,7 +728,8 @@ func TestReconcileSecurityGroupFromAnyDestinationAddressPrefixToLoadBalancerIP(t if err != nil { t.Errorf("Unexpected error: %q", err) } - sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error: %q", err) } @@ -1662,6 +1664,12 @@ func getBackendPort(port int32) int32 { return port + 10000 } +// TODO: This function should be merged into getTestService() +func makeTestServiceDualStack(svc *v1.Service) { + svc.Spec.ClusterIPs = []string{"10.0.0.2", "fd00::1907"} + svc.Spec.IPFamilies = []v1.IPFamily{v1.IPv4Protocol, v1.IPv6Protocol} +} + func getTestService(identifier string, proto v1.Protocol, annotations map[string]string, isIPv6 bool, requestedPorts ...int32) v1.Service { ports := []v1.ServicePort{} for _, port := range requestedPorts { @@ -1689,8 +1697,10 @@ func getTestService(identifier string, proto v1.Protocol, annotations map[string } svc.Spec.ClusterIP = "10.0.0.2" + svc.Spec.IPFamilies = []v1.IPFamily{v1.IPv4Protocol} if isIPv6 { svc.Spec.ClusterIP = "fd00::1907" + svc.Spec.IPFamilies = []v1.IPFamily{v1.IPv6Protocol} } return svc @@ -1740,7 +1750,8 @@ func getTestSecurityGroup(az *Cloud, services ...v1.Service) *network.SecurityGr for _, port := range service.Spec.Ports { sources := getServiceSourceRanges(&services[i]) for _, src := range sources { - ruleName := az.getSecurityRuleName(&services[i], port, src) + isIPv6 := utilnet.IsIPv6String(services[i].Spec.ClusterIP) + ruleName := az.getSecurityRuleName(&services[i], port, src, isIPv6) rules = append(rules, network.SecurityRule{ Name: pointer.String(ruleName), SecurityRulePropertiesFormat: &network.SecurityRulePropertiesFormat{ @@ -1790,7 +1801,8 @@ func validateLoadBalancer(t *testing.T, loadBalancer *network.LoadBalancer, serv } for _, wantedRule := range svc.Spec.Ports { expectedRuleCount++ - wantedRuleName := az.getLoadBalancerRuleName(&services[i], wantedRule.Protocol, wantedRule.Port) + isIPv6 := utilnet.IsIPv6String(services[i].Spec.ClusterIP) + wantedRuleName := az.getLoadBalancerRuleName(&services[i], wantedRule.Protocol, wantedRule.Port, isIPv6) foundRule := false for _, actualRule := range *loadBalancer.LoadBalancingRules { if strings.EqualFold(*actualRule.Name, wantedRuleName) && @@ -1813,7 +1825,8 @@ func validateLoadBalancer(t *testing.T, loadBalancer *network.LoadBalancer, serv foundProbe := false if servicehelpers.NeedsHealthCheck(&services[i]) { path, port := servicehelpers.GetServiceHealthCheckPathPort(&services[i]) - wantedRuleName := az.getLoadBalancerRuleName(&services[i], v1.ProtocolTCP, port) + isIPv6 := utilnet.IsIPv6String(services[i].Spec.ClusterIP) + wantedRuleName := az.getLoadBalancerRuleName(&services[i], v1.ProtocolTCP, port, isIPv6) for _, actualProbe := range *loadBalancer.Probes { if strings.EqualFold(*actualProbe.Name, wantedRuleName) && *actualProbe.Port == port && @@ -2001,12 +2014,13 @@ func validateSecurityGroup(t *testing.T, securityGroup *network.SecurityGroup, s for _, wantedRule := range svc.Spec.Ports { sources := getServiceSourceRanges(&services[i]) for _, source := range sources { - wantedRuleName := az.getSecurityRuleName(&services[i], wantedRule, source) + isIPv6 := utilnet.IsIPv6String(services[i].Spec.ClusterIP) + wantedRuleName := az.getSecurityRuleName(&services[i], wantedRule, source, isIPv6) seenRules[wantedRuleName] = wantedRuleName foundRule := false for _, actualRule := range *securityGroup.SecurityRules { if strings.EqualFold(*actualRule.Name, wantedRuleName) { - err := securityRuleMatches(source, wantedRule, getServiceLoadBalancerIP(&svc), actualRule) + err := securityRuleMatches(source, wantedRule, getServiceLoadBalancerIP(&svc, isIPv6), actualRule) if err != nil { t.Errorf("Found matching security rule %q but properties were incorrect: %v", wantedRuleName, err) } @@ -2452,7 +2466,8 @@ func TestIfServiceSpecifiesSharedRuleAndRuleDoesNotExistItIsCreated(t *testing.T sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc, pointer.String(getServiceLoadBalancerIP(&svc)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc, pointer.String(getServiceLoadBalancerIP(&svc, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error: %q", err) } @@ -2511,7 +2526,8 @@ func TestIfServiceSpecifiesSharedRuleAndRuleExistsThenTheServicesPortAndAddressA } setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc, pointer.String(getServiceLoadBalancerIP(&svc)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc, pointer.String(getServiceLoadBalancerIP(&svc, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error: %q", err) } @@ -2555,13 +2571,15 @@ func TestIfServicesSpecifySharedRuleButDifferentPortsThenSeparateRulesAreCreated sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } @@ -2628,13 +2646,15 @@ func TestIfServicesSpecifySharedRuleButDifferentProtocolsThenSeparateRulesAreCre sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } @@ -2701,13 +2721,15 @@ func TestIfServicesSpecifySharedRuleButDifferentSourceAddressesThenSeparateRules sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } @@ -2776,19 +2798,22 @@ func TestIfServicesSpecifySharedRuleButSomeAreOnDifferentPortsThenRulesAreSepara sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc3.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc3: %q", err) } @@ -2876,13 +2901,15 @@ func TestIfServiceSpecifiesSharedRuleAndServiceIsDeletedThenTheServicesPortAndAd sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } @@ -2890,7 +2917,8 @@ func TestIfServiceSpecifiesSharedRuleAndServiceIsDeletedThenTheServicesPortAndAd validateSecurityGroup(t, sg, svc1, svc2) setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc1: %q", err) } @@ -2939,19 +2967,22 @@ func TestIfSomeServicesShareARuleAndOneIsDeletedItIsRemovedFromTheRightRule(t *t sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc3.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc3: %q", err) } @@ -2959,7 +2990,8 @@ func TestIfSomeServicesShareARuleAndOneIsDeletedItIsRemovedFromTheRightRule(t *t validateSecurityGroup(t, sg, svc1, svc2, svc3) setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc1: %q", err) } @@ -3050,19 +3082,22 @@ func TestIfServiceSpecifiesSharedRuleAndLastServiceIsDeletedThenRuleIsDeleted(t sg := getTestSecurityGroup(az) setMockSecurityGroup(az, ctrl, sg) - sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err := az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc2.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc2, pointer.String(getServiceLoadBalancerIP(&svc2, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc2: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3)), nil, true) + isIPv6 = utilnet.IsIPv6String(svc3.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc3: %q", err) } @@ -3070,13 +3105,15 @@ func TestIfServiceSpecifiesSharedRuleAndLastServiceIsDeletedThenRuleIsDeleted(t validateSecurityGroup(t, sg, svc1, svc2, svc3) setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc3.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc3, pointer.String(getServiceLoadBalancerIP(&svc3, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc3: %q", err) } @@ -3143,15 +3180,18 @@ func TestCanCombineSharedAndPrivateRulesInSameGroup(t *testing.T) { testServices := []v1.Service{svc1, svc2, svc3, svc4, svc5} testRuleName23 := testRuleName2 - expectedRuleName4 := az.getSecurityRuleName(&svc4, v1.ServicePort{Port: 4444, Protocol: v1.ProtocolTCP}, "Internet") - expectedRuleName5 := az.getSecurityRuleName(&svc5, v1.ServicePort{Port: 8888, Protocol: v1.ProtocolTCP}, "Internet") + isIPv6 := utilnet.IsIPv6String(svc4.Spec.ClusterIP) + expectedRuleName4 := az.getSecurityRuleName(&svc4, v1.ServicePort{Port: 4444, Protocol: v1.ProtocolTCP}, "Internet", isIPv6) + isIPv6 = utilnet.IsIPv6String(svc5.Spec.ClusterIP) + expectedRuleName5 := az.getSecurityRuleName(&svc5, v1.ServicePort{Port: 8888, Protocol: v1.ProtocolTCP}, "Internet", isIPv6) sg := getTestSecurityGroup(az) for i, svc := range testServices { svc := svc setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &testServices[i], pointer.String(getServiceLoadBalancerIP(&svc)), nil, true) + isIPv6 := utilnet.IsIPv6String(svc.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &testServices[i], pointer.String(getServiceLoadBalancerIP(&svc, isIPv6)), nil, true) if err != nil { t.Errorf("Unexpected error adding svc%d: %q", i+1, err) } @@ -3226,8 +3266,9 @@ func TestCanCombineSharedAndPrivateRulesInSameGroup(t *testing.T) { if securityRule4.DestinationAddressPrefix == nil { t.Errorf("Expected unshared rule %s to have a destination IP address", expectedRuleName4) } else { - if !strings.EqualFold(*securityRule4.DestinationAddressPrefix, getServiceLoadBalancerIP(&svc4)) { - t.Errorf("Expected unshared rule %s to have a destination %s but had %s", expectedRuleName4, getServiceLoadBalancerIP(&svc4), *securityRule4.DestinationAddressPrefix) + isIPv6 := utilnet.IsIPv6String(svc4.Spec.ClusterIP) + if !strings.EqualFold(*securityRule4.DestinationAddressPrefix, getServiceLoadBalancerIP(&svc4, isIPv6)) { + t.Errorf("Expected unshared rule %s to have a destination %s but had %s", expectedRuleName4, getServiceLoadBalancerIP(&svc4, isIPv6), *securityRule4.DestinationAddressPrefix) } } @@ -3238,19 +3279,22 @@ func TestCanCombineSharedAndPrivateRulesInSameGroup(t *testing.T) { if securityRule5.DestinationAddressPrefix == nil { t.Errorf("Expected unshared rule %s to have a destination IP address", expectedRuleName5) } else { - if !strings.EqualFold(*securityRule5.DestinationAddressPrefix, getServiceLoadBalancerIP(&svc5)) { - t.Errorf("Expected unshared rule %s to have a destination %s but had %s", expectedRuleName5, getServiceLoadBalancerIP(&svc5), *securityRule5.DestinationAddressPrefix) + isIPv6 := utilnet.IsIPv6String(svc5.Spec.ClusterIP) + if !strings.EqualFold(*securityRule5.DestinationAddressPrefix, getServiceLoadBalancerIP(&svc5, isIPv6)) { + t.Errorf("Expected unshared rule %s to have a destination %s but had %s", expectedRuleName5, getServiceLoadBalancerIP(&svc5, isIPv6), *securityRule5.DestinationAddressPrefix) } } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc1.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc1, pointer.String(getServiceLoadBalancerIP(&svc1, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc1: %q", err) } setMockSecurityGroup(az, ctrl, sg) - sg, err = az.reconcileSecurityGroup(testClusterName, &svc5, pointer.String(getServiceLoadBalancerIP(&svc5)), nil, false) + isIPv6 = utilnet.IsIPv6String(svc5.Spec.ClusterIP) + sg, err = az.reconcileSecurityGroup(testClusterName, &svc5, pointer.String(getServiceLoadBalancerIP(&svc5, isIPv6)), nil, false) if err != nil { t.Errorf("Unexpected error removing svc5: %q", err) } diff --git a/pkg/provider/azure_utils.go b/pkg/provider/azure_utils.go index 4cf8ae561a..c0fa91397a 100644 --- a/pkg/provider/azure_utils.go +++ b/pkg/provider/azure_utils.go @@ -227,11 +227,10 @@ func getServiceAdditionalPublicIPs(service *v1.Service) ([]string, error) { return result, nil } -func getNodePrivateIPAddress(service *v1.Service, node *v1.Node) string { - isIPV6SVC := utilnet.IsIPv6String(service.Spec.ClusterIP) +func getNodePrivateIPAddress(node *v1.Node, isIPv6 bool) string { for _, nodeAddress := range node.Status.Addresses { if strings.EqualFold(string(nodeAddress.Type), string(v1.NodeInternalIP)) && - utilnet.IsIPv6String(nodeAddress.Address) == isIPV6SVC { + utilnet.IsIPv6String(nodeAddress.Address) == isIPv6 { klog.V(6).Infof("getNodePrivateIPAddress: node %s, ip %s", node.Name, nodeAddress.Address) return nodeAddress.Address } @@ -342,3 +341,152 @@ func extractVmssVMName(name string) (string, string, error) { instanceID := split[len(split)-1] return ssName, instanceID, nil } + +// getIPFamiliesEnabled checks if IPv4, IPv6 are enabled according to svc.Spec.IPFamilies. +func getIPFamiliesEnabled(svc *v1.Service) (v4Enabled bool, v6Enabled bool) { + for _, ipFamily := range svc.Spec.IPFamilies { + if ipFamily == v1.IPv4Protocol { + v4Enabled = true + } else if ipFamily == v1.IPv6Protocol { + v6Enabled = true + } + } + return +} + +// getServiceLoadBalancerIP retrieves LB IP from IPv4 annotation, then IPv6 annotation, then service.Spec.LoadBalancerIP. +func getServiceLoadBalancerIP(service *v1.Service, isIPv6 bool) string { + if service == nil { + return "" + } + + if ip, ok := service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[isIPv6]]; ok && ip != "" { + return ip + } + + // Retrieve LB IP from service.Spec.LoadBalancerIP (will be deprecated) + svcLBIP := service.Spec.LoadBalancerIP + if (net.ParseIP(svcLBIP).To4() != nil && !isIPv6) || + (net.ParseIP(svcLBIP).To4() == nil && isIPv6) { + return svcLBIP + } + return "" +} + +func getServiceLoadBalancerIPs(service *v1.Service) []string { + if service == nil { + return []string{} + } + + ips := []string{} + if ip, ok := service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[false]]; ok && ip != "" { + ips = append(ips, ip) + } + if ip, ok := service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[true]]; ok && ip != "" { + ips = append(ips, ip) + } + if len(ips) != 0 { + return ips + } + + lbIP := service.Spec.LoadBalancerIP + if lbIP != "" { + ips = append(ips, lbIP) + } + + return ips +} + +// setServiceLoadBalancerIP sets LB IP to a Service +func setServiceLoadBalancerIP(service *v1.Service, ip string) { + if service.Annotations == nil { + service.Annotations = map[string]string{} + } + isIPv6 := net.ParseIP(ip).To4() == nil + service.Annotations[consts.ServiceAnnotationLoadBalancerIPDualStack[isIPv6]] = ip +} + +func getServicePIPName(service *v1.Service, isIPv6 bool) string { + if service == nil { + return "" + } + + if consts.DualstackSupported { + if name, ok := service.Annotations[consts.ServiceAnnotationPIPNameDualStack[isIPv6]]; ok && name != "" { + return name + } + } + return service.Annotations[consts.ServiceAnnotationPIPName] +} + +func getServicePIPPrefixID(service *v1.Service, isIPv6 bool) string { + if service == nil { + return "" + } + + if consts.DualstackSupported { + if name, ok := service.Annotations[consts.ServiceAnnotationPIPPrefixIDDualStack[isIPv6]]; ok && name != "" { + return name + } + } + return service.Annotations[consts.ServiceAnnotationPIPPrefixID] +} + +func getResourceByIPFamily(resource string, isIPv6 bool) string { + if !consts.DualstackSupported { + return resource + } + + if isIPv6 { + return fmt.Sprintf("%s-%s", resource, v6Suffix) + } + return fmt.Sprintf("%s-%s", resource, v4Suffix) +} + +// isFIPIPv6 checks if the frontend IP configuration is of IPv6. +func (az *Cloud) isFIPIPv6(fip *network.FrontendIPConfiguration, pips *[]network.PublicIPAddress, isInternal bool) (isIPv6 bool, err error) { + if err := az.safeListPIP(az.ResourceGroup, pips); err != nil { + return false, fmt.Errorf("failed to ensure PIP is refreshed: %w", err) + } + if isInternal { + if fip.FrontendIPConfigurationPropertiesFormat != nil { + if fip.FrontendIPConfigurationPropertiesFormat.PrivateIPAddressVersion != "" { + return fip.FrontendIPConfigurationPropertiesFormat.PrivateIPAddressVersion == network.IPv6, nil + } + return net.ParseIP(pointer.StringDeref(fip.FrontendIPConfigurationPropertiesFormat.PrivateIPAddress, "")).To4() == nil, nil + } + klog.Errorf("Checking IP Family of frontend IP configuration %q of internal Service but its"+ + " FrontendIPConfigurationPropertiesFormat is nil. It's considered to be IPv4", + pointer.StringDeref(fip.Name, "")) + return + } + var fipPIPID string + if fip.FrontendIPConfigurationPropertiesFormat != nil && fip.FrontendIPConfigurationPropertiesFormat.PublicIPAddress != nil { + fipPIPID = pointer.StringDeref(fip.FrontendIPConfigurationPropertiesFormat.PublicIPAddress.ID, "") + } + for _, pip := range *pips { + id := pointer.StringDeref(pip.ID, "") + if !strings.EqualFold(fipPIPID, id) { + continue + } + if pip.PublicIPAddressPropertiesFormat != nil { + // First check PublicIPAddressVersion, then IPAddress + if pip.PublicIPAddressPropertiesFormat.PublicIPAddressVersion == network.IPv6 || + net.ParseIP(pointer.StringDeref(pip.PublicIPAddressPropertiesFormat.IPAddress, "")).To4() == nil { + isIPv6 = true + break + } + } + break + } + return isIPv6, nil +} + +// getResourceIDPrefix returns a substring from the provided one between beginning and the last "/". +func getResourceIDPrefix(id string) string { + idx := strings.LastIndexByte(id, '/') + if idx == -1 { + return id // Should not happen + } + return id[:idx] +} diff --git a/pkg/provider/azure_utils_test.go b/pkg/provider/azure_utils_test.go index afb53b65b6..25f3dc98c2 100644 --- a/pkg/provider/azure_utils_test.go +++ b/pkg/provider/azure_utils_test.go @@ -23,6 +23,7 @@ import ( "time" "github.com/Azure/azure-sdk-for-go/services/network/mgmt/2022-07-01/network" + "github.com/golang/mock/gomock" "github.com/stretchr/testify/assert" v1 "k8s.io/api/core/v1" @@ -261,6 +262,105 @@ func TestGetServiceAdditionalPublicIPs(t *testing.T) { } } +func TestGetNodePrivateIPAddress(t *testing.T) { + testcases := []struct { + desc string + node *v1.Node + isIPv6 bool + expectedIP string + }{ + { + "IPv4", + &v1.Node{ + Status: v1.NodeStatus{ + Addresses: []v1.NodeAddress{ + { + Type: v1.NodeExternalIP, + Address: "10.244.0.1", + }, + { + Type: v1.NodeInternalIP, + Address: "10.0.0.1", + }, + { + Type: v1.NodeInternalIP, + Address: "2001::1", + }, + }, + }, + }, + false, + "10.0.0.1", + }, + { + "IPv6", + &v1.Node{ + Status: v1.NodeStatus{ + Addresses: []v1.NodeAddress{ + { + Type: v1.NodeExternalIP, + Address: "2f00::1", + }, + { + Type: v1.NodeInternalIP, + Address: "10.0.0.1", + }, + { + Type: v1.NodeInternalIP, + Address: "2001::1", + }, + }, + }, + }, + true, + "2001::1", + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + ip := getNodePrivateIPAddress(tc.node, tc.isIPv6) + assert.Equal(t, tc.expectedIP, ip) + }) + } +} + +func TestGetNodePrivateIPAddresses(t *testing.T) { + testcases := []struct { + desc string + node *v1.Node + expetedIPs []string + }{ + { + "default", + &v1.Node{ + Status: v1.NodeStatus{ + Addresses: []v1.NodeAddress{ + { + Type: v1.NodeExternalIP, + Address: "2f00::1", + }, + { + Type: v1.NodeInternalIP, + Address: "10.0.0.1", + }, + { + Type: v1.NodeInternalIP, + Address: "2001::1", + }, + }, + }, + }, + []string{"10.0.0.1", "2001::1"}, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + ips := getNodePrivateIPAddresses(tc.node) + assert.Equal(t, tc.expetedIPs, ips) + }) + } +} + func TestRemoveDuplicatedSecurityRules(t *testing.T) { for _, testCase := range []struct { description string @@ -448,3 +548,294 @@ func TestExtractVmssVMName(t *testing.T) { assert.Equal(t, c.expectedInstanceID, instanceID, c.description) } } + +func TestGetIPFamiliesEnabled(t *testing.T) { + testcases := []struct { + desc string + svc *v1.Service + ExpectedV4Enabled bool + ExpectedV6Enabled bool + }{ + { + "IPv4", + &v1.Service{ + Spec: v1.ServiceSpec{IPFamilies: []v1.IPFamily{v1.IPv4Protocol}}, + }, + true, + false, + }, + { + "DualStack", + &v1.Service{ + Spec: v1.ServiceSpec{IPFamilies: []v1.IPFamily{v1.IPv4Protocol, v1.IPv6Protocol}}, + }, + true, + true, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + v4Enabled, v6Enabled := getIPFamiliesEnabled(tc.svc) + assert.Equal(t, tc.ExpectedV4Enabled, v4Enabled) + assert.Equal(t, tc.ExpectedV6Enabled, v6Enabled) + }) + } +} + +func TestGetServiceLoadBalancerIP(t *testing.T) { + testcases := []struct { + desc string + svc *v1.Service + isIPv6 bool + expectedIP string + }{ + { + "IPv6", + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "10.0.0.1", + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "2001::1", + }, + }, + Spec: v1.ServiceSpec{ + LoadBalancerIP: "10.0.0.2", + }, + }, + true, + "2001::1", + }, + { + "IPv4 but from LoadBalancerIP", + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{}, + }, + Spec: v1.ServiceSpec{ + LoadBalancerIP: "10.0.0.2", + }, + }, + false, + "10.0.0.2", + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + ip := getServiceLoadBalancerIP(tc.svc, tc.isIPv6) + assert.Equal(t, tc.expectedIP, ip) + }) + } +} + +func TestGetServiceLoadBalancerIPs(t *testing.T) { + testcases := []struct { + desc string + svc *v1.Service + expectedIPs []string + }{ + { + "Get IPv4 and IPv6 IPs", + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[false]: "10.0.0.1", + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "2001::1", + }, + }, + }, + []string{"10.0.0.1", "2001::1"}, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + ips := getServiceLoadBalancerIPs(tc.svc) + assert.Equal(t, tc.expectedIPs, ips) + }) + } +} + +func TestSetServiceLoadBalancerIP(t *testing.T) { + testcases := []struct { + desc string + ip string + svc *v1.Service + expectedSvc *v1.Service + }{ + { + "IPv6", + "2001::1", + &v1.Service{}, + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + consts.ServiceAnnotationLoadBalancerIPDualStack[true]: "2001::1", + }, + }, + }, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + setServiceLoadBalancerIP(tc.svc, tc.ip) + assert.Equal(t, tc.expectedSvc, tc.svc) + }) + } +} + +func TestGetServicePIPName(t *testing.T) { + testcases := []struct { + desc string + svc *v1.Service + isIPv6 bool + expectedName string + }{ + // TODO: Add new after DualStack finishes + { + "From ServiceAnnotationPIPName", + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + consts.ServiceAnnotationPIPName: "pip-name", + }, + }, + }, + false, + "pip-name", + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + name := getServicePIPName(tc.svc, tc.isIPv6) + assert.Equal(t, tc.expectedName, name) + }) + } +} + +func TestGetServicePIPPrefixID(t *testing.T) { + testcases := []struct { + desc string + svc *v1.Service + isIPv6 bool + expectedID string + }{ + // TODO: Add new after DualStack finishes + { + "From ServiceAnnotationPIPName", + &v1.Service{ + ObjectMeta: metav1.ObjectMeta{ + Annotations: map[string]string{ + consts.ServiceAnnotationPIPPrefixID: "pip-prefix-id", + }, + }, + }, + false, + "pip-prefix-id", + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + id := getServicePIPPrefixID(tc.svc, tc.isIPv6) + assert.Equal(t, tc.expectedID, id) + }) + } +} + +func TestGetResourceByIPFamily(t *testing.T) { + testcases := []struct { + desc string + resource string + isIPv6 bool + expectedResource string + }{ + // TODO: Add new test after DualStack finishes + { + "Direct", + "resource0", + false, + "resource0", + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + resource := getResourceByIPFamily(tc.resource, tc.isIPv6) + assert.Equal(t, tc.expectedResource, resource) + }) + } +} + +func TestIsFIPIPv6(t *testing.T) { + ctrl := gomock.NewController(t) + defer ctrl.Finish() + + testcases := []struct { + desc string + fip *network.FrontendIPConfiguration + pips *[]network.PublicIPAddress + isInternal bool + expectedIsIPv6 bool + }{ + { + "Internal IPv4", + &network.FrontendIPConfiguration{ + FrontendIPConfigurationPropertiesFormat: &network.FrontendIPConfigurationPropertiesFormat{ + PrivateIPAddressVersion: network.IPv4, + PrivateIPAddress: pointer.String("10.0.0.1"), + }, + }, + &[]network.PublicIPAddress{}, + true, + false, + }, + { + "External IPv6", + &network.FrontendIPConfiguration{ + FrontendIPConfigurationPropertiesFormat: &network.FrontendIPConfigurationPropertiesFormat{ + PublicIPAddress: &network.PublicIPAddress{ID: pointer.String("pip-id0")}, + }, + }, + &[]network.PublicIPAddress{ + { + ID: pointer.String("pip-id0"), + PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{ + PublicIPAddressVersion: network.IPv6, + IPAddress: pointer.String("2001::1"), + }, + }, + { + ID: pointer.String("pip-id1"), + PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{ + PublicIPAddressVersion: network.IPv4, + IPAddress: pointer.String("10.0.0.1"), + }, + }, + }, + false, + true, + }, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + az := GetTestCloud(ctrl) + isIPv6, err := az.isFIPIPv6(tc.fip, tc.pips, tc.isInternal) + assert.Nil(t, err) + assert.Equal(t, tc.expectedIsIPv6, isIPv6) + }) + } +} + +func TestGetResourceIDPrefix(t *testing.T) { + testcases := []struct { + desc string + id string + expectedPrefix string + }{ + {"normal", "a/b/c", "a/b"}, + {"no-slash", "ab", "ab"}, + } + for _, tc := range testcases { + t.Run(tc.desc, func(t *testing.T) { + prefix := getResourceIDPrefix(tc.id) + assert.Equal(t, tc.expectedPrefix, prefix) + }) + } +} diff --git a/pkg/provider/azure_vmsets.go b/pkg/provider/azure_vmsets.go index 432d3c6d34..f5a883f204 100644 --- a/pkg/provider/azure_vmsets.go +++ b/pkg/provider/azure_vmsets.go @@ -71,9 +71,9 @@ type VMSet interface { // participating in the specified LoadBalancer Backend Pool. EnsureHostInPool(service *v1.Service, nodeName types.NodeName, backendPoolID string, vmSetName string) (string, string, string, *compute.VirtualMachineScaleSetVM, error) // EnsureBackendPoolDeleted ensures the loadBalancer backendAddressPools deleted from the specified nodes. - EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) + EnsureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) //EnsureBackendPoolDeletedFromVMSets ensures the loadBalancer backendAddressPools deleted from the specified VMSS/VMAS - EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap map[string]bool, backendPoolID string) error + EnsureBackendPoolDeletedFromVMSets(vmSetNamesMap map[string]bool, backendPoolIDs []string) error // AttachDisk attaches a disk to vm AttachDisk(ctx context.Context, nodeName types.NodeName, diskMap map[string]*AttachDiskOptions) (*azure.Future, error) diff --git a/pkg/provider/azure_vmss.go b/pkg/provider/azure_vmss.go index c439572cc4..4f7491a8d5 100644 --- a/pkg/provider/azure_vmss.go +++ b/pkg/provider/azure_vmss.go @@ -33,7 +33,6 @@ import ( utilerrors "k8s.io/apimachinery/pkg/util/errors" cloudprovider "k8s.io/cloud-provider" "k8s.io/klog/v2" - utilnet "k8s.io/utils/net" "k8s.io/utils/pointer" azcache "sigs.k8s.io/cloud-provider-azure/pkg/cache" @@ -1143,7 +1142,7 @@ func (ss *ScaleSet) EnsureHostInPool(service *v1.Service, nodeName types.NodeNam } var primaryIPConfiguration *compute.VirtualMachineScaleSetIPConfiguration - ipv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + ipv6 := isBackendPoolIPv6(backendPoolID) // Find primary network interface configuration. if !ss.Cloud.ipv6DualStackEnabled && !ipv6 { // Find primary IP configuration. @@ -1317,7 +1316,7 @@ func (ss *ScaleSet) ensureVMSSInPool(service *v1.Service, nodes []*v1.Node, back return err } var primaryIPConfig *compute.VirtualMachineScaleSetIPConfiguration - ipv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + ipv6 := isBackendPoolIPv6(backendPoolID) // Find primary network interface configuration. if !ss.Cloud.ipv6DualStackEnabled && !ipv6 { // Find primary IP configuration. @@ -1583,7 +1582,7 @@ func (ss *ScaleSet) EnsureHostsInPool(service *v1.Service, nodes []*v1.Node, bac // ensureBackendPoolDeletedFromNode ensures the loadBalancer backendAddressPools deleted // from the specified node, which returns (resourceGroup, vmasName, instanceID, vmssVM, error). -func (ss *ScaleSet) ensureBackendPoolDeletedFromNode(nodeName, backendPoolID string) (string, string, string, *compute.VirtualMachineScaleSetVM, error) { +func (ss *ScaleSet) ensureBackendPoolDeletedFromNode(nodeName string, backendPoolIDs []string) (string, string, string, *compute.VirtualMachineScaleSetVM, error) { vm, err := ss.getVmssVM(nodeName, azcache.CacheReadTypeDefault) if err != nil { if errors.Is(err, cloudprovider.InstanceNotFound) { @@ -1621,10 +1620,12 @@ func (ss *ScaleSet) ensureBackendPoolDeletedFromNode(nodeName, backendPoolID str foundPool := false for i := len(existingBackendPools) - 1; i >= 0; i-- { curPool := existingBackendPools[i] - if strings.EqualFold(backendPoolID, *curPool.ID) { - klog.V(10).Infof("ensureBackendPoolDeletedFromNode gets unwanted backend pool %q for node %s", backendPoolID, nodeName) - foundPool = true - newBackendPools = append(existingBackendPools[:i], existingBackendPools[i+1:]...) + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(backendPoolID, *curPool.ID) { + klog.V(10).Infof("ensureBackendPoolDeletedFromNode gets unwanted backend pool %q for node %s", backendPoolID, nodeName) + foundPool = true + newBackendPools = append(existingBackendPools[:i], existingBackendPools[i+1:]...) + } } } @@ -1704,7 +1705,7 @@ func getScaleSetAndResourceGroupNameByIPConfigurationID(ipConfigurationID string return scaleSetName, resourceGroup, nil } -func (ss *ScaleSet) ensureBackendPoolDeletedFromVMSS(backendPoolID, vmSetName string) error { +func (ss *ScaleSet) ensureBackendPoolDeletedFromVMSS(backendPoolIDs []string, vmSetName string) error { if !ss.useStandardLoadBalancer() { found := false @@ -1724,7 +1725,7 @@ func (ss *ScaleSet) ensureBackendPoolDeletedFromVMSS(backendPoolID, vmSetName st return true }) if found { - return ss.ensureBackendPoolDeletedFromVmssUniform(backendPoolID, vmSetName) + return ss.ensureBackendPoolDeletedFromVmssUniform(backendPoolIDs, vmSetName) } flexScaleSet := ss.flexScaleSet.(*FlexScaleSet) @@ -1744,26 +1745,25 @@ func (ss *ScaleSet) ensureBackendPoolDeletedFromVMSS(backendPoolID, vmSetName st }) if found { - return flexScaleSet.ensureBackendPoolDeletedFromVmssFlex(backendPoolID, vmSetName) + return flexScaleSet.ensureBackendPoolDeletedFromVmssFlex(backendPoolIDs, vmSetName) } return cloudprovider.InstanceNotFound - } - err := ss.ensureBackendPoolDeletedFromVmssUniform(backendPoolID, vmSetName) + err := ss.ensureBackendPoolDeletedFromVmssUniform(backendPoolIDs, vmSetName) if err != nil { return err } if ss.EnableVmssFlexNodes { flexScaleSet := ss.flexScaleSet.(*FlexScaleSet) - err = flexScaleSet.ensureBackendPoolDeletedFromVmssFlex(backendPoolID, vmSetName) + err = flexScaleSet.ensureBackendPoolDeletedFromVmssFlex(backendPoolIDs, vmSetName) } return err } -func (ss *ScaleSet) ensureBackendPoolDeletedFromVmssUniform(backendPoolID, vmSetName string) error { - klog.V(2).Infof("ensureBackendPoolDeletedFromVmssUniform: vmSetName (%s), backendPoolID (%s)", vmSetName, backendPoolID) +func (ss *ScaleSet) ensureBackendPoolDeletedFromVmssUniform(backendPoolIDs []string, vmSetName string) error { + klog.V(2).Infof("ensureBackendPoolDeletedFromVmssUniform: vmSetName (%s), backendPoolIDs (%q)", vmSetName, backendPoolIDs) vmssNamesMap := make(map[string]bool) // the standard load balancer supports multiple vmss in its backend while the basic sku doesn't @@ -1818,9 +1818,11 @@ func (ss *ScaleSet) ensureBackendPoolDeletedFromVmssUniform(backendPoolID, vmSet } for _, loadBalancerBackendAddressPool := range loadBalancerBackendAddressPools { klog.V(4).Infof("ensureBackendPoolDeletedFromVMSS: loadBalancerBackendAddressPool (%s) on vmss (%s)", pointer.StringDeref(loadBalancerBackendAddressPool.ID, ""), pointer.StringDeref(vmss.Name, "")) - if strings.EqualFold(pointer.StringDeref(loadBalancerBackendAddressPool.ID, ""), backendPoolID) { - klog.V(4).Infof("ensureBackendPoolDeletedFromVMSS: found vmss %s with backend pool %s, removing it", pointer.StringDeref(vmss.Name, ""), backendPoolID) - vmssNamesMap[pointer.StringDeref(vmss.Name, "")] = true + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(pointer.StringDeref(loadBalancerBackendAddressPool.ID, ""), backendPoolID) { + klog.V(4).Infof("ensureBackendPoolDeletedFromVMSS: found vmss %s with backend pool %s, removing it", pointer.StringDeref(vmss.Name, ""), backendPoolID) + vmssNamesMap[pointer.StringDeref(vmss.Name, "")] = true + } } } @@ -1836,11 +1838,11 @@ func (ss *ScaleSet) ensureBackendPoolDeletedFromVmssUniform(backendPoolID, vmSet vmssNamesMap[vmSetName] = true } - return ss.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, backendPoolID) + return ss.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, backendPoolIDs) } // ensureBackendPoolDeleted ensures the loadBalancer backendAddressPools deleted from the specified nodes. -func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool) (bool, error) { +func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool) (bool, error) { // Returns nil if backend address pools already deleted. if backendAddressPools == nil { return false, nil @@ -1854,13 +1856,15 @@ func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, ipConfigurationIDs := []string{} for _, backendPool := range *backendAddressPools { - if strings.EqualFold(*backendPool.ID, backendPoolID) && backendPool.BackendIPConfigurations != nil { - for _, ipConf := range *backendPool.BackendIPConfigurations { - if ipConf.ID == nil { - continue + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(*backendPool.ID, backendPoolID) && backendPool.BackendIPConfigurations != nil { + for _, ipConf := range *backendPool.BackendIPConfigurations { + if ipConf.ID == nil { + continue + } + + ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) } - - ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) } } } @@ -1869,8 +1873,14 @@ func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, hostUpdates := make([]func() error, 0, len(ipConfigurationIDs)) nodeUpdates := make(map[vmssMetaInfo]map[string]compute.VirtualMachineScaleSetVM) allErrs := make([]error, 0) + visitedIPConfigIDPrefix := map[string]bool{} for i := range ipConfigurationIDs { ipConfigurationID := ipConfigurationIDs[i] + ipConfigurationIDPrefix := getResourceIDPrefix(ipConfigurationID) + if _, ok := visitedIPConfigIDPrefix[ipConfigurationIDPrefix]; ok { + continue + } + visitedIPConfigIDPrefix[ipConfigurationIDPrefix] = true var scaleSetName string var err error @@ -1898,10 +1908,10 @@ func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, continue } - nodeResourceGroup, nodeVMSS, nodeInstanceID, nodeVMSSVM, err := ss.ensureBackendPoolDeletedFromNode(nodeName, backendPoolID) + nodeResourceGroup, nodeVMSS, nodeInstanceID, nodeVMSSVM, err := ss.ensureBackendPoolDeletedFromNode(nodeName, backendPoolIDs) if err != nil { if !errors.Is(err, ErrorNotVmssInstance) { // Do nothing for the VMAS nodes. - klog.Errorf("ensureBackendPoolDeleted(%s): backendPoolID(%s) - failed with error %v", getServiceName(service), backendPoolID, err) + klog.Errorf("ensureBackendPoolDeleted(%s): backendPoolIDs(%q) - failed with error %v", getServiceName(service), backendPoolIDs, err) allErrs = append(allErrs, err) } continue @@ -1941,7 +1951,7 @@ func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, "operation", "EnsureBackendPoolDeleted UpdateVMSSVMs", "vmssName", meta.vmssName, "resourceGroup", meta.resourceGroup, - "backendPoolID", backendPoolID, + "backendPoolIDs", backendPoolIDs, } batchSize, err := ss.VMSSBatchSize(meta.vmssName) @@ -1976,37 +1986,39 @@ func (ss *ScaleSet) ensureBackendPoolDeleted(service *v1.Service, backendPoolID, } // EnsureBackendPoolDeleted ensures the loadBalancer backendAddressPools deleted from the specified nodes. -func (ss *ScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { +func (ss *ScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { if backendAddressPools == nil { return false, nil } - vmssUniformBackendIPConfigurations := []network.InterfaceIPConfiguration{} - vmssFlexBackendIPConfigurations := []network.InterfaceIPConfiguration{} - avSetBackendIPConfigurations := []network.InterfaceIPConfiguration{} + vmssUniformBackendIPConfigurationsMap := map[string][]network.InterfaceIPConfiguration{} + vmssFlexBackendIPConfigurationsMap := map[string][]network.InterfaceIPConfiguration{} + avSetBackendIPConfigurationsMap := map[string][]network.InterfaceIPConfiguration{} for _, backendPool := range *backendAddressPools { - if strings.EqualFold(*backendPool.ID, backendPoolID) && backendPool.BackendIPConfigurations != nil { - for _, ipConf := range *backendPool.BackendIPConfigurations { - if ipConf.ID == nil { - continue - } - - vmManagementType, err := ss.getVMManagementTypeByIPConfigurationID(*ipConf.ID, azcache.CacheReadTypeUnsafe) - if err != nil { - klog.Warningf("Failed to check VM management type by ipConfigurationID %s: %v, skip it", *ipConf.ID, err) - } - - if vmManagementType == ManagedByAvSet { - // vm is managed by availability set. - avSetBackendIPConfigurations = append(avSetBackendIPConfigurations, ipConf) - } - if vmManagementType == ManagedByVmssFlex { - // vm is managed by vmss flex. - vmssFlexBackendIPConfigurations = append(vmssFlexBackendIPConfigurations, ipConf) - } - if vmManagementType == ManagedByVmssUniform { - // vm is managed by vmss flex. - vmssUniformBackendIPConfigurations = append(vmssUniformBackendIPConfigurations, ipConf) + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(*backendPool.ID, backendPoolID) && backendPool.BackendIPConfigurations != nil { + for _, ipConf := range *backendPool.BackendIPConfigurations { + if ipConf.ID == nil { + continue + } + + vmManagementType, err := ss.getVMManagementTypeByIPConfigurationID(*ipConf.ID, azcache.CacheReadTypeUnsafe) + if err != nil { + klog.Warningf("Failed to check VM management type by ipConfigurationID %s: %v, skip it", *ipConf.ID, err) + } + + if vmManagementType == ManagedByAvSet { + // vm is managed by availability set. + avSetBackendIPConfigurationsMap[backendPoolID] = append(avSetBackendIPConfigurationsMap[backendPoolID], ipConf) + } + if vmManagementType == ManagedByVmssFlex { + // vm is managed by vmss flex. + vmssFlexBackendIPConfigurationsMap[backendPoolID] = append(vmssFlexBackendIPConfigurationsMap[backendPoolID], ipConf) + } + if vmManagementType == ManagedByVmssUniform { + // vm is managed by vmss flex. + vmssUniformBackendIPConfigurationsMap[backendPoolID] = append(vmssUniformBackendIPConfigurationsMap[backendPoolID], ipConf) + } } } } @@ -2015,23 +2027,25 @@ func (ss *ScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, // make sure all vmss including uniform and flex are decoupled from // the lb backend pool even if there is no ipConfigs in the backend pool. if deleteFromVMSet { - err := ss.ensureBackendPoolDeletedFromVMSS(backendPoolID, vmSetName) + err := ss.ensureBackendPoolDeletedFromVMSS(backendPoolIDs, vmSetName) if err != nil { return false, err } } var updated bool - if len(vmssUniformBackendIPConfigurations) > 0 { - vmssUniformBackendPools := &[]network.BackendAddressPool{ - { - ID: pointer.String(backendPoolID), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - BackendIPConfigurations: &vmssUniformBackendIPConfigurations, - }, + vmssUniformBackendPools := []network.BackendAddressPool{} + for backendPoolID, vmssUniformBackendIPConfigurations := range vmssUniformBackendIPConfigurationsMap { + vmssUniformBackendIPConfigurations := vmssUniformBackendIPConfigurations + vmssUniformBackendPools = append(vmssUniformBackendPools, network.BackendAddressPool{ + ID: pointer.String(backendPoolID), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + BackendIPConfigurations: &vmssUniformBackendIPConfigurations, }, - } - updatedVM, err := ss.ensureBackendPoolDeleted(service, backendPoolID, vmSetName, vmssUniformBackendPools) + }) + } + if len(vmssUniformBackendPools) > 0 { + updatedVM, err := ss.ensureBackendPoolDeleted(service, backendPoolIDs, vmSetName, &vmssUniformBackendPools) if err != nil { return false, err } @@ -2040,16 +2054,18 @@ func (ss *ScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, } } - if len(vmssFlexBackendIPConfigurations) > 0 { - vmssFlexBackendPools := &[]network.BackendAddressPool{ - { - ID: pointer.String(backendPoolID), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - BackendIPConfigurations: &vmssFlexBackendIPConfigurations, - }, + vmssFlexBackendPools := []network.BackendAddressPool{} + for backendPoolID, vmssFlexBackendIPConfigurations := range vmssFlexBackendIPConfigurationsMap { + vmssFlexBackendIPConfigurations := vmssFlexBackendIPConfigurations + vmssFlexBackendPools = append(vmssFlexBackendPools, network.BackendAddressPool{ + ID: pointer.String(backendPoolID), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + BackendIPConfigurations: &vmssFlexBackendIPConfigurations, }, - } - updatedNIC, err := ss.flexScaleSet.EnsureBackendPoolDeleted(service, backendPoolID, vmSetName, vmssFlexBackendPools, false) + }) + } + if len(vmssFlexBackendPools) > 0 { + updatedNIC, err := ss.flexScaleSet.EnsureBackendPoolDeleted(service, backendPoolIDs, vmSetName, &vmssFlexBackendPools, false) if err != nil { return false, err } @@ -2058,16 +2074,18 @@ func (ss *ScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, } } - if len(avSetBackendIPConfigurations) > 0 { - avSetBackendPools := &[]network.BackendAddressPool{ - { - ID: pointer.String(backendPoolID), - BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ - BackendIPConfigurations: &avSetBackendIPConfigurations, - }, + avSetBackendPools := []network.BackendAddressPool{} + for backendPoolID, avSetBackendIPConfigurations := range avSetBackendIPConfigurationsMap { + avSetBackendIPConfigurations := avSetBackendIPConfigurations + avSetBackendPools = append(avSetBackendPools, network.BackendAddressPool{ + ID: pointer.String(backendPoolID), + BackendAddressPoolPropertiesFormat: &network.BackendAddressPoolPropertiesFormat{ + BackendIPConfigurations: &avSetBackendIPConfigurations, }, - } - updatedNIC, err := ss.availabilitySet.EnsureBackendPoolDeleted(service, backendPoolID, vmSetName, avSetBackendPools, false) + }) + } + if len(avSetBackendPools) > 0 { + updatedNIC, err := ss.availabilitySet.EnsureBackendPoolDeleted(service, backendPoolIDs, vmSetName, &avSetBackendPools, false) if err != nil { return false, err } @@ -2125,7 +2143,7 @@ func (ss *ScaleSet) GetNodeCIDRMasksByProviderID(providerID string) (int, int, e } // EnsureBackendPoolDeletedFromVMSets ensures the loadBalancer backendAddressPools deleted from the specified VMSS -func (ss *ScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]bool, backendPoolID string) error { +func (ss *ScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]bool, backendPoolIDs []string) error { vmssUpdaters := make([]func() error, 0, len(vmssNamesMap)) errors := make([]error, 0, len(vmssNamesMap)) for vmssName := range vmssNamesMap { @@ -2169,10 +2187,12 @@ func (ss *ScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]b var newBackendPools []compute.SubResource for i := len(loadBalancerBackendAddressPools) - 1; i >= 0; i-- { curPool := loadBalancerBackendAddressPools[i] - if strings.EqualFold(backendPoolID, *curPool.ID) { - klog.V(10).Infof("EnsureBackendPoolDeletedFromVMSets gets unwanted backend pool %q for VMSS %s", backendPoolID, vmssName) - found = true - newBackendPools = append(loadBalancerBackendAddressPools[:i], loadBalancerBackendAddressPools[i+1:]...) + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(backendPoolID, *curPool.ID) { + klog.V(10).Infof("EnsureBackendPoolDeletedFromVMSets gets unwanted backend pool %q for VMSS %s", backendPoolID, vmssName) + found = true + newBackendPools = append(loadBalancerBackendAddressPools[:i], loadBalancerBackendAddressPools[i+1:]...) + } } } if !found { @@ -2193,10 +2213,10 @@ func (ss *ScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]b }, } - klog.V(2).Infof("EnsureBackendPoolDeletedFromVMSets begins to update vmss(%s) with backendPoolID %s", vmssName, backendPoolID) + klog.V(2).Infof("EnsureBackendPoolDeletedFromVMSets begins to update vmss(%s) with backendPoolIDs %q", vmssName, backendPoolIDs) rerr := ss.CreateOrUpdateVMSS(ss.ResourceGroup, vmssName, newVMSS) if rerr != nil { - klog.Errorf("EnsureBackendPoolDeletedFromVMSets CreateOrUpdateVMSS(%s) with new backendPoolID %s, err: %v", vmssName, backendPoolID, rerr) + klog.Errorf("EnsureBackendPoolDeletedFromVMSets CreateOrUpdateVMSS(%s) with new backendPoolIDs %q, err: %v", vmssName, backendPoolIDs, rerr) return rerr.Error() } diff --git a/pkg/provider/azure_vmss_test.go b/pkg/provider/azure_vmss_test.go index 2ba8abb623..4727df3527 100644 --- a/pkg/provider/azure_vmss_test.go +++ b/pkg/provider/azure_vmss_test.go @@ -57,21 +57,22 @@ const ( ) func buildTestVMSSWithLB(name, namePrefix string, lbBackendpoolIDs []string, ipv6 bool) compute.VirtualMachineScaleSet { - lbBackendpools := make([]compute.SubResource, 0) + lbBackendpoolsV4, lbBackendpoolsV6 := make([]compute.SubResource, 0), make([]compute.SubResource, 0) for _, id := range lbBackendpoolIDs { - lbBackendpools = append(lbBackendpools, compute.SubResource{ID: pointer.String(id)}) + lbBackendpoolsV4 = append(lbBackendpoolsV4, compute.SubResource{ID: pointer.String(id)}) + lbBackendpoolsV6 = append(lbBackendpoolsV6, compute.SubResource{ID: pointer.String(id + "-" + v6Suffix)}) } ipConfig := []compute.VirtualMachineScaleSetIPConfiguration{ { VirtualMachineScaleSetIPConfigurationProperties: &compute.VirtualMachineScaleSetIPConfigurationProperties{ - LoadBalancerBackendAddressPools: &lbBackendpools, + LoadBalancerBackendAddressPools: &lbBackendpoolsV4, }, }, } if ipv6 { ipConfig = append(ipConfig, compute.VirtualMachineScaleSetIPConfiguration{ VirtualMachineScaleSetIPConfigurationProperties: &compute.VirtualMachineScaleSetIPConfigurationProperties{ - LoadBalancerBackendAddressPools: &lbBackendpools, + LoadBalancerBackendAddressPools: &lbBackendpoolsV6, PrivateIPAddressVersion: compute.IPv6, }, }) @@ -167,7 +168,7 @@ func buildTestVirtualMachineEnv(ss *Cloud, scaleSetName, zone string, faultDomai Name: pointer.String("ipconfig1"), VirtualMachineScaleSetIPConfigurationProperties: &compute.VirtualMachineScaleSetIPConfigurationProperties{ Primary: pointer.Bool(false), - LoadBalancerBackendAddressPools: &[]compute.SubResource{{ID: pointer.String(testLBBackendpoolID0)}}, + LoadBalancerBackendAddressPools: &[]compute.SubResource{{ID: pointer.String(testLBBackendpoolID0 + "-" + v6Suffix)}}, PrivateIPAddressVersion: compute.IPv6, }, }, @@ -2133,7 +2134,7 @@ func TestEnsureVMSSInPool(t *testing.T) { }, }, isBasicLB: false, - backendPoolID: testLBBackendpoolID1, + backendPoolID: testLBBackendpoolID1 + "-" + v6Suffix, clusterIP: "fd00::e68b", expectedPutVMSS: false, expectedErr: fmt.Errorf("failed to find a IPconfiguration(IPv6=true) for the scale set VM \"\""), @@ -2148,7 +2149,7 @@ func TestEnsureVMSSInPool(t *testing.T) { }, }, isBasicLB: false, - backendPoolID: testLBBackendpoolID1, + backendPoolID: testLBBackendpoolID1 + "-" + v6Suffix, setIPv6Config: true, clusterIP: "fd00::e68b", expectedPutVMSS: true, @@ -2214,41 +2215,43 @@ func TestEnsureVMSSInPool(t *testing.T) { } for _, test := range testCases { - ss, err := NewTestScaleSet(ctrl) - assert.NoError(t, err, test.description) + t.Run(test.description, func(t *testing.T) { + ss, err := NewTestScaleSet(ctrl) + assert.NoError(t, err, test.description) - if !test.isBasicLB { - ss.LoadBalancerSku = consts.LoadBalancerSkuStandard - } + if !test.isBasicLB { + ss.LoadBalancerSku = consts.LoadBalancerSkuStandard + } - expectedVMSS := buildTestVMSSWithLB(testVMSSName, "vmss-vm-", []string{testLBBackendpoolID0}, test.setIPv6Config) - if test.isVMSSDeallocating { - expectedVMSS.ProvisioningState = pointer.String(consts.VirtualMachineScaleSetsDeallocating) - } - if test.isVMSSNilNICConfig { - expectedVMSS.VirtualMachineProfile.NetworkProfile.NetworkInterfaceConfigurations = nil - } - mockVMSSClient := ss.cloud.VirtualMachineScaleSetsClient.(*mockvmssclient.MockInterface) - mockVMSSClient.EXPECT().List(gomock.Any(), ss.ResourceGroup).Return([]compute.VirtualMachineScaleSet{expectedVMSS}, nil).AnyTimes() - vmssPutTimes := 0 - if test.expectedPutVMSS { - vmssPutTimes = 1 - mockVMSSClient.EXPECT().Get(gomock.Any(), ss.ResourceGroup, testVMSSName).Return(expectedVMSS, nil) - } - mockVMSSClient.EXPECT().CreateOrUpdate(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(nil).Times(vmssPutTimes) + expectedVMSS := buildTestVMSSWithLB(testVMSSName, "vmss-vm-", []string{testLBBackendpoolID0}, test.setIPv6Config) + if test.isVMSSDeallocating { + expectedVMSS.ProvisioningState = pointer.String(consts.VirtualMachineScaleSetsDeallocating) + } + if test.isVMSSNilNICConfig { + expectedVMSS.VirtualMachineProfile.NetworkProfile.NetworkInterfaceConfigurations = nil + } + mockVMSSClient := ss.cloud.VirtualMachineScaleSetsClient.(*mockvmssclient.MockInterface) + mockVMSSClient.EXPECT().List(gomock.Any(), ss.ResourceGroup).Return([]compute.VirtualMachineScaleSet{expectedVMSS}, nil).AnyTimes() + vmssPutTimes := 0 + if test.expectedPutVMSS { + vmssPutTimes = 1 + mockVMSSClient.EXPECT().Get(gomock.Any(), ss.ResourceGroup, testVMSSName).Return(expectedVMSS, nil) + } + mockVMSSClient.EXPECT().CreateOrUpdate(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(nil).Times(vmssPutTimes) - expectedVMSSVMs, _, _ := buildTestVirtualMachineEnv(ss.cloud, testVMSSName, "", 0, []string{"vmss-vm-000000"}, "", test.setIPv6Config) - mockVMSSVMClient := ss.cloud.VirtualMachineScaleSetVMsClient.(*mockvmssvmclient.MockInterface) - mockVMSSVMClient.EXPECT().List(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(expectedVMSSVMs, nil).AnyTimes() + expectedVMSSVMs, _, _ := buildTestVirtualMachineEnv(ss.cloud, testVMSSName, "", 0, []string{"vmss-vm-000000"}, "", test.setIPv6Config) + mockVMSSVMClient := ss.cloud.VirtualMachineScaleSetVMsClient.(*mockvmssvmclient.MockInterface) + mockVMSSVMClient.EXPECT().List(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(expectedVMSSVMs, nil).AnyTimes() - if test.expectedGetInstanceID != "" { - mockVMSet := NewMockVMSet(ctrl) - mockVMSet.EXPECT().GetInstanceIDByNodeName(gomock.Any()).Return(test.expectedGetInstanceID, test.getInstanceIDErr) - ss.VMSet = mockVMSet - } + if test.expectedGetInstanceID != "" { + mockVMSet := NewMockVMSet(ctrl) + mockVMSet.EXPECT().GetInstanceIDByNodeName(gomock.Any()).Return(test.expectedGetInstanceID, test.getInstanceIDErr) + ss.VMSet = mockVMSet + } - err = ss.ensureVMSSInPool(&v1.Service{Spec: v1.ServiceSpec{ClusterIP: test.clusterIP}}, test.nodes, test.backendPoolID, test.vmSetName) - assert.Equal(t, test.expectedErr, err, test.description+", but an error occurs") + err = ss.ensureVMSSInPool(&v1.Service{Spec: v1.ServiceSpec{ClusterIP: test.clusterIP}}, test.nodes, test.backendPoolID, test.vmSetName) + assert.Equal(t, test.expectedErr, err, test.description+", but an error occurs") + }) } } @@ -2421,7 +2424,7 @@ func TestEnsureBackendPoolDeletedFromNode(t *testing.T) { mockVMSSVMClient := ss.cloud.VirtualMachineScaleSetVMsClient.(*mockvmssvmclient.MockInterface) mockVMSSVMClient.EXPECT().List(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(expectedVMSSVMs, nil).AnyTimes() - nodeResourceGroup, ssName, instanceID, vm, err := ss.ensureBackendPoolDeletedFromNode(test.nodeName, test.backendpoolID) + nodeResourceGroup, ssName, instanceID, vm, err := ss.ensureBackendPoolDeletedFromNode(test.nodeName, []string{test.backendpoolID}) assert.Equal(t, test.expectedErr, err, test.description+", but an error occurs") assert.Equal(t, test.expectedNodeResourceGroup, nodeResourceGroup, test.description) assert.Equal(t, test.expectedVMSSName, ssName, test.description) @@ -2518,7 +2521,7 @@ func TestEnsureBackendPoolDeletedFromVMSS(t *testing.T) { mockVMSSVMClient := ss.cloud.VirtualMachineScaleSetVMsClient.(*mockvmssvmclient.MockInterface) mockVMSSVMClient.EXPECT().List(gomock.Any(), ss.ResourceGroup, testVMSSName, gomock.Any()).Return(expectedVMSSVMs, nil).AnyTimes() - err = ss.ensureBackendPoolDeletedFromVMSS(test.backendPoolID, testVMSSName) + err = ss.ensureBackendPoolDeletedFromVMSS([]string{test.backendPoolID}, testVMSSName) if test.expectedErr != nil { assert.EqualError(t, test.expectedErr, err.Error(), test.description+", but an error occurs") } @@ -2628,7 +2631,7 @@ func TestEnsureBackendPoolDeleted(t *testing.T) { mockVMsClient := ss.cloud.VirtualMachinesClient.(*mockvmclient.MockInterface) mockVMsClient.EXPECT().List(gomock.Any(), gomock.Any()).Return([]compute.VirtualMachine{}, nil).AnyTimes() - updated, err := ss.EnsureBackendPoolDeleted(&v1.Service{}, test.backendpoolID, testVMSSName, test.backendAddressPools, true) + updated, err := ss.EnsureBackendPoolDeleted(&v1.Service{}, []string{test.backendpoolID}, testVMSSName, test.backendAddressPools, true) assert.Equal(t, test.expectedErr, err != nil, test.description+", but an error occurs") if !test.expectedErr && test.expectedVMSSVMPutTimes > 0 { assert.True(t, updated, test.description) @@ -2721,7 +2724,7 @@ func TestEnsureBackendPoolDeletedConcurrently(t *testing.T) { i := i id := id testFunc = append(testFunc, func() error { - _, err := ss.EnsureBackendPoolDeleted(&v1.Service{}, id, testVMSSNames[i], backendAddressPools, true) + _, err := ss.EnsureBackendPoolDeleted(&v1.Service{}, []string{id}, testVMSSNames[i], backendAddressPools, true) return err }) } diff --git a/pkg/provider/azure_vmssflex.go b/pkg/provider/azure_vmssflex.go index 0dc34db35f..36d7f9cf65 100644 --- a/pkg/provider/azure_vmssflex.go +++ b/pkg/provider/azure_vmssflex.go @@ -31,7 +31,6 @@ import ( utilerrors "k8s.io/apimachinery/pkg/util/errors" cloudprovider "k8s.io/cloud-provider" "k8s.io/klog/v2" - utilnet "k8s.io/utils/net" "k8s.io/utils/pointer" azcache "sigs.k8s.io/cloud-provider-azure/pkg/cache" @@ -511,7 +510,7 @@ func (fs *FlexScaleSet) EnsureHostInPool(service *v1.Service, nodeName types.Nod } var primaryIPConfig *network.InterfaceIPConfiguration - ipv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + ipv6 := isBackendPoolIPv6(backendPoolID) if !fs.Cloud.ipv6DualStackEnabled && !ipv6 { primaryIPConfig, err = getPrimaryIPConfig(nic) if err != nil { @@ -662,7 +661,7 @@ func (fs *FlexScaleSet) ensureVMSSFlexInPool(service *v1.Service, nodes []*v1.No return err } var primaryIPConfig *compute.VirtualMachineScaleSetIPConfiguration - ipv6 := utilnet.IsIPv6String(service.Spec.ClusterIP) + ipv6 := isBackendPoolIPv6(backendPoolID) // Find primary network interface configuration. if !fs.Cloud.ipv6DualStackEnabled && !ipv6 { // Find primary IP configuration. @@ -797,7 +796,7 @@ func (fs *FlexScaleSet) EnsureHostsInPool(service *v1.Service, nodes []*v1.Node, return nil } -func (fs *FlexScaleSet) ensureBackendPoolDeletedFromVmssFlex(backendPoolID string, vmSetName string) error { +func (fs *FlexScaleSet) ensureBackendPoolDeletedFromVmssFlex(backendPoolIDs []string, vmSetName string) error { vmssNamesMap := make(map[string]bool) if fs.useStandardLoadBalancer() && !fs.EnableMultipleStandardLoadBalancers { cached, err := fs.vmssFlexCache.Get(consts.VmssFlexKey, azcache.CacheReadTypeDefault) @@ -814,11 +813,11 @@ func (fs *FlexScaleSet) ensureBackendPoolDeletedFromVmssFlex(backendPoolID strin } else { vmssNamesMap[vmSetName] = true } - return fs.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, backendPoolID) + return fs.EnsureBackendPoolDeletedFromVMSets(vmssNamesMap, backendPoolIDs) } // EnsureBackendPoolDeletedFromVMSets ensures the loadBalancer backendAddressPools deleted from the specified VMSS Flex -func (fs *FlexScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]bool, backendPoolID string) error { +func (fs *FlexScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[string]bool, backendPoolIDs []string) error { vmssUpdaters := make([]func() error, 0, len(vmssNamesMap)) errors := make([]error, 0, len(vmssNamesMap)) for vmssName := range vmssNamesMap { @@ -862,10 +861,12 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[stri var newBackendPools []compute.SubResource for i := len(loadBalancerBackendAddressPools) - 1; i >= 0; i-- { curPool := loadBalancerBackendAddressPools[i] - if strings.EqualFold(backendPoolID, *curPool.ID) { - klog.V(10).Infof("fs.EnsureBackendPoolDeletedFromVMSets gets unwanted backend pool %q for VMSS %s", backendPoolID, vmssName) - found = true - newBackendPools = append(loadBalancerBackendAddressPools[:i], loadBalancerBackendAddressPools[i+1:]...) + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(backendPoolID, *curPool.ID) { + klog.V(10).Infof("fs.EnsureBackendPoolDeletedFromVMSets gets unwanted backend pool %q for VMSS %s", backendPoolID, vmssName) + found = true + newBackendPools = append(loadBalancerBackendAddressPools[:i], loadBalancerBackendAddressPools[i+1:]...) + } } } if !found { @@ -891,10 +892,10 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[stri _ = fs.vmssFlexCache.Delete(consts.VmssFlexKey) }() - klog.V(2).Infof("fs.EnsureBackendPoolDeletedFromVMSets begins to delete backendPoolID %s from vmss(%s)", backendPoolID, vmssName) + klog.V(2).Infof("fs.EnsureBackendPoolDeletedFromVMSets begins to delete backendPoolIDs %q from vmss(%s)", backendPoolIDs, vmssName) rerr := fs.CreateOrUpdateVMSS(fs.ResourceGroup, vmssName, newVMSS) if rerr != nil { - klog.Errorf("fs.EnsureBackendPoolDeletedFromVMSets CreateOrUpdateVMSS(%s) for backendPoolID %s, err: %v", vmssName, backendPoolID, rerr) + klog.Errorf("fs.EnsureBackendPoolDeletedFromVMSets CreateOrUpdateVMSS(%s) for backendPoolIDs %q, err: %v", vmssName, backendPoolIDs, rerr) return rerr.Error() } @@ -915,7 +916,7 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeletedFromVMSets(vmssNamesMap map[stri } // EnsureBackendPoolDeleted ensures the loadBalancer backendAddressPools deleted from the specified nodes. -func (fs *FlexScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolID, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { +func (fs *FlexScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoolIDs []string, vmSetName string, backendAddressPools *[]network.BackendAddressPool, deleteFromVMSet bool) (bool, error) { // Returns nil if backend address pools already deleted. if backendAddressPools == nil { return false, nil @@ -929,13 +930,15 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoo ipConfigurationIDs := []string{} for _, backendPool := range *backendAddressPools { - if strings.EqualFold(pointer.StringDeref(backendPool.ID, ""), backendPoolID) && backendPool.BackendAddressPoolPropertiesFormat != nil && backendPool.BackendIPConfigurations != nil { - for _, ipConf := range *backendPool.BackendIPConfigurations { - if ipConf.ID == nil { - continue - } + for _, backendPoolID := range backendPoolIDs { + if strings.EqualFold(pointer.StringDeref(backendPool.ID, ""), backendPoolID) && backendPool.BackendAddressPoolPropertiesFormat != nil && backendPool.BackendIPConfigurations != nil { + for _, ipConf := range *backendPool.BackendIPConfigurations { + if ipConf.ID == nil { + continue + } - ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) + ipConfigurationIDs = append(ipConfigurationIDs, *ipConf.ID) + } } } } @@ -971,19 +974,17 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoo } } - // 1. Ensure the backendPoolID is deleted from the VMSS. - klog.V(2).Infof("1. Ensure the backendPoolID is deleted from the VMSS.") + klog.V(2).Infof("Ensure backendPoolIDs %q deleted from the VMSS.", backendPoolIDs) if deleteFromVMSet { - err := fs.ensureBackendPoolDeletedFromVmssFlex(backendPoolID, vmSetName) + err := fs.ensureBackendPoolDeletedFromVmssFlex(backendPoolIDs, vmSetName) if err != nil { allErrs = append(allErrs, err) } } - klog.V(2).Infof("2. Ensure the backendPoolID is deleted from the VMSS VMs.") - // 2. Ensure the backendPoolID is deleted from the VMSS VMs. + klog.V(2).Infof("Ensure backendPoolIDs deleted from the VMSS VMs.", backendPoolIDs) klog.V(2).Infof("go into fs.ensureBackendPoolDeletedFromNode, vmssFlexVMNameMap: %s, size: %s", vmssFlexVMNameMap, len(vmssFlexVMNameMap)) - nicUpdated, err := fs.ensureBackendPoolDeletedFromNode(vmssFlexVMNameMap, backendPoolID) + nicUpdated, err := fs.ensureBackendPoolDeletedFromNode(vmssFlexVMNameMap, backendPoolIDs) klog.V(2).Infof("exit from fs.ensureBackendPoolDeletedFromNode") if err != nil { allErrs = append(allErrs, err) @@ -998,14 +999,15 @@ func (fs *FlexScaleSet) EnsureBackendPoolDeleted(service *v1.Service, backendPoo } -func (fs *FlexScaleSet) ensureBackendPoolDeletedFromNode(vmssFlexVMNameMap map[string]string, backendPoolID string) (bool, error) { +func (fs *FlexScaleSet) ensureBackendPoolDeletedFromNode(vmssFlexVMNameMap map[string]string, backendPoolIDs []string) (bool, error) { nicUpdaters := make([]func() error, 0) allErrs := make([]error, 0) - i := 0 var nicUpdated bool + nics := map[string]network.Interface{} // nicName -> nic for nodeName, nicName := range vmssFlexVMNameMap { - i++ - klog.V(2).Infof("i = %s", i) + if _, ok := nics[nicName]; ok { + continue + } ctx, cancel := getContextWithCancel() defer cancel() @@ -1020,40 +1022,45 @@ func (fs *FlexScaleSet) ensureBackendPoolDeletedFromNode(vmssFlexVMNameMap map[s } if nic.InterfacePropertiesFormat != nil && nic.InterfacePropertiesFormat.IPConfigurations != nil { - newIPConfigs := *nic.IPConfigurations - for j, ipConf := range newIPConfigs { - if !pointer.BoolDeref(ipConf.Primary, false) { - continue - } - // found primary ip configuration - if ipConf.LoadBalancerBackendAddressPools != nil { - newLBAddressPools := *ipConf.LoadBalancerBackendAddressPools - for k := len(newLBAddressPools) - 1; k >= 0; k-- { - pool := newLBAddressPools[k] + nicName := pointer.StringDeref(nic.Name, "") + nics[nicName] = nic + } + } + for _, nic := range nics { + newIPConfigs := *nic.IPConfigurations + for j, ipConf := range newIPConfigs { + if !pointer.BoolDeref(ipConf.Primary, false) { + continue + } + // found primary ip configuration + if ipConf.LoadBalancerBackendAddressPools != nil { + newLBAddressPools := *ipConf.LoadBalancerBackendAddressPools + for k := len(newLBAddressPools) - 1; k >= 0; k-- { + pool := newLBAddressPools[k] + for _, backendPoolID := range backendPoolIDs { if strings.EqualFold(pointer.StringDeref(pool.ID, ""), backendPoolID) { newLBAddressPools = append(newLBAddressPools[:k], newLBAddressPools[k+1:]...) - break } } - newIPConfigs[j].LoadBalancerBackendAddressPools = &newLBAddressPools } + newIPConfigs[j].LoadBalancerBackendAddressPools = &newLBAddressPools } - nic.IPConfigurations = &newIPConfigs - - nicUpdaters = append(nicUpdaters, func() error { - ctx, cancel := getContextWithCancel() - defer cancel() - klog.V(2).Infof("EnsureBackendPoolDeleted begins to CreateOrUpdate for NIC(%s, %s) with backendPoolID %s", fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), backendPoolID) - rerr := fs.InterfacesClient.CreateOrUpdate(ctx, fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), nic) - if rerr != nil { - klog.Errorf("EnsureBackendPoolDeleted CreateOrUpdate for NIC(%s, %s) failed with error %v", fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), rerr.Error()) - return rerr.Error() - } - nicUpdated = true - klog.V(2).Infof("EnsureBackendPoolDeleted done") - return nil - }) } + nic.IPConfigurations = &newIPConfigs + + nicUpdaters = append(nicUpdaters, func() error { + ctx, cancel := getContextWithCancel() + defer cancel() + klog.V(2).Infof("EnsureBackendPoolDeleted begins to CreateOrUpdate for NIC(%s, %s) with backendPoolIDs %q", fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), backendPoolIDs) + rerr := fs.InterfacesClient.CreateOrUpdate(ctx, fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), nic) + if rerr != nil { + klog.Errorf("EnsureBackendPoolDeleted CreateOrUpdate for NIC(%s, %s) failed with error %v", fs.ResourceGroup, pointer.StringDeref(nic.Name, ""), rerr.Error()) + return rerr.Error() + } + nicUpdated = true + klog.V(2).Infof("EnsureBackendPoolDeleted done") + return nil + }) } klog.V(2).Infof("nicUpdaters size: %s", len(nicUpdaters)) errs := utilerrors.AggregateGoroutines(nicUpdaters...) diff --git a/pkg/provider/azure_vmssflex_test.go b/pkg/provider/azure_vmssflex_test.go index d9220fe664..5157ad3e43 100644 --- a/pkg/provider/azure_vmssflex_test.go +++ b/pkg/provider/azure_vmssflex_test.go @@ -1688,7 +1688,7 @@ func TestEnsureBackendPoolDeletedFromVMSetsVmssFlex(t *testing.T) { mockVMSSClient.EXPECT().Get(gomock.Any(), gomock.Any(), gomock.Any()).Return(testVmssFlex1, nil).AnyTimes() mockVMSSClient.EXPECT().CreateOrUpdate(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(tc.vmssPutErr).AnyTimes() - err = fs.EnsureBackendPoolDeletedFromVMSets(tc.vmssNamesMap, tc.backendPoolID) + err = fs.EnsureBackendPoolDeletedFromVMSets(tc.vmssNamesMap, []string{tc.backendPoolID}) _, _ = fs.getVmssFlexByName("vmssflex1") if tc.expectedErr != nil { @@ -1767,7 +1767,7 @@ func TestEnsureBackendPoolDeletedFromNodeVmssFlex(t *testing.T) { mockInterfacesClient.EXPECT().Get(gomock.Any(), gomock.Any(), "testvm1-nic", gomock.Any()).Return(tc.nic, tc.nicGetErr).AnyTimes() mockInterfacesClient.EXPECT().CreateOrUpdate(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(tc.nicPutErr).Times(tc.expectedPutNICTimes) - updated, err := fs.ensureBackendPoolDeletedFromNode(tc.vmssFlexVMNameMap, tc.backendPoolID) + updated, err := fs.ensureBackendPoolDeletedFromNode(tc.vmssFlexVMNameMap, []string{tc.backendPoolID}) if tc.expectedErr != nil { assert.EqualError(t, err, tc.expectedErr.Error(), tc.description) @@ -1916,7 +1916,7 @@ func TestEnsureBackendPoolDeletedVmssFlex(t *testing.T) { mockInterfacesClient.EXPECT().Get(gomock.Any(), gomock.Any(), "testvm1-nic", gomock.Any()).Return(tc.nic, tc.nicGetErr).AnyTimes() mockInterfacesClient.EXPECT().CreateOrUpdate(gomock.Any(), gomock.Any(), gomock.Any(), gomock.Any()).Return(tc.nicPutErr).AnyTimes() - _, err = fs.EnsureBackendPoolDeleted(tc.service, tc.backendPoolID, tc.vmSetName, tc.backendAddressPools, tc.deleteFromVMSet) + _, err = fs.EnsureBackendPoolDeleted(tc.service, []string{tc.backendPoolID}, tc.vmSetName, tc.backendAddressPools, tc.deleteFromVMSet) if tc.expectedErr != nil { assert.EqualError(t, err, tc.expectedErr.Error(), tc.description)