mirror of https://github.com/kubernetes/kops.git
314 lines
8.1 KiB
Go
314 lines
8.1 KiB
Go
/*
|
|
Copyright 2019 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 openstacktasks
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/gophercloud/gophercloud/v2/openstack/networking/v2/ports"
|
|
|
|
l3floatingip "github.com/gophercloud/gophercloud/v2/openstack/networking/v2/extensions/layer3/floatingips"
|
|
"k8s.io/apimachinery/pkg/util/wait"
|
|
"k8s.io/klog/v2"
|
|
"k8s.io/kops/pkg/wellknownservices"
|
|
"k8s.io/kops/upup/pkg/fi"
|
|
"k8s.io/kops/upup/pkg/fi/cloudup/openstack"
|
|
"k8s.io/kops/util/pkg/vfs"
|
|
)
|
|
|
|
// +kops:fitask
|
|
type FloatingIP struct {
|
|
Name *string
|
|
ID *string
|
|
LB *LB
|
|
IP *string
|
|
Lifecycle fi.Lifecycle
|
|
|
|
// WellKnownServices indicates which services are supported by this resource.
|
|
// This field is internal and is not rendered to the cloud.
|
|
WellKnownServices []wellknownservices.WellKnownService
|
|
}
|
|
|
|
var _ fi.HasAddress = &FloatingIP{}
|
|
|
|
var readBackoff = wait.Backoff{
|
|
Duration: time.Second,
|
|
Factor: 1.5,
|
|
Jitter: 0.1,
|
|
Steps: 10,
|
|
}
|
|
|
|
// this function tries to find l3 floating, and retries x times to find that. In some cases the floatingip is not in place in first request
|
|
func findL3Floating(cloud openstack.OpenstackCloud, opts l3floatingip.ListOpts) ([]l3floatingip.FloatingIP, error) {
|
|
var result []l3floatingip.FloatingIP
|
|
done, err := vfs.RetryWithBackoff(readBackoff, func() (bool, error) {
|
|
fips, err := cloud.ListL3FloatingIPs(opts)
|
|
if err != nil {
|
|
return false, fmt.Errorf("Failed to list L3 floating ip: %v", err)
|
|
}
|
|
if len(fips) == 0 {
|
|
return false, nil
|
|
}
|
|
result = fips
|
|
return true, nil
|
|
})
|
|
if !done {
|
|
if err == nil {
|
|
err = wait.ErrWaitTimeout
|
|
}
|
|
return result, err
|
|
}
|
|
return result, nil
|
|
}
|
|
|
|
// GetWellKnownServices implements fi.HasAddress::GetWellKnownServices.
|
|
// It indicates which services we support with this address.
|
|
func (e *FloatingIP) GetWellKnownServices() []wellknownservices.WellKnownService {
|
|
return e.WellKnownServices
|
|
}
|
|
|
|
func (e *FloatingIP) FindAddresses(context *fi.CloudupContext) ([]string, error) {
|
|
if e.ID == nil {
|
|
if e.LB != nil && e.LB.ID == nil {
|
|
return nil, nil
|
|
}
|
|
}
|
|
|
|
cloud := context.T.Cloud.(openstack.OpenstackCloud)
|
|
// try to find ip address using LB port
|
|
if e.ID == nil && e.LB != nil && e.LB.PortID != nil {
|
|
fips, err := findL3Floating(cloud, l3floatingip.ListOpts{
|
|
PortID: fi.ValueOf(e.LB.PortID),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if len(fips) == 1 && fips[0].PortID == fi.ValueOf(e.LB.PortID) {
|
|
return []string{fips[0].FloatingIP}, nil
|
|
}
|
|
return nil, fmt.Errorf("Could not find port floatingips port=%s", fi.ValueOf(e.LB.PortID))
|
|
}
|
|
|
|
fip, err := cloud.GetL3FloatingIP(fi.ValueOf(e.ID))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return []string{fip.FloatingIP}, nil
|
|
}
|
|
|
|
// GetDependencies returns the dependencies of the Instance task
|
|
func (e *FloatingIP) GetDependencies(tasks map[string]fi.CloudupTask) []fi.CloudupTask {
|
|
var deps []fi.CloudupTask
|
|
for _, task := range tasks {
|
|
if _, ok := task.(*LB); ok {
|
|
deps = append(deps, task)
|
|
}
|
|
// We can't create a floating IP until the router with access to the external network
|
|
// Has created an interface to our subnet
|
|
if _, ok := task.(*RouterInterface); ok {
|
|
deps = append(deps, task)
|
|
}
|
|
}
|
|
return deps
|
|
}
|
|
|
|
var _ fi.CompareWithID = &FloatingIP{}
|
|
|
|
func (e *FloatingIP) CompareWithID() *string {
|
|
return e.ID
|
|
}
|
|
|
|
func (e *FloatingIP) Find(c *fi.CloudupContext) (*FloatingIP, error) {
|
|
if e == nil {
|
|
return nil, nil
|
|
}
|
|
cloud := c.T.Cloud.(openstack.OpenstackCloud)
|
|
if e.LB != nil && e.LB.PortID != nil {
|
|
fip, err := findFipByPortID(cloud, fi.ValueOf(e.LB.PortID))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to find floating ip: %v", err)
|
|
}
|
|
if fip == nil {
|
|
return nil, nil
|
|
}
|
|
actual := &FloatingIP{
|
|
Name: fi.PtrTo(fip.Description),
|
|
ID: fi.PtrTo(fip.ID),
|
|
LB: e.LB,
|
|
Lifecycle: e.Lifecycle,
|
|
}
|
|
e.ID = actual.ID
|
|
return actual, nil
|
|
}
|
|
fipname := fi.ValueOf(e.Name)
|
|
fips, err := cloud.ListL3FloatingIPs(l3floatingip.ListOpts{
|
|
Description: fipname,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to list layer 3 floating ips: %v", err)
|
|
}
|
|
|
|
for _, fip := range fips {
|
|
if fip.Description == fi.ValueOf(e.Name) {
|
|
actual := &FloatingIP{
|
|
ID: fi.PtrTo(fips[0].ID),
|
|
Name: e.Name,
|
|
IP: fi.PtrTo(fip.FloatingIP),
|
|
Lifecycle: e.Lifecycle,
|
|
}
|
|
e.ID = actual.ID
|
|
e.IP = actual.IP
|
|
return actual, nil
|
|
}
|
|
}
|
|
|
|
if len(fips) == 0 {
|
|
// If we fail to find an IP address we need to look for IP addresses attached to a port with similar name
|
|
// TODO: remove this in kops 1.21 where we can expect that the description field has been added
|
|
portname := "port-" + strings.TrimPrefix(fipname, "fip-")
|
|
|
|
ports, err := cloud.ListPorts(ports.ListOpts{
|
|
Name: portname,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to list ports: %v", err)
|
|
}
|
|
|
|
if len(ports) == 1 {
|
|
|
|
fip, err := findFipByPortID(cloud, ports[0].ID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to find floating ip: %v", err)
|
|
}
|
|
if fip == nil {
|
|
return nil, nil
|
|
}
|
|
actual := &FloatingIP{
|
|
Name: fi.PtrTo(fip.Description),
|
|
ID: fi.PtrTo(fip.ID),
|
|
Lifecycle: e.Lifecycle,
|
|
}
|
|
e.ID = actual.ID
|
|
return actual, nil
|
|
}
|
|
|
|
}
|
|
return nil, nil
|
|
}
|
|
|
|
func findFipByPortID(cloud openstack.OpenstackCloud, id string) (fip *l3floatingip.FloatingIP, err error) {
|
|
fips, err := cloud.ListL3FloatingIPs(l3floatingip.ListOpts{
|
|
PortID: id,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to list layer 3 floating ips for port ID %s: %v", id, err)
|
|
}
|
|
if len(fips) == 0 {
|
|
return nil, nil
|
|
}
|
|
if len(fips) > 1 {
|
|
return nil, fmt.Errorf("multiple floating ips associated to port: %s", id)
|
|
}
|
|
return &fips[0], nil
|
|
}
|
|
|
|
func (e *FloatingIP) Run(c *fi.CloudupContext) error {
|
|
return fi.CloudupDefaultDeltaRunMethod(e, c)
|
|
}
|
|
|
|
func (_ *FloatingIP) CheckChanges(a, e, changes *FloatingIP) error {
|
|
if a == nil {
|
|
if e.Name == nil {
|
|
return fi.RequiredField("Name")
|
|
}
|
|
} else {
|
|
if changes.ID != nil {
|
|
return fi.CannotChangeField("ID")
|
|
}
|
|
//TODO: add back into kops 1.21
|
|
/*
|
|
if changes.Name != nil && fi.ValueOf(a.Name) != "" {
|
|
return fi.CannotChangeField("Name")
|
|
}
|
|
*/
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (_ *FloatingIP) ShouldCreate(a, e, changes *FloatingIP) (bool, error) {
|
|
if a == nil {
|
|
return true, nil
|
|
}
|
|
if changes.Name != nil {
|
|
return true, nil
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
func (f *FloatingIP) RenderOpenstack(t *openstack.OpenstackAPITarget, a, e, changes *FloatingIP) error {
|
|
cloud := t.Cloud
|
|
|
|
if a == nil {
|
|
external, err := cloud.GetExternalNetwork()
|
|
if err != nil {
|
|
return fmt.Errorf("Failed to find external network: %v", err)
|
|
}
|
|
|
|
opts := l3floatingip.CreateOpts{
|
|
FloatingNetworkID: external.ID,
|
|
Description: fi.ValueOf(e.Name),
|
|
}
|
|
|
|
if e.LB != nil {
|
|
opts.PortID = fi.ValueOf(e.LB.PortID)
|
|
}
|
|
|
|
// instance floatingips comes from the same subnet as the kubernetes API floatingip
|
|
lbSubnet, err := cloud.GetLBFloatingSubnet()
|
|
if err != nil {
|
|
return fmt.Errorf("Failed to find floatingip subnet: %v", err)
|
|
}
|
|
if lbSubnet != nil {
|
|
opts.SubnetID = lbSubnet.ID
|
|
}
|
|
fip, err := cloud.CreateL3FloatingIP(opts)
|
|
if err != nil {
|
|
return fmt.Errorf("Failed to create floating IP: %v", err)
|
|
}
|
|
|
|
e.ID = fi.PtrTo(fip.ID)
|
|
e.IP = fi.PtrTo(fip.FloatingIP)
|
|
|
|
return nil
|
|
}
|
|
if changes.Name != nil {
|
|
_, err := l3floatingip.Update(context.TODO(), cloud.NetworkingClient(), fi.ValueOf(a.ID), l3floatingip.UpdateOpts{
|
|
Description: e.Name,
|
|
}).Extract()
|
|
if err != nil {
|
|
return fmt.Errorf("failed to update floating ip %v: %v", fi.ValueOf(e.Name), err)
|
|
}
|
|
|
|
}
|
|
|
|
klog.V(2).Infof("Openstack task Instance::RenderOpenstack did nothing")
|
|
return nil
|
|
}
|