mirror of https://github.com/kubernetes/kops.git
204 lines
5.4 KiB
Go
204 lines
5.4 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"
|
|
|
|
"github.com/gophercloud/gophercloud/v2"
|
|
"github.com/gophercloud/gophercloud/v2/openstack/networking/v2/subnets"
|
|
"k8s.io/klog/v2"
|
|
"k8s.io/kops/upup/pkg/fi"
|
|
"k8s.io/kops/upup/pkg/fi/cloudup/openstack"
|
|
)
|
|
|
|
// +kops:fitask
|
|
type Subnet struct {
|
|
ID *string
|
|
Name *string
|
|
Network *Network
|
|
CIDR *string
|
|
DNSServers []*string
|
|
Tag *string
|
|
Lifecycle fi.Lifecycle
|
|
}
|
|
|
|
// GetDependencies returns the dependencies of the Port task
|
|
func (e *Subnet) GetDependencies(tasks map[string]fi.CloudupTask) []fi.CloudupTask {
|
|
var deps []fi.CloudupTask
|
|
for _, task := range tasks {
|
|
if _, ok := task.(*Network); ok {
|
|
deps = append(deps, task)
|
|
}
|
|
}
|
|
return deps
|
|
}
|
|
|
|
var _ fi.CompareWithID = &Subnet{}
|
|
|
|
func (s *Subnet) CompareWithID() *string {
|
|
return s.ID
|
|
}
|
|
|
|
func NewSubnetTaskFromCloud(cloud openstack.OpenstackCloud, lifecycle fi.Lifecycle, subnet *subnets.Subnet, find *Subnet) (*Subnet, error) {
|
|
network, err := cloud.GetNetwork(subnet.NetworkID)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("NewSubnetTaskFromCloud: Failed to get network with ID %s: %v", subnet.NetworkID, err)
|
|
}
|
|
networkTask, err := NewNetworkTaskFromCloud(cloud, lifecycle, network, find.Tag)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("error creating network task from cloud: %v", err)
|
|
}
|
|
|
|
nameservers := make([]*string, len(subnet.DNSNameservers))
|
|
for i, ns := range subnet.DNSNameservers {
|
|
nameservers[i] = fi.PtrTo(ns)
|
|
}
|
|
|
|
tag := ""
|
|
if find != nil && fi.ArrayContains(subnet.Tags, fi.ValueOf(find.Tag)) {
|
|
tag = fi.ValueOf(find.Tag)
|
|
}
|
|
|
|
actual := &Subnet{
|
|
ID: fi.PtrTo(subnet.ID),
|
|
Name: fi.PtrTo(subnet.Name),
|
|
Network: networkTask,
|
|
CIDR: fi.PtrTo(subnet.CIDR),
|
|
Lifecycle: lifecycle,
|
|
DNSServers: nameservers,
|
|
Tag: fi.PtrTo(tag),
|
|
}
|
|
if find != nil {
|
|
find.ID = actual.ID
|
|
}
|
|
return actual, nil
|
|
}
|
|
|
|
func (s *Subnet) Find(context *fi.CloudupContext) (*Subnet, error) {
|
|
cloud := context.T.Cloud.(openstack.OpenstackCloud)
|
|
opt := subnets.ListOpts{
|
|
ID: fi.ValueOf(s.ID),
|
|
Name: fi.ValueOf(s.Name),
|
|
NetworkID: fi.ValueOf(s.Network.ID),
|
|
CIDR: fi.ValueOf(s.CIDR),
|
|
EnableDHCP: fi.PtrTo(true),
|
|
IPVersion: 4,
|
|
}
|
|
rs, err := cloud.ListSubnets(opt)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if rs == nil {
|
|
return nil, nil
|
|
} else if len(rs) != 1 {
|
|
return nil, fmt.Errorf("found multiple subnets with name: %s", fi.ValueOf(s.Name))
|
|
}
|
|
return NewSubnetTaskFromCloud(cloud, s.Lifecycle, &rs[0], s)
|
|
}
|
|
|
|
func (s *Subnet) Run(context *fi.CloudupContext) error {
|
|
return fi.CloudupDefaultDeltaRunMethod(s, context)
|
|
}
|
|
|
|
func (*Subnet) CheckChanges(a, e, changes *Subnet) error {
|
|
if a == nil {
|
|
if e.Name == nil {
|
|
return fi.RequiredField("Name")
|
|
}
|
|
if e.Network == nil {
|
|
return fi.RequiredField("Network")
|
|
}
|
|
if e.CIDR == nil {
|
|
return fi.RequiredField("CIDR")
|
|
}
|
|
} else {
|
|
if changes.Name != nil {
|
|
return fi.CannotChangeField("Name")
|
|
}
|
|
if changes.Network != nil {
|
|
return fi.CannotChangeField("Network")
|
|
}
|
|
if changes.CIDR != nil {
|
|
return fi.CannotChangeField("CIDR")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (*Subnet) RenderOpenstack(t *openstack.OpenstackAPITarget, a, e, changes *Subnet) error {
|
|
if a == nil {
|
|
klog.V(2).Infof("Creating Subnet with name:%q", fi.ValueOf(e.Name))
|
|
|
|
opt := subnets.CreateOpts{
|
|
Name: fi.ValueOf(e.Name),
|
|
NetworkID: fi.ValueOf(e.Network.ID),
|
|
IPVersion: gophercloud.IPv4,
|
|
CIDR: fi.ValueOf(e.CIDR),
|
|
EnableDHCP: fi.PtrTo(true),
|
|
}
|
|
|
|
if len(e.DNSServers) > 0 {
|
|
dnsNameSrv := make([]string, len(e.DNSServers))
|
|
for i, ns := range e.DNSServers {
|
|
dnsNameSrv[i] = fi.ValueOf(ns)
|
|
}
|
|
opt.DNSNameservers = dnsNameSrv
|
|
}
|
|
v, err := t.Cloud.CreateSubnet(opt)
|
|
if err != nil {
|
|
return fmt.Errorf("Error creating subnet: %v", err)
|
|
}
|
|
|
|
err = t.Cloud.AppendTag(openstack.ResourceTypeSubnet, v.ID, fi.ValueOf(e.Tag))
|
|
if err != nil {
|
|
return fmt.Errorf("Error appending tag to subnet: %v", err)
|
|
}
|
|
|
|
e.ID = fi.PtrTo(v.ID)
|
|
klog.V(2).Infof("Creating a new Openstack subnet, id=%s", v.ID)
|
|
return nil
|
|
} else {
|
|
if changes.Tag != nil {
|
|
err := t.Cloud.AppendTag(openstack.ResourceTypeSubnet, fi.ValueOf(a.ID), fi.ValueOf(changes.Tag))
|
|
if err != nil {
|
|
return fmt.Errorf("error appending tag to subnet: %v", err)
|
|
}
|
|
}
|
|
client := t.Cloud.NetworkingClient()
|
|
|
|
opt := subnets.UpdateOpts{}
|
|
|
|
if changes.DNSServers != nil {
|
|
dnsNameSrv := make([]string, len(e.DNSServers))
|
|
for i, ns := range e.DNSServers {
|
|
dnsNameSrv[i] = fi.ValueOf(ns)
|
|
}
|
|
opt.DNSNameservers = &dnsNameSrv
|
|
}
|
|
result := subnets.Update(context.TODO(), client, fi.ValueOf(a.ID), opt)
|
|
klog.Infof("Updated %v", opt)
|
|
if result.Err != nil {
|
|
return fmt.Errorf("error updating subnet %v: %v", a.ID, result.Err)
|
|
}
|
|
}
|
|
e.ID = a.ID
|
|
klog.V(2).Infof("Using an existing Openstack subnet, id=%s", fi.ValueOf(e.ID))
|
|
return nil
|
|
}
|