Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

add support for azure public loadbalancer #10915

Merged
merged 1 commit into from
Feb 25, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 9 additions & 10 deletions pkg/model/azuremodel/api_loadbalancer.go
Original file line number Diff line number Diff line change
Expand Up @@ -48,16 +48,6 @@ func (b *APILoadBalancerModelBuilder) Build(c *fi.ModelBuilderContext) error {
return nil
}

switch lbSpec.Type {
case kops.LoadBalancerTypeInternal:
// OK
case kops.LoadBalancerTypePublic:
// TODO: Implement creating public ip and attach to public loadbalancer
return fmt.Errorf("only internal loadbalancer for API server is implemented in Azure")
default:
return fmt.Errorf("unhandled LoadBalancer type %q", lbSpec.Type)
}

// Create LoadBalancer for API ELB
lb := &azuretasks.LoadBalancer{
Name: fi.String(b.NameForLoadBalancer()),
Expand All @@ -76,6 +66,15 @@ func (b *APILoadBalancerModelBuilder) Build(c *fi.ModelBuilderContext) error {
lb.Subnet = b.LinkToAzureSubnet(subnet)
case kops.LoadBalancerTypePublic:
lb.External = to.BoolPtr(true)

// Create Public IP Address for Public Loadbalacer
p := &azuretasks.PublicIPAddress{
Name: fi.String(b.NameForLoadBalancer()),
Lifecycle: b.Lifecycle,
ResourceGroup: b.LinkToResourceGroup(),
Tags: map[string]*string{},
}
c.AddTask(p)
default:
return fmt.Errorf("unknown load balancer Type: %q", lbSpec.Type)
}
Expand Down
56 changes: 45 additions & 11 deletions pkg/resources/azure/azure.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,14 +32,15 @@ import (
)

const (
typeResourceGroup = "ResourceGroup"
typeVirtualNetwork = "VirtualNetwork"
typeSubnet = "Subnet"
typeRouteTable = "RouteTable"
typeVMScaleSet = "VMScaleSet"
typeDisk = "Disk"
typeRoleAssignment = "RoleAssignment"
typeLoadBalancer = "LoadBalancer"
typeResourceGroup = "ResourceGroup"
typeVirtualNetwork = "VirtualNetwork"
typeSubnet = "Subnet"
typeRouteTable = "RouteTable"
typeVMScaleSet = "VMScaleSet"
typeDisk = "Disk"
typeRoleAssignment = "RoleAssignment"
typeLoadBalancer = "LoadBalancer"
typePublicIPAddress = "PublicIPAddress"
)

// ListResourcesAzure lists all resources for the cluster by quering Azure.
Expand Down Expand Up @@ -89,6 +90,7 @@ func (g *resourceGetter) listAll() ([]*resources.Resource, error) {
g.listVMScaleSetsAndRoleAssignments,
g.listDisks,
g.listLoadBalancers,
g.listPublicIPAddresses,
}

var resources []*resources.Resource
Expand Down Expand Up @@ -408,11 +410,11 @@ func (g *resourceGetter) listLoadBalancers(ctx context.Context) ([]*resources.Re

var rs []*resources.Resource
for i := range loadBalancers {
rt := &loadBalancers[i]
if !g.isOwnedByCluster(rt.Tags) {
lb := &loadBalancers[i]
if !g.isOwnedByCluster(lb.Tags) {
continue
}
rs = append(rs, g.toLoadBalancerResource(rt))
rs = append(rs, g.toLoadBalancerResource(lb))
}
return rs, nil
}
Expand All @@ -432,6 +434,38 @@ func (g *resourceGetter) deleteLoadBalancer(_ fi.Cloud, r *resources.Resource) e
return g.cloud.LoadBalancer().Delete(context.TODO(), g.resourceGroupName(), r.Name)
}

func (g *resourceGetter) listPublicIPAddresses(ctx context.Context) ([]*resources.Resource, error) {
publicIPAddresses, err := g.cloud.PublicIPAddress().List(ctx, g.resourceGroupName())
if err != nil {
return nil, err
}

var rs []*resources.Resource
for i := range publicIPAddresses {
p := &publicIPAddresses[i]
if !g.isOwnedByCluster(p.Tags) {
continue
}
rs = append(rs, g.toPublicIPAddressResource(p))
}
return rs, nil
}

func (g *resourceGetter) toPublicIPAddressResource(publicIPAddress *network.PublicIPAddress) *resources.Resource {
return &resources.Resource{
Obj: publicIPAddress,
Type: typePublicIPAddress,
ID: *publicIPAddress.Name,
Name: *publicIPAddress.Name,
Deleter: g.deletePublicIPAddress,
Blocks: []string{toKey(typeResourceGroup, g.resourceGroupName())},
}
}

func (g *resourceGetter) deletePublicIPAddress(_ fi.Cloud, r *resources.Resource) error {
return g.cloud.PublicIPAddress().Delete(context.TODO(), g.resourceGroupName(), r.Name)
}

// isOwnedByCluster returns true if the resource is owned by the cluster.
func (g *resourceGetter) isOwnedByCluster(tags map[string]*string) bool {
for k, v := range tags {
Expand Down
1 change: 1 addition & 0 deletions upup/pkg/fi/cloudup/azure/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ go_library(
"disk.go",
"loadbalancer.go",
"networkinterface.go",
"publicipaddress.go",
"resourcegroup.go",
"roleassignment.go",
"routetable.go",
Expand Down
7 changes: 7 additions & 0 deletions upup/pkg/fi/cloudup/azure/azure_cloud.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,7 @@ type AzureCloud interface {
RoleAssignment() RoleAssignmentsClient
NetworkInterface() NetworkInterfacesClient
LoadBalancer() LoadBalancersClient
PublicIPAddress() PublicIPAddressesClient
}

type azureCloudImplementation struct {
Expand All @@ -72,6 +73,7 @@ type azureCloudImplementation struct {
roleAssignmentsClient RoleAssignmentsClient
networkInterfacesClient NetworkInterfacesClient
loadBalancersClient LoadBalancersClient
publicIPAddressesClient PublicIPAddressesClient
}

var _ fi.Cloud = &azureCloudImplementation{}
Expand All @@ -97,6 +99,7 @@ func NewAzureCloud(subscriptionID, location string, tags map[string]string) (Azu
roleAssignmentsClient: newRoleAssignmentsClientImpl(subscriptionID, authorizer),
networkInterfacesClient: newNetworkInterfacesClientImpl(subscriptionID, authorizer),
loadBalancersClient: newLoadBalancersClientImpl(subscriptionID, authorizer),
publicIPAddressesClient: newPublicIPAddressesClientImpl(subscriptionID, authorizer),
}, nil
}

Expand Down Expand Up @@ -266,3 +269,7 @@ func (c *azureCloudImplementation) NetworkInterface() NetworkInterfacesClient {
func (c *azureCloudImplementation) LoadBalancer() LoadBalancersClient {
return c.loadBalancersClient
}

func (c *azureCloudImplementation) PublicIPAddress() PublicIPAddressesClient {
return c.publicIPAddressesClient
}
73 changes: 73 additions & 0 deletions upup/pkg/fi/cloudup/azure/publicipaddress.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
/*
Copyright 2020 The Kubernetes Authors.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

package azure

import (
"context"
"fmt"

"github.com/Azure/azure-sdk-for-go/services/network/mgmt/2020-06-01/network"
"github.com/Azure/go-autorest/autorest"
)

// PublicIPAddressesClient is a client for public ip addresses.
type PublicIPAddressesClient interface {
CreateOrUpdate(ctx context.Context, resourceGroupName, publicIPAddressName string, parameters network.PublicIPAddress) error
List(ctx context.Context, resourceGroupName string) ([]network.PublicIPAddress, error)
Delete(ctx context.Context, resourceGroupName, publicIPAddressName string) error
}

type publicIPAddressesClientImpl struct {
c *network.PublicIPAddressesClient
}

var _ PublicIPAddressesClient = &publicIPAddressesClientImpl{}

func (c *publicIPAddressesClientImpl) CreateOrUpdate(ctx context.Context, resourceGroupName, publicIPAddressName string, parameters network.PublicIPAddress) error {
_, err := c.c.CreateOrUpdate(ctx, resourceGroupName, publicIPAddressName, parameters)
return err
}

func (c *publicIPAddressesClientImpl) List(ctx context.Context, resourceGroupName string) ([]network.PublicIPAddress, error) {
var l []network.PublicIPAddress
for iter, err := c.c.ListComplete(ctx, resourceGroupName); iter.NotDone(); err = iter.Next() {
if err != nil {
return nil, err
}
l = append(l, iter.Value())
}
return l, nil
}

func (c *publicIPAddressesClientImpl) Delete(ctx context.Context, resourceGroupName, publicIPAddressName string) error {
future, err := c.c.Delete(ctx, resourceGroupName, publicIPAddressName)
if err != nil {
return fmt.Errorf("error deleting public ip address: %s", err)
}
if err := future.WaitForCompletionRef(ctx, c.c.Client); err != nil {
return fmt.Errorf("error waiting for public ip address deletion completion: %s", err)
}
return nil
}

func newPublicIPAddressesClientImpl(subscriptionID string, authorizer autorest.Authorizer) *publicIPAddressesClientImpl {
c := network.NewPublicIPAddressesClient(subscriptionID)
c.Authorizer = authorizer
return &publicIPAddressesClientImpl{
c: &c,
}
}
3 changes: 3 additions & 0 deletions upup/pkg/fi/cloudup/azuretasks/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ go_library(
"disk_fitask.go",
"loadbalancer.go",
"loadbalancer_fitask.go",
"publicipaddress.go",
"publicipaddress_fitask.go",
"resourcegroup.go",
"resourcegroup_fitask.go",
"roleassignment.go",
Expand Down Expand Up @@ -45,6 +47,7 @@ go_test(
srcs = [
"disk_test.go",
"loadbalancer_test.go",
"publicipaddress_test.go",
"resourcegroup_test.go",
"roleassignment_test.go",
"subnet_test.go",
Expand Down
7 changes: 3 additions & 4 deletions upup/pkg/fi/cloudup/azuretasks/loadbalancer.go
Original file line number Diff line number Diff line change
Expand Up @@ -131,10 +131,9 @@ func (*LoadBalancer) RenderAzure(t *azure.AzureAPITarget, a, e, changes *LoadBal
idPrefix := fmt.Sprintf("subscriptions/%s/resourceGroups/%s/providers/Microsoft.Network", t.Cloud.SubscriptionID(), *e.ResourceGroup.Name)
feConfigProperties := &network.FrontendIPConfigurationPropertiesFormat{}
if *e.External {
// TODO: Implement public load balancer
// feConfigProperties.PublicIPAddress = &network.PublicIPAddress{
// ID: to.StringPtr(fmt.Sprintf("/%s/publicIPAddresses/%s", idPrefix, *e.PublicIPName)),
// }
feConfigProperties.PublicIPAddress = &network.PublicIPAddress{
ID: to.StringPtr(fmt.Sprintf("/%s/publicIPAddresses/%s", idPrefix, *e.Name)),
}
} else {
feConfigProperties.PrivateIPAllocationMethod = network.Dynamic
feConfigProperties.Subnet = &network.Subnet{
Expand Down
123 changes: 123 additions & 0 deletions upup/pkg/fi/cloudup/azuretasks/publicipaddress.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,123 @@
/*
Copyright 2020 The Kubernetes Authors.

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/

package azuretasks

import (
"context"

"github.com/Azure/azure-sdk-for-go/services/network/mgmt/2020-06-01/network"
"github.com/Azure/go-autorest/autorest/to"
"k8s.io/klog/v2"
"k8s.io/kops/upup/pkg/fi"
"k8s.io/kops/upup/pkg/fi/cloudup/azure"
)

//go:generate fitask -type=PublicIPAddress

// PublicIPAddress is an Azure Cloud Public IP Address
type PublicIPAddress struct {
Name *string
Lifecycle *fi.Lifecycle
ResourceGroup *ResourceGroup

Tags map[string]*string
}

var _ fi.Task = &PublicIPAddress{}
var _ fi.CompareWithID = &PublicIPAddress{}

// CompareWithID returns the Name of the Public IP Address
func (p *PublicIPAddress) CompareWithID() *string {
return p.Name
}

// Find discovers the Public IP Address in the cloud provider
func (p *PublicIPAddress) Find(c *fi.Context) (*PublicIPAddress, error) {
cloud := c.Cloud.(azure.AzureCloud)
l, err := cloud.PublicIPAddress().List(context.TODO(), *p.ResourceGroup.Name)
if err != nil {
return nil, err
}
var found *network.PublicIPAddress
for _, v := range l {
if *v.Name == *p.Name {
found = &v
break
}
}
if found == nil {
return nil, nil
}

return &PublicIPAddress{
Name: p.Name,
Lifecycle: p.Lifecycle,
ResourceGroup: &ResourceGroup{
Name: p.ResourceGroup.Name,
},

Tags: found.Tags,
}, nil
}

// Run implements fi.Task.Run.
func (p *PublicIPAddress) Run(c *fi.Context) error {
c.Cloud.(azure.AzureCloud).AddClusterTags(p.Tags)
return fi.DefaultDeltaRunMethod(p, c)
}

// CheckChanges returns an error if a change is not allowed.
func (*PublicIPAddress) CheckChanges(a, e, changes *PublicIPAddress) error {
if a == nil {
// Check if required fields are set when a new resource is created.
if e.Name == nil {
return fi.RequiredField("Name")
}
return nil
}

// Check if unchanegable fields won't be changed.
if changes.Name != nil {
return fi.CannotChangeField("Name")
}
return nil
}

// RenderAzure creates or updates a Public IP Address.
func (*PublicIPAddress) RenderAzure(t *azure.AzureAPITarget, a, e, changes *PublicIPAddress) error {
if a == nil {
klog.Infof("Creating a new Public IP Address with name: %s", fi.StringValue(e.Name))
} else {
klog.Infof("Updating a Public IP Address with name: %s", fi.StringValue(e.Name))
}

p := network.PublicIPAddress{
Location: to.StringPtr(t.Cloud.Region()),
Name: to.StringPtr(*e.Name),
PublicIPAddressPropertiesFormat: &network.PublicIPAddressPropertiesFormat{
PublicIPAddressVersion: network.IPv4,
PublicIPAllocationMethod: network.Dynamic,
},
Tags: e.Tags,
}

return t.Cloud.PublicIPAddress().CreateOrUpdate(
context.TODO(),
*e.ResourceGroup.Name,
*e.Name,
p)
}
Loading