Skip to content

Commit

Permalink
Merge pull request #4498 from k8s-infra-cherrypick-robot/cherry-pick-…
Browse files Browse the repository at this point in the history
…4449-to-release-2.2

[release-2.2] Set availability zones in status.networkStatus.apiServerElb when using BYO NLB
  • Loading branch information
k8s-ci-robot authored Sep 12, 2023
2 parents 764e47e + 50fd444 commit 46e6fb1
Show file tree
Hide file tree
Showing 2 changed files with 140 additions and 2 deletions.
7 changes: 5 additions & 2 deletions pkg/cloud/services/elb/loadbalancer.go
Original file line number Diff line number Diff line change
Expand Up @@ -1467,17 +1467,20 @@ func fromSDKTypeToClassicELB(v *elb.LoadBalancerDescription, attrs *elb.LoadBala

func fromSDKTypeToLB(v *elbv2.LoadBalancer, attrs []*elbv2.LoadBalancerAttribute, tags []*elbv2.Tag) *infrav1.LoadBalancer {
subnetIds := make([]*string, len(v.AvailabilityZones))
availabilityZones := make([]*string, len(v.AvailabilityZones))
for i, az := range v.AvailabilityZones {
subnetIds[i] = az.SubnetId
availabilityZones[i] = az.ZoneName
}
res := &infrav1.LoadBalancer{
ARN: aws.StringValue(v.LoadBalancerArn),
Name: aws.StringValue(v.LoadBalancerName),
Scheme: infrav1.ELBScheme(aws.StringValue(v.Scheme)),
SubnetIDs: aws.StringValueSlice(subnetIds),
// SecurityGroupIDs: aws.StringValueSlice(v.SecurityGroups),
DNSName: aws.StringValue(v.DNSName),
Tags: converters.V2TagsToMap(tags),
AvailabilityZones: aws.StringValueSlice(availabilityZones),
DNSName: aws.StringValue(v.DNSName),
Tags: converters.V2TagsToMap(tags),
}

infraAttrs := make(map[string]*string, len(attrs))
Expand Down
135 changes: 135 additions & 0 deletions pkg/cloud/services/elb/loadbalancer_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1584,6 +1584,141 @@ func TestCreateNLB(t *testing.T) {
}
}

func TestReconcileV2LB(t *testing.T) {
const (
namespace = "foo"
clusterName = "bar"
clusterSubnetID = "subnet-1"
elbName = "bar-apiserver"
elbArn = "arn::apiserver"
vpcID = "vpc-id"
az = "us-west-1a"
)

tests := []struct {
name string
elbV2APIMocks func(m *mocks.MockELBV2APIMockRecorder)
check func(t *testing.T, lb *infrav1.LoadBalancer, err error)
awsCluster func(acl infrav1.AWSCluster) infrav1.AWSCluster
spec func(spec infrav1.LoadBalancer) infrav1.LoadBalancer
}{
{
name: "ensure status populated with BYO NLB",
spec: func(spec infrav1.LoadBalancer) infrav1.LoadBalancer {
return spec
},
awsCluster: func(acl infrav1.AWSCluster) infrav1.AWSCluster {
acl.Spec.ControlPlaneLoadBalancer.Name = aws.String(elbName)
return acl
},
elbV2APIMocks: func(m *mocks.MockELBV2APIMockRecorder) {
m.DescribeLoadBalancers(gomock.Eq(&elbv2.DescribeLoadBalancersInput{
Names: aws.StringSlice([]string{elbName}),
})).
Return(&elbv2.DescribeLoadBalancersOutput{
LoadBalancers: []*elbv2.LoadBalancer{
{
LoadBalancerArn: aws.String(elbArn),
LoadBalancerName: aws.String(elbName),
Scheme: aws.String(string(infrav1.ELBSchemeInternetFacing)),
AvailabilityZones: []*elbv2.AvailabilityZone{
{
SubnetId: aws.String(clusterSubnetID),
ZoneName: aws.String(az),
},
},
VpcId: aws.String(vpcID),
},
},
}, nil)
m.DescribeLoadBalancerAttributes(&elbv2.DescribeLoadBalancerAttributesInput{LoadBalancerArn: aws.String(elbArn)}).Return(
&elbv2.DescribeLoadBalancerAttributesOutput{
Attributes: []*elbv2.LoadBalancerAttribute{
{
Key: aws.String("load_balancing.cross_zone.enabled"),
Value: aws.String("false"),
},
},
},
nil,
)
m.DescribeTags(&elbv2.DescribeTagsInput{ResourceArns: []*string{aws.String(elbArn)}}).Return(
&elbv2.DescribeTagsOutput{
TagDescriptions: []*elbv2.TagDescription{
{
ResourceArn: aws.String(elbArn),
Tags: []*elbv2.Tag{},
},
},
},
nil,
)
},
check: func(t *testing.T, lb *infrav1.LoadBalancer, err error) {
t.Helper()
if err != nil {
t.Fatalf("did not expect error: %v", err)
}
if len(lb.AvailabilityZones) != 1 {
t.Errorf("Expected LB to contain 1 availability zone, got %v", len(lb.AvailabilityZones))
}
},
},
}

for _, tc := range tests {
t.Run(tc.name, func(t *testing.T) {
mockCtrl := gomock.NewController(t)
defer mockCtrl.Finish()
elbV2APIMocks := mocks.NewMockELBV2API(mockCtrl)

scheme, err := setupScheme()
if err != nil {
t.Fatal(err)
}
awsCluster := &infrav1.AWSCluster{
ObjectMeta: metav1.ObjectMeta{Name: clusterName},
Spec: infrav1.AWSClusterSpec{
ControlPlaneLoadBalancer: &infrav1.AWSLoadBalancerSpec{
Name: aws.String(elbName),
LoadBalancerType: infrav1.LoadBalancerTypeNLB,
},
NetworkSpec: infrav1.NetworkSpec{
VPC: infrav1.VPCSpec{
ID: vpcID,
},
},
},
}
client := fake.NewClientBuilder().WithScheme(scheme).Build()
cluster := tc.awsCluster(*awsCluster)
clusterScope, err := scope.NewClusterScope(scope.ClusterScopeParams{
Client: client,
Cluster: &clusterv1.Cluster{
ObjectMeta: metav1.ObjectMeta{
Namespace: namespace,
Name: clusterName,
},
},
AWSCluster: &cluster,
})
if err != nil {
t.Fatal(err)
}

tc.elbV2APIMocks(elbV2APIMocks.EXPECT())

s := &Service{
scope: clusterScope,
ELBV2Client: elbV2APIMocks,
}
err = s.reconcileV2LB()
lb := s.scope.Network().APIServerELB
tc.check(t, &lb, err)
})
}
}

func TestDeleteAPIServerELB(t *testing.T) {
clusterName := "bar" //nolint:goconst // does not need to be a package-level const
elbName := "bar-apiserver"
Expand Down

0 comments on commit 46e6fb1

Please sign in to comment.