2014-10-08 01:29:28 +00:00
|
|
|
/*
|
2016-06-03 00:25:58 +00:00
|
|
|
Copyright 2014 The Kubernetes Authors.
|
2014-10-08 01:29:28 +00:00
|
|
|
|
|
|
|
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 openstack
|
|
|
|
|
|
|
|
import (
|
2017-02-13 07:50:04 +00:00
|
|
|
"crypto/tls"
|
2014-10-09 15:21:06 +00:00
|
|
|
"errors"
|
2014-10-09 13:22:01 +00:00
|
|
|
"fmt"
|
2014-10-08 01:29:28 +00:00
|
|
|
"io"
|
2015-11-12 10:10:38 +00:00
|
|
|
"io/ioutil"
|
2015-05-06 18:52:21 +00:00
|
|
|
"net/http"
|
2014-10-09 15:21:06 +00:00
|
|
|
"regexp"
|
2017-01-12 19:20:20 +00:00
|
|
|
"sort"
|
2015-04-10 16:54:01 +00:00
|
|
|
"strings"
|
2014-10-30 03:16:50 +00:00
|
|
|
"time"
|
2014-10-08 01:29:28 +00:00
|
|
|
|
2016-11-07 08:35:42 +00:00
|
|
|
"github.com/gophercloud/gophercloud"
|
|
|
|
"github.com/gophercloud/gophercloud/openstack"
|
2017-01-12 19:20:20 +00:00
|
|
|
apiversions_v1 "github.com/gophercloud/gophercloud/openstack/blockstorage/v1/apiversions"
|
2017-08-08 15:29:37 +00:00
|
|
|
"github.com/gophercloud/gophercloud/openstack/compute/v2/extensions/attachinterfaces"
|
2016-11-07 08:35:42 +00:00
|
|
|
"github.com/gophercloud/gophercloud/openstack/compute/v2/servers"
|
|
|
|
"github.com/gophercloud/gophercloud/openstack/identity/v3/extensions/trusts"
|
|
|
|
tokens3 "github.com/gophercloud/gophercloud/openstack/identity/v3/tokens"
|
|
|
|
"github.com/gophercloud/gophercloud/pagination"
|
2016-09-02 06:24:54 +00:00
|
|
|
"github.com/mitchellh/mapstructure"
|
|
|
|
"gopkg.in/gcfg.v1"
|
2014-10-09 13:22:01 +00:00
|
|
|
|
2015-08-05 22:05:17 +00:00
|
|
|
"github.com/golang/glog"
|
2017-06-22 18:24:23 +00:00
|
|
|
"k8s.io/api/core/v1"
|
2017-01-11 14:09:48 +00:00
|
|
|
"k8s.io/apimachinery/pkg/types"
|
2017-02-13 07:50:04 +00:00
|
|
|
netutil "k8s.io/apimachinery/pkg/util/net"
|
|
|
|
certutil "k8s.io/client-go/util/cert"
|
2017-04-13 20:19:08 +00:00
|
|
|
v1helper "k8s.io/kubernetes/pkg/api/v1/helper"
|
2015-08-05 22:03:47 +00:00
|
|
|
"k8s.io/kubernetes/pkg/cloudprovider"
|
2017-05-17 21:38:25 +00:00
|
|
|
"k8s.io/kubernetes/pkg/controller"
|
2014-10-08 01:29:28 +00:00
|
|
|
)
|
|
|
|
|
2017-08-25 15:08:00 +00:00
|
|
|
const (
|
|
|
|
ProviderName = "openstack"
|
|
|
|
AvailabilityZone = "availability_zone"
|
|
|
|
)
|
2015-05-05 14:10:24 +00:00
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
var ErrNotFound = errors.New("Failed to find object")
|
|
|
|
var ErrMultipleResults = errors.New("Multiple results where only one expected")
|
2014-10-29 08:30:55 +00:00
|
|
|
var ErrNoAddressFound = errors.New("No address found for host")
|
2014-10-09 15:21:06 +00:00
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
// encoding.TextUnmarshaler interface for time.Duration
|
|
|
|
type MyDuration struct {
|
|
|
|
time.Duration
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *MyDuration) UnmarshalText(text []byte) error {
|
|
|
|
res, err := time.ParseDuration(string(text))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
d.Duration = res
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-05-17 22:05:03 +00:00
|
|
|
type LoadBalancer struct {
|
|
|
|
network *gophercloud.ServiceClient
|
|
|
|
compute *gophercloud.ServiceClient
|
|
|
|
opts LoadBalancerOpts
|
|
|
|
}
|
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
type LoadBalancerOpts struct {
|
2017-07-07 09:05:21 +00:00
|
|
|
LBVersion string `gcfg:"lb-version"` // overrides autodetection. v1 or v2
|
2017-08-16 06:20:43 +00:00
|
|
|
SubnetId string `gcfg:"subnet-id"` // overrides autodetection.
|
2017-07-07 09:05:21 +00:00
|
|
|
FloatingNetworkId string `gcfg:"floating-network-id"` // If specified, will create floating ip for loadbalancer, or do not create floating ip.
|
|
|
|
LBMethod string `gcfg:"lb-method"` // default to ROUND_ROBIN.
|
2016-09-07 00:28:09 +00:00
|
|
|
CreateMonitor bool `gcfg:"create-monitor"`
|
|
|
|
MonitorDelay MyDuration `gcfg:"monitor-delay"`
|
|
|
|
MonitorTimeout MyDuration `gcfg:"monitor-timeout"`
|
|
|
|
MonitorMaxRetries uint `gcfg:"monitor-max-retries"`
|
|
|
|
ManageSecurityGroups bool `gcfg:"manage-security-groups"`
|
|
|
|
NodeSecurityGroupID string `gcfg:"node-security-group"`
|
2014-10-30 03:16:50 +00:00
|
|
|
}
|
|
|
|
|
2016-09-22 11:59:43 +00:00
|
|
|
type BlockStorageOpts struct {
|
2017-01-12 19:20:20 +00:00
|
|
|
BSVersion string `gcfg:"bs-version"` // overrides autodetection. v1 or v2. Defaults to auto
|
|
|
|
TrustDevicePath bool `gcfg:"trust-device-path"` // See Issue #33128
|
2016-09-22 11:59:43 +00:00
|
|
|
}
|
|
|
|
|
2016-09-02 06:24:54 +00:00
|
|
|
type RouterOpts struct {
|
|
|
|
RouterId string `gcfg:"router-id"` // required
|
|
|
|
}
|
|
|
|
|
2017-09-18 03:32:28 +00:00
|
|
|
type MetadataOpts struct {
|
|
|
|
SearchOrder string `gcfg:"search-order"`
|
|
|
|
}
|
|
|
|
|
2014-10-08 01:29:28 +00:00
|
|
|
// OpenStack is an implementation of cloud provider Interface for OpenStack.
|
2014-10-09 13:22:01 +00:00
|
|
|
type OpenStack struct {
|
2017-09-18 03:32:28 +00:00
|
|
|
provider *gophercloud.ProviderClient
|
|
|
|
region string
|
|
|
|
lbOpts LoadBalancerOpts
|
|
|
|
bsOpts BlockStorageOpts
|
|
|
|
routeOpts RouterOpts
|
|
|
|
metadataOpts MetadataOpts
|
2015-11-12 10:10:38 +00:00
|
|
|
// InstanceID of the server where this OpenStack object is instantiated.
|
|
|
|
localInstanceID string
|
2014-10-09 13:22:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
|
|
|
Global struct {
|
2014-10-30 03:16:50 +00:00
|
|
|
AuthUrl string `gcfg:"auth-url"`
|
|
|
|
Username string
|
|
|
|
UserId string `gcfg:"user-id"`
|
|
|
|
Password string
|
|
|
|
TenantId string `gcfg:"tenant-id"`
|
|
|
|
TenantName string `gcfg:"tenant-name"`
|
2016-09-06 14:54:02 +00:00
|
|
|
TrustId string `gcfg:"trust-id"`
|
2014-10-30 03:16:50 +00:00
|
|
|
DomainId string `gcfg:"domain-id"`
|
|
|
|
DomainName string `gcfg:"domain-name"`
|
|
|
|
Region string
|
2017-02-13 07:50:04 +00:00
|
|
|
CAFile string `gcfg:"ca-file"`
|
2014-10-09 13:22:01 +00:00
|
|
|
}
|
2014-10-30 03:16:50 +00:00
|
|
|
LoadBalancer LoadBalancerOpts
|
2016-09-22 11:59:43 +00:00
|
|
|
BlockStorage BlockStorageOpts
|
2016-09-02 06:24:54 +00:00
|
|
|
Route RouterOpts
|
2017-09-18 03:32:28 +00:00
|
|
|
Metadata MetadataOpts
|
2014-10-09 13:22:01 +00:00
|
|
|
}
|
2014-10-08 01:29:28 +00:00
|
|
|
|
|
|
|
func init() {
|
2017-05-17 09:37:43 +00:00
|
|
|
RegisterMetrics()
|
|
|
|
|
2015-05-05 14:10:24 +00:00
|
|
|
cloudprovider.RegisterCloudProvider(ProviderName, func(config io.Reader) (cloudprovider.Interface, error) {
|
2014-10-09 13:22:01 +00:00
|
|
|
cfg, err := readConfig(config)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return newOpenStack(cfg)
|
2014-10-08 01:29:28 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2014-10-09 13:22:01 +00:00
|
|
|
func (cfg Config) toAuthOptions() gophercloud.AuthOptions {
|
|
|
|
return gophercloud.AuthOptions{
|
2014-10-29 08:30:55 +00:00
|
|
|
IdentityEndpoint: cfg.Global.AuthUrl,
|
|
|
|
Username: cfg.Global.Username,
|
|
|
|
UserID: cfg.Global.UserId,
|
|
|
|
Password: cfg.Global.Password,
|
|
|
|
TenantID: cfg.Global.TenantId,
|
|
|
|
TenantName: cfg.Global.TenantName,
|
2015-12-01 22:12:25 +00:00
|
|
|
DomainID: cfg.Global.DomainId,
|
|
|
|
DomainName: cfg.Global.DomainName,
|
2014-10-09 13:22:01 +00:00
|
|
|
|
2015-04-18 09:41:54 +00:00
|
|
|
// Persistent service, so we need to be able to renew tokens.
|
2014-10-09 13:22:01 +00:00
|
|
|
AllowReauth: true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-07 08:35:42 +00:00
|
|
|
func (cfg Config) toAuth3Options() tokens3.AuthOptions {
|
|
|
|
return tokens3.AuthOptions{
|
|
|
|
IdentityEndpoint: cfg.Global.AuthUrl,
|
|
|
|
Username: cfg.Global.Username,
|
|
|
|
UserID: cfg.Global.UserId,
|
|
|
|
Password: cfg.Global.Password,
|
|
|
|
DomainID: cfg.Global.DomainId,
|
|
|
|
DomainName: cfg.Global.DomainName,
|
|
|
|
AllowReauth: true,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-09 13:22:01 +00:00
|
|
|
func readConfig(config io.Reader) (Config, error) {
|
|
|
|
if config == nil {
|
2014-11-20 10:00:36 +00:00
|
|
|
err := fmt.Errorf("no OpenStack cloud provider config file given")
|
2014-10-09 13:22:01 +00:00
|
|
|
return Config{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var cfg Config
|
2016-09-22 11:59:43 +00:00
|
|
|
|
|
|
|
// Set default values for config params
|
2017-01-12 19:20:20 +00:00
|
|
|
cfg.BlockStorage.BSVersion = "auto"
|
2016-09-22 11:59:43 +00:00
|
|
|
cfg.BlockStorage.TrustDevicePath = false
|
2017-09-18 03:32:28 +00:00
|
|
|
cfg.Metadata.SearchOrder = fmt.Sprintf("%s,%s", configDriveID, metadataID)
|
2016-09-22 11:59:43 +00:00
|
|
|
|
2014-10-09 13:22:01 +00:00
|
|
|
err := gcfg.ReadInto(&cfg, config)
|
|
|
|
return cfg, err
|
|
|
|
}
|
|
|
|
|
2016-09-02 06:24:54 +00:00
|
|
|
// Tiny helper for conditional unwind logic
|
|
|
|
type Caller bool
|
|
|
|
|
|
|
|
func NewCaller() Caller { return Caller(true) }
|
|
|
|
func (c *Caller) Disarm() { *c = false }
|
|
|
|
|
|
|
|
func (c *Caller) Call(f func()) {
|
|
|
|
if *c {
|
|
|
|
f()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-18 03:32:28 +00:00
|
|
|
func readInstanceID(searchOrder string) (string, error) {
|
2015-11-12 10:10:38 +00:00
|
|
|
// Try to find instance ID on the local filesystem (created by cloud-init)
|
|
|
|
const instanceIDFile = "/var/lib/cloud/data/instance-id"
|
|
|
|
idBytes, err := ioutil.ReadFile(instanceIDFile)
|
|
|
|
if err == nil {
|
|
|
|
instanceID := string(idBytes)
|
|
|
|
instanceID = strings.TrimSpace(instanceID)
|
|
|
|
glog.V(3).Infof("Got instance id from %s: %s", instanceIDFile, instanceID)
|
|
|
|
if instanceID != "" {
|
|
|
|
return instanceID, nil
|
|
|
|
}
|
2016-08-30 02:26:25 +00:00
|
|
|
// Fall through to metadata server lookup
|
2015-11-12 10:10:38 +00:00
|
|
|
}
|
|
|
|
|
2017-09-18 03:32:28 +00:00
|
|
|
md, err := getMetadata(searchOrder)
|
2015-11-12 10:10:38 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2016-08-30 02:26:25 +00:00
|
|
|
return md.Uuid, nil
|
2015-11-12 10:10:38 +00:00
|
|
|
}
|
|
|
|
|
2017-07-07 09:05:21 +00:00
|
|
|
// check opts for OpenStack
|
|
|
|
func checkOpenStackOpts(openstackOpts *OpenStack) error {
|
|
|
|
lbOpts := openstackOpts.lbOpts
|
|
|
|
|
|
|
|
// if need to create health monitor for Neutron LB,
|
|
|
|
// monitor-delay, monitor-timeout and monitor-max-retries should be set.
|
|
|
|
emptyDuration := MyDuration{}
|
|
|
|
if lbOpts.CreateMonitor {
|
|
|
|
if lbOpts.MonitorDelay == emptyDuration {
|
|
|
|
return fmt.Errorf("monitor-delay not set in cloud provider config")
|
|
|
|
}
|
|
|
|
if lbOpts.MonitorTimeout == emptyDuration {
|
|
|
|
return fmt.Errorf("monitor-timeout not set in cloud provider config")
|
|
|
|
}
|
|
|
|
if lbOpts.MonitorMaxRetries == uint(0) {
|
|
|
|
return fmt.Errorf("monitor-max-retries not set in cloud provider config")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// if enable ManageSecurityGroups, node-security-group should be set.
|
|
|
|
if lbOpts.ManageSecurityGroups {
|
|
|
|
if len(lbOpts.NodeSecurityGroupID) == 0 {
|
|
|
|
return fmt.Errorf("node-security-group not set in cloud provider config")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2014-10-09 13:22:01 +00:00
|
|
|
func newOpenStack(cfg Config) (*OpenStack, error) {
|
2016-09-06 14:54:02 +00:00
|
|
|
provider, err := openstack.NewClient(cfg.Global.AuthUrl)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-02-13 07:50:04 +00:00
|
|
|
if cfg.Global.CAFile != "" {
|
|
|
|
roots, err := certutil.NewPool(cfg.Global.CAFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
config := &tls.Config{}
|
|
|
|
config.RootCAs = roots
|
|
|
|
provider.HTTPClient.Transport = netutil.SetOldTransportDefaults(&http.Transport{TLSClientConfig: config})
|
|
|
|
|
|
|
|
}
|
2016-09-06 14:54:02 +00:00
|
|
|
if cfg.Global.TrustId != "" {
|
2016-11-07 08:35:42 +00:00
|
|
|
opts := cfg.toAuth3Options()
|
|
|
|
authOptsExt := trusts.AuthOptsExt{
|
|
|
|
TrustID: cfg.Global.TrustId,
|
|
|
|
AuthOptionsBuilder: &opts,
|
2016-09-06 14:54:02 +00:00
|
|
|
}
|
2016-11-07 08:35:42 +00:00
|
|
|
err = openstack.AuthenticateV3(provider, authOptsExt, gophercloud.EndpointOpts{})
|
2016-09-06 14:54:02 +00:00
|
|
|
} else {
|
|
|
|
err = openstack.Authenticate(provider, cfg.toAuthOptions())
|
|
|
|
}
|
|
|
|
|
2017-09-18 03:32:28 +00:00
|
|
|
err = validateMetadataSearchOrder(cfg.Metadata.SearchOrder)
|
|
|
|
|
2014-10-29 08:30:55 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2014-10-09 13:22:01 +00:00
|
|
|
os := OpenStack{
|
2017-09-18 03:32:28 +00:00
|
|
|
provider: provider,
|
|
|
|
region: cfg.Global.Region,
|
|
|
|
lbOpts: cfg.LoadBalancer,
|
|
|
|
bsOpts: cfg.BlockStorage,
|
|
|
|
routeOpts: cfg.Route,
|
|
|
|
metadataOpts: cfg.Metadata,
|
2014-10-09 13:22:01 +00:00
|
|
|
}
|
2015-11-12 10:10:38 +00:00
|
|
|
|
2017-07-07 09:05:21 +00:00
|
|
|
err = checkOpenStackOpts(&os)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2014-10-29 08:30:55 +00:00
|
|
|
return &os, nil
|
2014-10-08 01:29:28 +00:00
|
|
|
}
|
|
|
|
|
2017-05-17 21:38:25 +00:00
|
|
|
// Initialize passes a Kubernetes clientBuilder interface to the cloud provider
|
|
|
|
func (os *OpenStack) Initialize(clientBuilder controller.ControllerClientBuilder) {}
|
|
|
|
|
2016-07-16 06:10:29 +00:00
|
|
|
// mapNodeNameToServerName maps a k8s NodeName to an OpenStack Server Name
|
|
|
|
// This is a simple string cast.
|
|
|
|
func mapNodeNameToServerName(nodeName types.NodeName) string {
|
|
|
|
return string(nodeName)
|
|
|
|
}
|
|
|
|
|
|
|
|
// mapServerToNodeName maps an OpenStack Server to a k8s NodeName
|
|
|
|
func mapServerToNodeName(server *servers.Server) types.NodeName {
|
2016-09-02 06:24:54 +00:00
|
|
|
// Node names are always lowercase, and (at least)
|
|
|
|
// routecontroller does case-sensitive string comparisons
|
|
|
|
// assuming this
|
|
|
|
return types.NodeName(strings.ToLower(server.Name))
|
|
|
|
}
|
|
|
|
|
|
|
|
func foreachServer(client *gophercloud.ServiceClient, opts servers.ListOptsBuilder, handler func(*servers.Server) (bool, error)) error {
|
|
|
|
pager := servers.List(client, opts)
|
|
|
|
|
|
|
|
err := pager.EachPage(func(page pagination.Page) (bool, error) {
|
|
|
|
s, err := servers.ExtractServers(page)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
for _, server := range s {
|
|
|
|
ok, err := handler(&server)
|
|
|
|
if !ok || err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
})
|
|
|
|
return err
|
2016-07-16 06:10:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func getServerByName(client *gophercloud.ServiceClient, name types.NodeName) (*servers.Server, error) {
|
2014-10-29 08:30:55 +00:00
|
|
|
opts := servers.ListOpts{
|
2016-07-16 06:10:29 +00:00
|
|
|
Name: fmt.Sprintf("^%s$", regexp.QuoteMeta(mapNodeNameToServerName(name))),
|
2014-10-29 08:30:55 +00:00
|
|
|
Status: "ACTIVE",
|
|
|
|
}
|
|
|
|
pager := servers.List(client, opts)
|
|
|
|
|
|
|
|
serverList := make([]servers.Server, 0, 1)
|
2014-10-09 15:21:06 +00:00
|
|
|
|
2014-10-29 08:30:55 +00:00
|
|
|
err := pager.EachPage(func(page pagination.Page) (bool, error) {
|
|
|
|
s, err := servers.ExtractServers(page)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
serverList = append(serverList, s...)
|
|
|
|
if len(serverList) > 1 {
|
2014-10-30 03:16:50 +00:00
|
|
|
return false, ErrMultipleResults
|
2014-10-29 08:30:55 +00:00
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
})
|
2014-10-09 15:21:06 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2014-10-29 08:30:55 +00:00
|
|
|
if len(serverList) == 0 {
|
2014-10-30 03:16:50 +00:00
|
|
|
return nil, ErrNotFound
|
2014-10-09 15:21:06 +00:00
|
|
|
}
|
|
|
|
|
2014-10-29 08:30:55 +00:00
|
|
|
return &serverList[0], nil
|
|
|
|
}
|
|
|
|
|
2016-09-02 06:24:54 +00:00
|
|
|
func nodeAddresses(srv *servers.Server) ([]v1.NodeAddress, error) {
|
|
|
|
addrs := []v1.NodeAddress{}
|
|
|
|
|
|
|
|
type Address struct {
|
|
|
|
IpType string `mapstructure:"OS-EXT-IPS:type"`
|
|
|
|
Addr string
|
|
|
|
}
|
|
|
|
|
|
|
|
var addresses map[string][]Address
|
|
|
|
err := mapstructure.Decode(srv.Addresses, &addresses)
|
2015-12-17 14:52:31 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2014-10-29 08:30:55 +00:00
|
|
|
}
|
2015-12-17 14:52:31 +00:00
|
|
|
|
2017-01-11 12:14:01 +00:00
|
|
|
for network, addrList := range addresses {
|
|
|
|
for _, props := range addrList {
|
2016-11-18 20:58:42 +00:00
|
|
|
var addressType v1.NodeAddressType
|
2016-09-02 06:24:54 +00:00
|
|
|
if props.IpType == "floating" || network == "public" {
|
2016-11-18 20:58:42 +00:00
|
|
|
addressType = v1.NodeExternalIP
|
2015-12-17 14:52:31 +00:00
|
|
|
} else {
|
2016-11-18 20:58:42 +00:00
|
|
|
addressType = v1.NodeInternalIP
|
2015-12-17 14:52:31 +00:00
|
|
|
}
|
|
|
|
|
2017-04-13 20:19:08 +00:00
|
|
|
v1helper.AddToNodeAddresses(&addrs,
|
2016-11-18 20:58:42 +00:00
|
|
|
v1.NodeAddress{
|
2015-12-17 14:52:31 +00:00
|
|
|
Type: addressType,
|
2016-09-02 06:24:54 +00:00
|
|
|
Address: props.Addr,
|
2015-12-17 14:52:31 +00:00
|
|
|
},
|
|
|
|
)
|
2015-03-16 06:11:36 +00:00
|
|
|
}
|
2014-10-29 08:30:55 +00:00
|
|
|
}
|
2015-12-17 14:52:31 +00:00
|
|
|
|
|
|
|
// AccessIPs are usually duplicates of "public" addresses.
|
|
|
|
if srv.AccessIPv4 != "" {
|
2017-04-13 20:19:08 +00:00
|
|
|
v1helper.AddToNodeAddresses(&addrs,
|
2016-11-18 20:58:42 +00:00
|
|
|
v1.NodeAddress{
|
|
|
|
Type: v1.NodeExternalIP,
|
2015-12-17 14:52:31 +00:00
|
|
|
Address: srv.AccessIPv4,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if srv.AccessIPv6 != "" {
|
2017-04-13 20:19:08 +00:00
|
|
|
v1helper.AddToNodeAddresses(&addrs,
|
2016-11-18 20:58:42 +00:00
|
|
|
v1.NodeAddress{
|
|
|
|
Type: v1.NodeExternalIP,
|
2015-12-17 14:52:31 +00:00
|
|
|
Address: srv.AccessIPv6,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return addrs, nil
|
2014-10-09 15:21:06 +00:00
|
|
|
}
|
|
|
|
|
2016-09-02 06:24:54 +00:00
|
|
|
func getAddressesByName(client *gophercloud.ServiceClient, name types.NodeName) ([]v1.NodeAddress, error) {
|
|
|
|
srv, err := getServerByName(client, name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nodeAddresses(srv)
|
|
|
|
}
|
|
|
|
|
2016-07-16 06:10:29 +00:00
|
|
|
func getAddressByName(client *gophercloud.ServiceClient, name types.NodeName) (string, error) {
|
2015-12-17 14:52:31 +00:00
|
|
|
addrs, err := getAddressesByName(client, name)
|
2014-10-09 15:21:06 +00:00
|
|
|
if err != nil {
|
2014-10-30 03:16:50 +00:00
|
|
|
return "", err
|
2015-12-17 14:52:31 +00:00
|
|
|
} else if len(addrs) == 0 {
|
|
|
|
return "", ErrNoAddressFound
|
2014-10-09 15:21:06 +00:00
|
|
|
}
|
|
|
|
|
2015-12-17 14:52:31 +00:00
|
|
|
for _, addr := range addrs {
|
2016-11-18 20:58:42 +00:00
|
|
|
if addr.Type == v1.NodeInternalIP {
|
2015-12-17 14:52:31 +00:00
|
|
|
return addr.Address, nil
|
2015-03-16 06:11:36 +00:00
|
|
|
}
|
2014-10-29 08:30:55 +00:00
|
|
|
}
|
2015-12-17 14:52:31 +00:00
|
|
|
|
|
|
|
return addrs[0].Address, nil
|
2015-06-12 15:42:38 +00:00
|
|
|
}
|
|
|
|
|
2017-08-08 15:29:37 +00:00
|
|
|
// getAttachedInterfacesByID returns the node interfaces of the specified instance.
|
|
|
|
func getAttachedInterfacesByID(client *gophercloud.ServiceClient, serviceID string) ([]attachinterfaces.Interface, error) {
|
|
|
|
var interfaces []attachinterfaces.Interface
|
|
|
|
|
|
|
|
pager := attachinterfaces.List(client, serviceID)
|
|
|
|
err := pager.EachPage(func(page pagination.Page) (bool, error) {
|
|
|
|
s, err := attachinterfaces.ExtractInterfaces(page)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
interfaces = append(interfaces, s...)
|
|
|
|
return true, nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return interfaces, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return interfaces, nil
|
|
|
|
}
|
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
func (os *OpenStack) Clusters() (cloudprovider.Clusters, bool) {
|
2014-11-13 20:35:03 +00:00
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2015-05-05 14:10:24 +00:00
|
|
|
// ProviderName returns the cloud provider ID.
|
|
|
|
func (os *OpenStack) ProviderName() string {
|
|
|
|
return ProviderName
|
|
|
|
}
|
|
|
|
|
2015-10-24 00:01:49 +00:00
|
|
|
// ScrubDNS filters DNS settings for pods.
|
2017-01-11 12:14:01 +00:00
|
|
|
func (os *OpenStack) ScrubDNS(nameServers, searches []string) ([]string, []string) {
|
|
|
|
return nameServers, searches
|
2015-10-24 00:01:49 +00:00
|
|
|
}
|
|
|
|
|
2017-07-17 04:28:57 +00:00
|
|
|
// HasClusterID returns true if the cluster has a clusterID
|
|
|
|
func (os *OpenStack) HasClusterID() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2015-09-28 20:57:58 +00:00
|
|
|
func (os *OpenStack) LoadBalancer() (cloudprovider.LoadBalancer, bool) {
|
|
|
|
glog.V(4).Info("openstack.LoadBalancer() called")
|
2014-12-30 04:47:04 +00:00
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
// TODO: Search for and support Rackspace loadbalancer API, and others.
|
2017-05-25 13:41:30 +00:00
|
|
|
network, err := os.NewNetworkV2()
|
2014-10-30 03:16:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2017-05-25 13:41:30 +00:00
|
|
|
compute, err := os.NewComputeV2()
|
2014-10-30 03:16:50 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2017-01-11 12:14:01 +00:00
|
|
|
lbVersion := os.lbOpts.LBVersion
|
|
|
|
if lbVersion == "" {
|
2016-07-28 07:07:10 +00:00
|
|
|
// No version specified, try newest supported by server
|
|
|
|
netExts, err := networkExtensions(network)
|
|
|
|
if err != nil {
|
|
|
|
glog.Warningf("Failed to list neutron extensions: %v", err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
if netExts["lbaasv2"] {
|
2017-01-11 12:14:01 +00:00
|
|
|
lbVersion = "v2"
|
2016-07-28 07:07:10 +00:00
|
|
|
} else if netExts["lbaas"] {
|
2017-01-11 12:14:01 +00:00
|
|
|
lbVersion = "v1"
|
2016-07-28 07:07:10 +00:00
|
|
|
} else {
|
|
|
|
glog.Warningf("Failed to find neutron LBaaS extension (v1 or v2)")
|
|
|
|
return nil, false
|
|
|
|
}
|
2017-01-11 12:14:01 +00:00
|
|
|
glog.V(3).Infof("Using LBaaS extension %v", lbVersion)
|
2016-07-28 07:07:10 +00:00
|
|
|
}
|
|
|
|
|
2015-09-28 20:57:58 +00:00
|
|
|
glog.V(1).Info("Claiming to support LoadBalancer")
|
2014-11-26 05:41:11 +00:00
|
|
|
|
2017-01-11 12:14:01 +00:00
|
|
|
if lbVersion == "v2" {
|
2016-05-17 22:05:03 +00:00
|
|
|
return &LbaasV2{LoadBalancer{network, compute, os.lbOpts}}, true
|
2017-01-11 12:14:01 +00:00
|
|
|
} else if lbVersion == "v1" {
|
2016-05-17 22:05:03 +00:00
|
|
|
return &LbaasV1{LoadBalancer{network, compute, os.lbOpts}}, true
|
2016-07-28 07:07:10 +00:00
|
|
|
} else {
|
2017-01-11 12:14:01 +00:00
|
|
|
glog.Warningf("Config error: unrecognised lb-version \"%v\"", lbVersion)
|
2016-07-28 07:07:10 +00:00
|
|
|
return nil, false
|
2016-05-17 22:05:03 +00:00
|
|
|
}
|
2014-10-30 03:16:50 +00:00
|
|
|
}
|
|
|
|
|
2015-06-02 07:26:03 +00:00
|
|
|
func isNotFound(err error) bool {
|
2016-11-07 08:35:42 +00:00
|
|
|
e, ok := err.(*gophercloud.ErrUnexpectedResponseCode)
|
2015-06-02 07:26:03 +00:00
|
|
|
return ok && e.Actual == http.StatusNotFound
|
|
|
|
}
|
|
|
|
|
2014-10-08 01:29:28 +00:00
|
|
|
func (os *OpenStack) Zones() (cloudprovider.Zones, bool) {
|
2014-11-26 05:41:11 +00:00
|
|
|
glog.V(1).Info("Claiming to support Zones")
|
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
return os, true
|
|
|
|
}
|
2017-08-25 15:08:00 +00:00
|
|
|
|
2014-10-30 03:16:50 +00:00
|
|
|
func (os *OpenStack) GetZone() (cloudprovider.Zone, error) {
|
2017-09-18 03:32:28 +00:00
|
|
|
md, err := getMetadata(os.metadataOpts.SearchOrder)
|
2016-08-30 02:41:43 +00:00
|
|
|
if err != nil {
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
zone := cloudprovider.Zone{
|
|
|
|
FailureDomain: md.AvailabilityZone,
|
|
|
|
Region: os.region,
|
|
|
|
}
|
|
|
|
glog.V(1).Infof("Current zone is %v", zone)
|
2014-11-26 05:41:11 +00:00
|
|
|
|
2016-08-30 02:41:43 +00:00
|
|
|
return zone, nil
|
2014-10-08 01:29:28 +00:00
|
|
|
}
|
2015-05-15 21:49:26 +00:00
|
|
|
|
2017-08-17 18:46:25 +00:00
|
|
|
// GetZoneByProviderID implements Zones.GetZoneByProviderID
|
|
|
|
// This is particularly useful in external cloud providers where the kubelet
|
|
|
|
// does not initialize node data.
|
|
|
|
func (os *OpenStack) GetZoneByProviderID(providerID string) (cloudprovider.Zone, error) {
|
2017-08-25 15:08:00 +00:00
|
|
|
instanceID, err := instanceIDFromProviderID(providerID)
|
|
|
|
if err != nil {
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
compute, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
srv, err := servers.Get(compute, instanceID).Extract()
|
|
|
|
if err != nil {
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
zone := cloudprovider.Zone{
|
|
|
|
FailureDomain: srv.Metadata[AvailabilityZone],
|
|
|
|
Region: os.region,
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("The instance %s in zone %v", srv.Name, zone)
|
|
|
|
|
|
|
|
return zone, nil
|
2017-08-17 18:46:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetZoneByNodeName implements Zones.GetZoneByNodeName
|
|
|
|
// This is particularly useful in external cloud providers where the kubelet
|
|
|
|
// does not initialize node data.
|
|
|
|
func (os *OpenStack) GetZoneByNodeName(nodeName types.NodeName) (cloudprovider.Zone, error) {
|
2017-08-25 15:08:00 +00:00
|
|
|
compute, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
srv, err := getServerByName(compute, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrNotFound {
|
|
|
|
return cloudprovider.Zone{}, cloudprovider.InstanceNotFound
|
|
|
|
}
|
|
|
|
return cloudprovider.Zone{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
zone := cloudprovider.Zone{
|
|
|
|
FailureDomain: srv.Metadata[AvailabilityZone],
|
|
|
|
Region: os.region,
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("The instance %s in zone %v", srv.Name, zone)
|
|
|
|
|
|
|
|
return zone, nil
|
2017-08-17 18:46:25 +00:00
|
|
|
}
|
|
|
|
|
2015-05-15 21:49:26 +00:00
|
|
|
func (os *OpenStack) Routes() (cloudprovider.Routes, bool) {
|
2016-09-02 06:24:54 +00:00
|
|
|
glog.V(4).Info("openstack.Routes() called")
|
|
|
|
|
2017-05-25 13:41:30 +00:00
|
|
|
network, err := os.NewNetworkV2()
|
2016-09-02 06:24:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
netExts, err := networkExtensions(network)
|
|
|
|
if err != nil {
|
|
|
|
glog.Warningf("Failed to list neutron extensions: %v", err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
if !netExts["extraroute"] {
|
|
|
|
glog.V(3).Infof("Neutron extraroute extension not found, required for Routes support")
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
2017-05-25 13:41:30 +00:00
|
|
|
compute, err := os.NewComputeV2()
|
2016-09-02 06:24:54 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := NewRoutes(compute, network, os.routeOpts)
|
|
|
|
if err != nil {
|
|
|
|
glog.Warningf("Error initialising Routes support: %v", err)
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
|
|
|
|
glog.V(1).Info("Claiming to support Routes")
|
|
|
|
|
|
|
|
return r, true
|
2015-05-15 21:49:26 +00:00
|
|
|
}
|
2017-01-12 19:20:20 +00:00
|
|
|
|
|
|
|
// Implementation of sort interface for blockstorage version probing
|
|
|
|
type APIVersionsByID []apiversions_v1.APIVersion
|
|
|
|
|
|
|
|
func (apiVersions APIVersionsByID) Len() int {
|
|
|
|
return len(apiVersions)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (apiVersions APIVersionsByID) Swap(i, j int) {
|
|
|
|
apiVersions[i], apiVersions[j] = apiVersions[j], apiVersions[i]
|
|
|
|
}
|
|
|
|
|
|
|
|
func (apiVersions APIVersionsByID) Less(i, j int) bool {
|
|
|
|
return apiVersions[i].ID > apiVersions[j].ID
|
|
|
|
}
|
|
|
|
|
|
|
|
func autoVersionSelector(apiVersion *apiversions_v1.APIVersion) string {
|
|
|
|
switch strings.ToLower(apiVersion.ID) {
|
|
|
|
case "v2.0":
|
|
|
|
return "v2"
|
|
|
|
case "v1.0":
|
|
|
|
return "v1"
|
|
|
|
default:
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func doBsApiVersionAutodetect(availableApiVersions []apiversions_v1.APIVersion) string {
|
|
|
|
sort.Sort(APIVersionsByID(availableApiVersions))
|
|
|
|
for _, status := range []string{"CURRENT", "SUPPORTED"} {
|
|
|
|
for _, version := range availableApiVersions {
|
|
|
|
if strings.ToUpper(version.Status) == status {
|
|
|
|
if detectedApiVersion := autoVersionSelector(&version); detectedApiVersion != "" {
|
|
|
|
glog.V(3).Infof("Blockstorage API version probing has found a suitable %s api version: %s", status, detectedApiVersion)
|
|
|
|
return detectedApiVersion
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (os *OpenStack) volumeService(forceVersion string) (volumeService, error) {
|
|
|
|
bsVersion := ""
|
|
|
|
if forceVersion == "" {
|
|
|
|
bsVersion = os.bsOpts.BSVersion
|
|
|
|
} else {
|
|
|
|
bsVersion = forceVersion
|
|
|
|
}
|
|
|
|
|
|
|
|
switch bsVersion {
|
|
|
|
case "v1":
|
2017-05-25 13:41:30 +00:00
|
|
|
sClient, err := os.NewBlockStorageV1()
|
|
|
|
if err != nil {
|
2017-01-12 19:20:20 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &VolumesV1{sClient, os.bsOpts}, nil
|
|
|
|
case "v2":
|
2017-05-25 13:41:30 +00:00
|
|
|
sClient, err := os.NewBlockStorageV2()
|
|
|
|
if err != nil {
|
2017-01-12 19:20:20 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &VolumesV2{sClient, os.bsOpts}, nil
|
|
|
|
case "auto":
|
2017-05-25 13:41:30 +00:00
|
|
|
sClient, err := os.NewBlockStorageV1()
|
|
|
|
if err != nil {
|
2017-01-12 19:20:20 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
availableApiVersions := []apiversions_v1.APIVersion{}
|
|
|
|
err = apiversions_v1.List(sClient).EachPage(func(page pagination.Page) (bool, error) {
|
|
|
|
// returning false from this handler stops page iteration, error is propagated to the upper function
|
|
|
|
apiversions, err := apiversions_v1.ExtractAPIVersions(page)
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Unable to extract api versions from page: %v", err)
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
availableApiVersions = append(availableApiVersions, apiversions...)
|
|
|
|
return true, nil
|
|
|
|
})
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
glog.Errorf("Error when retrieving list of supported blockstorage api versions: %v", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if autodetectedVersion := doBsApiVersionAutodetect(availableApiVersions); autodetectedVersion != "" {
|
|
|
|
return os.volumeService(autodetectedVersion)
|
|
|
|
} else {
|
2017-07-20 17:49:57 +00:00
|
|
|
// Nothing suitable found, failed autodetection, just exit with appropriate message
|
|
|
|
err_txt := "BlockStorage API version autodetection failed. " +
|
|
|
|
"Please set it explicitly in cloud.conf in section [BlockStorage] with key `bs-version`"
|
|
|
|
return nil, errors.New(err_txt)
|
2017-01-12 19:20:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
err_txt := fmt.Sprintf("Config error: unrecognised bs-version \"%v\"", os.bsOpts.BSVersion)
|
|
|
|
glog.Warningf(err_txt)
|
|
|
|
return nil, errors.New(err_txt)
|
|
|
|
}
|
|
|
|
}
|
2017-09-18 03:32:28 +00:00
|
|
|
|
|
|
|
func validateMetadataSearchOrder(order string) error {
|
|
|
|
if order == "" {
|
|
|
|
return errors.New("Invalid value in section [Metadata] with key `search-order`. Value cannot be empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
elements := strings.Split(order, ",")
|
|
|
|
if len(elements) > 2 {
|
|
|
|
return errors.New("Invalid value in section [Metadata] with key `search-order`. Value cannot contain more than 2 elements")
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, o := range elements {
|
|
|
|
switch o {
|
|
|
|
case configDriveID:
|
|
|
|
case metadataID:
|
|
|
|
default:
|
|
|
|
errTxt := "Invalid element '%s' found in section [Metadata] with key `search-order`." +
|
|
|
|
"Supported elements include '%s' and '%s'"
|
|
|
|
return fmt.Errorf(errTxt, o, configDriveID, metadataID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|