2019-09-27 21:51:53 +00:00
|
|
|
// +build !providerless
|
|
|
|
|
|
|
|
/*
|
|
|
|
Copyright 2016 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 openstack
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
v1 "k8s.io/api/core/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/api/resource"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
cloudprovider "k8s.io/cloud-provider"
|
|
|
|
cloudvolume "k8s.io/cloud-provider/volume"
|
|
|
|
volerr "k8s.io/cloud-provider/volume/errors"
|
|
|
|
volumehelpers "k8s.io/cloud-provider/volume/helpers"
|
2019-12-12 01:27:03 +00:00
|
|
|
"k8s.io/component-base/metrics"
|
2019-09-27 21:51:53 +00:00
|
|
|
|
|
|
|
"github.com/gophercloud/gophercloud"
|
|
|
|
volumeexpand "github.com/gophercloud/gophercloud/openstack/blockstorage/extensions/volumeactions"
|
|
|
|
volumes_v1 "github.com/gophercloud/gophercloud/openstack/blockstorage/v1/volumes"
|
|
|
|
volumes_v2 "github.com/gophercloud/gophercloud/openstack/blockstorage/v2/volumes"
|
|
|
|
volumes_v3 "github.com/gophercloud/gophercloud/openstack/blockstorage/v3/volumes"
|
|
|
|
"github.com/gophercloud/gophercloud/openstack/compute/v2/extensions/volumeattach"
|
2020-08-10 17:43:49 +00:00
|
|
|
"k8s.io/klog/v2"
|
2019-09-27 21:51:53 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type volumeService interface {
|
|
|
|
createVolume(opts volumeCreateOpts) (string, string, error)
|
|
|
|
getVolume(volumeID string) (Volume, error)
|
|
|
|
deleteVolume(volumeName string) error
|
|
|
|
expandVolume(volumeID string, newSize int) error
|
|
|
|
}
|
|
|
|
|
|
|
|
// VolumesV1 is a Volumes implementation for cinder v1
|
|
|
|
type VolumesV1 struct {
|
|
|
|
blockstorage *gophercloud.ServiceClient
|
|
|
|
opts BlockStorageOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
// VolumesV2 is a Volumes implementation for cinder v2
|
|
|
|
type VolumesV2 struct {
|
|
|
|
blockstorage *gophercloud.ServiceClient
|
|
|
|
opts BlockStorageOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
// VolumesV3 is a Volumes implementation for cinder v3
|
|
|
|
type VolumesV3 struct {
|
|
|
|
blockstorage *gophercloud.ServiceClient
|
|
|
|
opts BlockStorageOpts
|
|
|
|
}
|
|
|
|
|
|
|
|
// Volume stores information about a single volume
|
|
|
|
type Volume struct {
|
|
|
|
// ID of the instance, to which this volume is attached. "" if not attached
|
|
|
|
AttachedServerID string
|
|
|
|
// Device file path
|
|
|
|
AttachedDevice string
|
|
|
|
// availabilityZone is which availability zone the volume is in
|
|
|
|
AvailabilityZone string
|
|
|
|
// Unique identifier for the volume.
|
|
|
|
ID string
|
|
|
|
// Human-readable display name for the volume.
|
|
|
|
Name string
|
|
|
|
// Current status of the volume.
|
|
|
|
Status string
|
|
|
|
// Volume size in GB
|
|
|
|
Size int
|
|
|
|
}
|
|
|
|
|
|
|
|
type volumeCreateOpts struct {
|
|
|
|
Size int
|
|
|
|
Availability string
|
|
|
|
Name string
|
|
|
|
VolumeType string
|
|
|
|
Metadata map[string]string
|
|
|
|
}
|
|
|
|
|
|
|
|
// implements PVLabeler.
|
|
|
|
var _ cloudprovider.PVLabeler = (*OpenStack)(nil)
|
|
|
|
|
|
|
|
const (
|
|
|
|
volumeAvailableStatus = "available"
|
|
|
|
volumeInUseStatus = "in-use"
|
|
|
|
volumeDeletedStatus = "deleted"
|
|
|
|
volumeErrorStatus = "error"
|
|
|
|
|
|
|
|
// On some environments, we need to query the metadata service in order
|
|
|
|
// to locate disks. We'll use the Newton version, which includes device
|
|
|
|
// metadata.
|
|
|
|
newtonMetadataVersion = "2016-06-30"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (volumes *VolumesV1) createVolume(opts volumeCreateOpts) (string, string, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
|
|
|
|
createOpts := volumes_v1.CreateOpts{
|
|
|
|
Name: opts.Name,
|
|
|
|
Size: opts.Size,
|
|
|
|
VolumeType: opts.VolumeType,
|
|
|
|
AvailabilityZone: opts.Availability,
|
|
|
|
Metadata: opts.Metadata,
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := volumes_v1.Create(volumes.blockstorage, createOpts).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("create_v1_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
return vol.ID, vol.AvailabilityZone, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV2) createVolume(opts volumeCreateOpts) (string, string, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
|
|
|
|
createOpts := volumes_v2.CreateOpts{
|
|
|
|
Name: opts.Name,
|
|
|
|
Size: opts.Size,
|
|
|
|
VolumeType: opts.VolumeType,
|
|
|
|
AvailabilityZone: opts.Availability,
|
|
|
|
Metadata: opts.Metadata,
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := volumes_v2.Create(volumes.blockstorage, createOpts).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("create_v2_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
return vol.ID, vol.AvailabilityZone, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV3) createVolume(opts volumeCreateOpts) (string, string, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
|
|
|
|
createOpts := volumes_v3.CreateOpts{
|
|
|
|
Name: opts.Name,
|
|
|
|
Size: opts.Size,
|
|
|
|
VolumeType: opts.VolumeType,
|
|
|
|
AvailabilityZone: opts.Availability,
|
|
|
|
Metadata: opts.Metadata,
|
|
|
|
}
|
|
|
|
|
|
|
|
vol, err := volumes_v3.Create(volumes.blockstorage, createOpts).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("create_v3_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
return vol.ID, vol.AvailabilityZone, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV1) getVolume(volumeID string) (Volume, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
volumeV1, err := volumes_v1.Get(volumes.blockstorage, volumeID).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("get_v1_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
if isNotFound(err) {
|
|
|
|
return Volume{}, ErrNotFound
|
|
|
|
}
|
|
|
|
return Volume{}, fmt.Errorf("error occurred getting volume by ID: %s, err: %v", volumeID, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
volume := Volume{
|
|
|
|
AvailabilityZone: volumeV1.AvailabilityZone,
|
|
|
|
ID: volumeV1.ID,
|
|
|
|
Name: volumeV1.Name,
|
|
|
|
Status: volumeV1.Status,
|
|
|
|
Size: volumeV1.Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(volumeV1.Attachments) > 0 && volumeV1.Attachments[0]["server_id"] != nil {
|
|
|
|
volume.AttachedServerID = volumeV1.Attachments[0]["server_id"].(string)
|
|
|
|
volume.AttachedDevice = volumeV1.Attachments[0]["device"].(string)
|
|
|
|
}
|
|
|
|
|
|
|
|
return volume, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV2) getVolume(volumeID string) (Volume, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
volumeV2, err := volumes_v2.Get(volumes.blockstorage, volumeID).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("get_v2_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
if isNotFound(err) {
|
|
|
|
return Volume{}, ErrNotFound
|
|
|
|
}
|
|
|
|
return Volume{}, fmt.Errorf("error occurred getting volume by ID: %s, err: %v", volumeID, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
volume := Volume{
|
|
|
|
AvailabilityZone: volumeV2.AvailabilityZone,
|
|
|
|
ID: volumeV2.ID,
|
|
|
|
Name: volumeV2.Name,
|
|
|
|
Status: volumeV2.Status,
|
|
|
|
Size: volumeV2.Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(volumeV2.Attachments) > 0 {
|
|
|
|
volume.AttachedServerID = volumeV2.Attachments[0].ServerID
|
|
|
|
volume.AttachedDevice = volumeV2.Attachments[0].Device
|
|
|
|
}
|
|
|
|
|
|
|
|
return volume, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV3) getVolume(volumeID string) (Volume, error) {
|
|
|
|
startTime := time.Now()
|
|
|
|
volumeV3, err := volumes_v3.Get(volumes.blockstorage, volumeID).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("get_v3_volume", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
if isNotFound(err) {
|
|
|
|
return Volume{}, ErrNotFound
|
|
|
|
}
|
|
|
|
return Volume{}, fmt.Errorf("error occurred getting volume by ID: %s, err: %v", volumeID, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
volume := Volume{
|
|
|
|
AvailabilityZone: volumeV3.AvailabilityZone,
|
|
|
|
ID: volumeV3.ID,
|
|
|
|
Name: volumeV3.Name,
|
|
|
|
Status: volumeV3.Status,
|
|
|
|
Size: volumeV3.Size,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(volumeV3.Attachments) > 0 {
|
|
|
|
volume.AttachedServerID = volumeV3.Attachments[0].ServerID
|
|
|
|
volume.AttachedDevice = volumeV3.Attachments[0].Device
|
|
|
|
}
|
|
|
|
|
|
|
|
return volume, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV1) deleteVolume(volumeID string) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
err := volumes_v1.Delete(volumes.blockstorage, volumeID).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("delete_v1_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV2) deleteVolume(volumeID string) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
err := volumes_v2.Delete(volumes.blockstorage, volumeID, nil).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("delete_v2_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV3) deleteVolume(volumeID string) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
err := volumes_v3.Delete(volumes.blockstorage, volumeID, nil).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("delete_v3_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV1) expandVolume(volumeID string, newSize int) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
createOpts := volumeexpand.ExtendSizeOpts{
|
|
|
|
NewSize: newSize,
|
|
|
|
}
|
|
|
|
err := volumeexpand.ExtendSize(volumes.blockstorage, volumeID, createOpts).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("expand_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV2) expandVolume(volumeID string, newSize int) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
createOpts := volumeexpand.ExtendSizeOpts{
|
|
|
|
NewSize: newSize,
|
|
|
|
}
|
|
|
|
err := volumeexpand.ExtendSize(volumes.blockstorage, volumeID, createOpts).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("expand_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (volumes *VolumesV3) expandVolume(volumeID string, newSize int) error {
|
|
|
|
startTime := time.Now()
|
|
|
|
createOpts := volumeexpand.ExtendSizeOpts{
|
|
|
|
NewSize: newSize,
|
|
|
|
}
|
|
|
|
err := volumeexpand.ExtendSize(volumes.blockstorage, volumeID, createOpts).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("expand_volume", timeTaken, err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// OperationPending checks if there is an operation pending on a volume
|
|
|
|
func (os *OpenStack) OperationPending(diskName string) (bool, string, error) {
|
|
|
|
volume, err := os.getVolume(diskName)
|
|
|
|
if err != nil {
|
|
|
|
return false, "", err
|
|
|
|
}
|
|
|
|
volumeStatus := volume.Status
|
|
|
|
if volumeStatus == volumeErrorStatus {
|
|
|
|
err = fmt.Errorf("status of volume %s is %s", diskName, volumeStatus)
|
|
|
|
return false, volumeStatus, err
|
|
|
|
}
|
|
|
|
if volumeStatus == volumeAvailableStatus || volumeStatus == volumeInUseStatus || volumeStatus == volumeDeletedStatus {
|
|
|
|
return false, volume.Status, nil
|
|
|
|
}
|
|
|
|
return true, volumeStatus, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// AttachDisk attaches given cinder volume to the compute running kubelet
|
|
|
|
func (os *OpenStack) AttachDisk(instanceID, volumeID string) (string, error) {
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
cClient, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if volume.AttachedServerID != "" {
|
|
|
|
if instanceID == volume.AttachedServerID {
|
|
|
|
klog.V(4).Infof("Disk %s is already attached to instance %s", volumeID, instanceID)
|
|
|
|
return volume.ID, nil
|
|
|
|
}
|
|
|
|
nodeName, err := os.GetNodeNameByID(volume.AttachedServerID)
|
|
|
|
attachErr := fmt.Sprintf("disk %s path %s is attached to a different instance (%s)", volumeID, volume.AttachedDevice, volume.AttachedServerID)
|
|
|
|
if err != nil {
|
|
|
|
klog.Error(attachErr)
|
|
|
|
return "", errors.New(attachErr)
|
|
|
|
}
|
|
|
|
// using volume.AttachedDevice may cause problems because cinder does not report device path correctly see issue #33128
|
|
|
|
devicePath := volume.AttachedDevice
|
|
|
|
danglingErr := volerr.NewDanglingError(attachErr, nodeName, devicePath)
|
|
|
|
klog.V(2).Infof("Found dangling volume %s attached to node %s", volumeID, nodeName)
|
|
|
|
return "", danglingErr
|
|
|
|
}
|
|
|
|
|
|
|
|
startTime := time.Now()
|
|
|
|
// add read only flag here if possible spothanis
|
|
|
|
_, err = volumeattach.Create(cClient, instanceID, &volumeattach.CreateOpts{
|
|
|
|
VolumeID: volume.ID,
|
|
|
|
}).Extract()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("attach_disk", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
return "", fmt.Errorf("failed to attach %s volume to %s compute: %v", volumeID, instanceID, err)
|
|
|
|
}
|
|
|
|
klog.V(2).Infof("Successfully attached %s volume to %s compute", volumeID, instanceID)
|
|
|
|
return volume.ID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DetachDisk detaches given cinder volume from the compute running kubelet
|
|
|
|
func (os *OpenStack) DetachDisk(instanceID, volumeID string) error {
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if volume.Status == volumeAvailableStatus {
|
|
|
|
// "available" is fine since that means the volume is detached from instance already.
|
|
|
|
klog.V(2).Infof("volume: %s has been detached from compute: %s ", volume.ID, instanceID)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if volume.Status != volumeInUseStatus {
|
|
|
|
return fmt.Errorf("can not detach volume %s, its status is %s", volume.Name, volume.Status)
|
|
|
|
}
|
|
|
|
cClient, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if volume.AttachedServerID != instanceID {
|
|
|
|
return fmt.Errorf("disk: %s has no attachments or is not attached to compute: %s", volume.Name, instanceID)
|
|
|
|
}
|
|
|
|
|
|
|
|
startTime := time.Now()
|
|
|
|
// This is a blocking call and effects kubelet's performance directly.
|
|
|
|
// We should consider kicking it out into a separate routine, if it is bad.
|
|
|
|
err = volumeattach.Delete(cClient, instanceID, volume.ID).ExtractErr()
|
|
|
|
timeTaken := time.Since(startTime).Seconds()
|
|
|
|
recordOpenstackOperationMetric("detach_disk", timeTaken, err)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("failed to delete volume %s from compute %s attached %v", volume.ID, instanceID, err)
|
|
|
|
}
|
|
|
|
klog.V(2).Infof("Successfully detached volume: %s from compute: %s", volume.ID, instanceID)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExpandVolume expands the size of specific cinder volume (in GiB)
|
|
|
|
func (os *OpenStack) ExpandVolume(volumeID string, oldSize resource.Quantity, newSize resource.Quantity) (resource.Quantity, error) {
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return oldSize, err
|
|
|
|
}
|
|
|
|
if volume.Status != volumeAvailableStatus {
|
|
|
|
// cinder volume can not be expanded if its status is not available
|
2020-08-10 17:43:49 +00:00
|
|
|
if volume.Status == volumeInUseStatus {
|
|
|
|
// Send a nice event when the volume is used
|
|
|
|
return oldSize, fmt.Errorf("PVC used by a Pod can not be expanded, please ensure the PVC is not used by any Pod and is fully detached from a node")
|
|
|
|
}
|
|
|
|
// Send not so nice event when the volume is in any other state (deleted, error)
|
|
|
|
return oldSize, fmt.Errorf("volume in state %q can not be expanded, it must be \"available\"", volume.Status)
|
2019-09-27 21:51:53 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Cinder works with gigabytes, convert to GiB with rounding up
|
|
|
|
volSizeGiB, err := volumehelpers.RoundUpToGiBInt(newSize)
|
|
|
|
if err != nil {
|
|
|
|
return oldSize, err
|
|
|
|
}
|
|
|
|
newSizeQuant := resource.MustParse(fmt.Sprintf("%dGi", volSizeGiB))
|
|
|
|
|
|
|
|
// if volume size equals to or greater than the newSize, return nil
|
|
|
|
if volume.Size >= volSizeGiB {
|
|
|
|
return newSizeQuant, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
volumes, err := os.volumeService("")
|
|
|
|
if err != nil {
|
|
|
|
return oldSize, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = volumes.expandVolume(volumeID, volSizeGiB)
|
|
|
|
if err != nil {
|
|
|
|
return oldSize, err
|
|
|
|
}
|
|
|
|
return newSizeQuant, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// getVolume retrieves Volume by its ID.
|
|
|
|
func (os *OpenStack) getVolume(volumeID string) (Volume, error) {
|
|
|
|
volumes, err := os.volumeService("")
|
|
|
|
if err != nil {
|
|
|
|
return Volume{}, fmt.Errorf("unable to initialize cinder client for region: %s, err: %v", os.region, err)
|
|
|
|
}
|
|
|
|
return volumes.getVolume(volumeID)
|
|
|
|
}
|
|
|
|
|
|
|
|
// CreateVolume creates a volume of given size (in GiB)
|
|
|
|
func (os *OpenStack) CreateVolume(name string, size int, vtype, availability string, tags *map[string]string) (string, string, string, bool, error) {
|
|
|
|
volumes, err := os.volumeService("")
|
|
|
|
if err != nil {
|
|
|
|
return "", "", "", os.bsOpts.IgnoreVolumeAZ, fmt.Errorf("unable to initialize cinder client for region: %s, err: %v", os.region, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
opts := volumeCreateOpts{
|
|
|
|
Name: name,
|
|
|
|
Size: size,
|
|
|
|
VolumeType: vtype,
|
|
|
|
Availability: availability,
|
|
|
|
}
|
|
|
|
if tags != nil {
|
|
|
|
opts.Metadata = *tags
|
|
|
|
}
|
|
|
|
|
|
|
|
volumeID, volumeAZ, err := volumes.createVolume(opts)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return "", "", "", os.bsOpts.IgnoreVolumeAZ, fmt.Errorf("failed to create a %d GB volume: %v", size, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
klog.Infof("Created volume %v in Availability Zone: %v Region: %v Ignore volume AZ: %v", volumeID, volumeAZ, os.region, os.bsOpts.IgnoreVolumeAZ)
|
|
|
|
return volumeID, volumeAZ, os.region, os.bsOpts.IgnoreVolumeAZ, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDevicePathBySerialID returns the path of an attached block storage volume, specified by its id.
|
|
|
|
func (os *OpenStack) GetDevicePathBySerialID(volumeID string) string {
|
|
|
|
// Build a list of candidate device paths.
|
|
|
|
// Certain Nova drivers will set the disk serial ID, including the Cinder volume id.
|
|
|
|
candidateDeviceNodes := []string{
|
|
|
|
// KVM
|
|
|
|
fmt.Sprintf("virtio-%s", volumeID[:20]),
|
|
|
|
// KVM virtio-scsi
|
|
|
|
fmt.Sprintf("scsi-0QEMU_QEMU_HARDDISK_%s", volumeID[:20]),
|
|
|
|
// ESXi
|
|
|
|
fmt.Sprintf("wwn-0x%s", strings.Replace(volumeID, "-", "", -1)),
|
|
|
|
}
|
|
|
|
|
|
|
|
files, _ := ioutil.ReadDir("/dev/disk/by-id/")
|
|
|
|
|
|
|
|
for _, f := range files {
|
|
|
|
for _, c := range candidateDeviceNodes {
|
|
|
|
if c == f.Name() {
|
|
|
|
klog.V(4).Infof("Found disk attached as %q; full devicepath: %s\n", f.Name(), path.Join("/dev/disk/by-id/", f.Name()))
|
|
|
|
return path.Join("/dev/disk/by-id/", f.Name())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
klog.V(4).Infof("Failed to find device for the volumeID: %q by serial ID", volumeID)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (os *OpenStack) getDevicePathFromInstanceMetadata(volumeID string) string {
|
|
|
|
// Nova Hyper-V hosts cannot override disk SCSI IDs. In order to locate
|
|
|
|
// volumes, we're querying the metadata service. Note that the Hyper-V
|
|
|
|
// driver will include device metadata for untagged volumes as well.
|
|
|
|
//
|
|
|
|
// We're avoiding using cached metadata (or the configdrive),
|
|
|
|
// relying on the metadata service.
|
|
|
|
instanceMetadata, err := getMetadataFromMetadataService(
|
|
|
|
newtonMetadataVersion)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
klog.V(4).Infof(
|
|
|
|
"Could not retrieve instance metadata. Error: %v", err)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, device := range instanceMetadata.Devices {
|
|
|
|
if device.Type == "disk" && device.Serial == volumeID {
|
|
|
|
klog.V(4).Infof(
|
|
|
|
"Found disk metadata for volumeID %q. Bus: %q, Address: %q",
|
|
|
|
volumeID, device.Bus, device.Address)
|
|
|
|
|
|
|
|
diskPattern := fmt.Sprintf(
|
|
|
|
"/dev/disk/by-path/*-%s-%s",
|
|
|
|
device.Bus, device.Address)
|
|
|
|
diskPaths, err := filepath.Glob(diskPattern)
|
|
|
|
if err != nil {
|
|
|
|
klog.Errorf(
|
|
|
|
"could not retrieve disk path for volumeID: %q. Error filepath.Glob(%q): %v",
|
|
|
|
volumeID, diskPattern, err)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(diskPaths) == 1 {
|
|
|
|
return diskPaths[0]
|
|
|
|
}
|
|
|
|
|
|
|
|
klog.Errorf(
|
|
|
|
"expecting to find one disk path for volumeID %q, found %d: %v",
|
|
|
|
volumeID, len(diskPaths), diskPaths)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
klog.V(4).Infof(
|
|
|
|
"Could not retrieve device metadata for volumeID: %q", volumeID)
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetDevicePath returns the path of an attached block storage volume, specified by its id.
|
|
|
|
func (os *OpenStack) GetDevicePath(volumeID string) string {
|
|
|
|
devicePath := os.GetDevicePathBySerialID(volumeID)
|
|
|
|
|
|
|
|
if devicePath == "" {
|
|
|
|
devicePath = os.getDevicePathFromInstanceMetadata(volumeID)
|
|
|
|
}
|
|
|
|
|
|
|
|
if devicePath == "" {
|
|
|
|
klog.Warningf("Failed to find device for the volumeID: %q", volumeID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return devicePath
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteVolume deletes a volume given volume name.
|
|
|
|
func (os *OpenStack) DeleteVolume(volumeID string) error {
|
|
|
|
used, err := os.diskIsUsed(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if used {
|
|
|
|
msg := fmt.Sprintf("Cannot delete the volume %q, it's still attached to a node", volumeID)
|
|
|
|
return volerr.NewDeletedVolumeInUseError(msg)
|
|
|
|
}
|
|
|
|
|
|
|
|
volumes, err := os.volumeService("")
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to initialize cinder client for region: %s, err: %v", os.region, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
err = volumes.deleteVolume(volumeID)
|
|
|
|
return err
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetAttachmentDiskPath gets device path of attached volume to the compute running kubelet, as known by cinder
|
|
|
|
func (os *OpenStack) GetAttachmentDiskPath(instanceID, volumeID string) (string, error) {
|
|
|
|
// See issue #33128 - Cinder does not always tell you the right device path, as such
|
|
|
|
// we must only use this value as a last resort.
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
if volume.Status != volumeInUseStatus {
|
|
|
|
return "", fmt.Errorf("can not get device path of volume %s, its status is %s ", volume.Name, volume.Status)
|
|
|
|
}
|
|
|
|
if volume.AttachedServerID != "" {
|
|
|
|
if instanceID == volume.AttachedServerID {
|
|
|
|
// Attachment[0]["device"] points to the device path
|
|
|
|
// see http://developer.openstack.org/api-ref-blockstorage-v1.html
|
|
|
|
return volume.AttachedDevice, nil
|
|
|
|
}
|
|
|
|
return "", fmt.Errorf("disk %q is attached to a different compute: %q, should be detached before proceeding", volumeID, volume.AttachedServerID)
|
|
|
|
}
|
|
|
|
return "", fmt.Errorf("volume %s has no ServerId", volumeID)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DiskIsAttached queries if a volume is attached to a compute instance
|
|
|
|
func (os *OpenStack) DiskIsAttached(instanceID, volumeID string) (bool, error) {
|
|
|
|
if instanceID == "" {
|
|
|
|
klog.Warningf("calling DiskIsAttached with empty instanceid: %s %s", instanceID, volumeID)
|
|
|
|
}
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrNotFound {
|
|
|
|
// Volume does not exists, it can't be attached.
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return instanceID == volume.AttachedServerID, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DiskIsAttachedByName queries if a volume is attached to a compute instance by name
|
|
|
|
func (os *OpenStack) DiskIsAttachedByName(nodeName types.NodeName, volumeID string) (bool, string, error) {
|
|
|
|
cClient, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return false, "", err
|
|
|
|
}
|
|
|
|
srv, err := getServerByName(cClient, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrNotFound {
|
|
|
|
// instance not found anymore in cloudprovider, assume that cinder is detached
|
|
|
|
return false, "", nil
|
|
|
|
}
|
|
|
|
return false, "", err
|
|
|
|
}
|
|
|
|
instanceID := "/" + srv.ID
|
|
|
|
if ind := strings.LastIndex(instanceID, "/"); ind >= 0 {
|
|
|
|
instanceID = instanceID[(ind + 1):]
|
|
|
|
}
|
|
|
|
attached, err := os.DiskIsAttached(instanceID, volumeID)
|
|
|
|
return attached, instanceID, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DisksAreAttached queries if a list of volumes are attached to a compute instance
|
|
|
|
func (os *OpenStack) DisksAreAttached(instanceID string, volumeIDs []string) (map[string]bool, error) {
|
|
|
|
attached := make(map[string]bool)
|
|
|
|
for _, volumeID := range volumeIDs {
|
|
|
|
isAttached, err := os.DiskIsAttached(instanceID, volumeID)
|
|
|
|
if err != nil && err != ErrNotFound {
|
|
|
|
attached[volumeID] = true
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
attached[volumeID] = isAttached
|
|
|
|
}
|
|
|
|
return attached, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DisksAreAttachedByName queries if a list of volumes are attached to a compute instance by name
|
|
|
|
func (os *OpenStack) DisksAreAttachedByName(nodeName types.NodeName, volumeIDs []string) (map[string]bool, error) {
|
|
|
|
attached := make(map[string]bool)
|
|
|
|
cClient, err := os.NewComputeV2()
|
|
|
|
if err != nil {
|
|
|
|
return attached, err
|
|
|
|
}
|
|
|
|
srv, err := getServerByName(cClient, nodeName)
|
|
|
|
if err != nil {
|
|
|
|
if err == ErrNotFound {
|
|
|
|
// instance not found anymore, mark all volumes as detached
|
|
|
|
for _, volumeID := range volumeIDs {
|
|
|
|
attached[volumeID] = false
|
|
|
|
}
|
|
|
|
return attached, nil
|
|
|
|
}
|
|
|
|
return attached, err
|
|
|
|
}
|
|
|
|
instanceID := "/" + srv.ID
|
|
|
|
if ind := strings.LastIndex(instanceID, "/"); ind >= 0 {
|
|
|
|
instanceID = instanceID[(ind + 1):]
|
|
|
|
}
|
|
|
|
return os.DisksAreAttached(instanceID, volumeIDs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// diskIsUsed returns true a disk is attached to any node.
|
|
|
|
func (os *OpenStack) diskIsUsed(volumeID string) (bool, error) {
|
|
|
|
volume, err := os.getVolume(volumeID)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return volume.AttachedServerID != "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ShouldTrustDevicePath queries if we should trust the cinder provide deviceName, See issue #33128
|
|
|
|
func (os *OpenStack) ShouldTrustDevicePath() bool {
|
|
|
|
return os.bsOpts.TrustDevicePath
|
|
|
|
}
|
|
|
|
|
|
|
|
// NodeVolumeAttachLimit specifies number of cinder volumes that can be attached to this node.
|
|
|
|
func (os *OpenStack) NodeVolumeAttachLimit() int {
|
|
|
|
return os.bsOpts.NodeVolumeAttachLimit
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetLabelsForVolume implements PVLabeler.GetLabelsForVolume
|
|
|
|
func (os *OpenStack) GetLabelsForVolume(ctx context.Context, pv *v1.PersistentVolume) (map[string]string, error) {
|
|
|
|
// Ignore if not Cinder.
|
|
|
|
if pv.Spec.Cinder == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore any volumes that are being provisioned
|
|
|
|
if pv.Spec.Cinder.VolumeID == cloudvolume.ProvisionedVolumeName {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
// if volume az is to be ignored we should return nil from here
|
|
|
|
if os.bsOpts.IgnoreVolumeAZ {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2019-09-27 21:51:53 +00:00
|
|
|
// Get Volume
|
|
|
|
volume, err := os.getVolume(pv.Spec.Cinder.VolumeID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct Volume Labels
|
|
|
|
labels := make(map[string]string)
|
2020-12-01 01:06:26 +00:00
|
|
|
if volume.AvailabilityZone != "" {
|
|
|
|
labels[v1.LabelFailureDomainBetaZone] = volume.AvailabilityZone
|
|
|
|
}
|
|
|
|
if os.region != "" {
|
|
|
|
labels[v1.LabelFailureDomainBetaRegion] = os.region
|
|
|
|
}
|
2019-09-27 21:51:53 +00:00
|
|
|
klog.V(4).Infof("The Volume %s has labels %v", pv.Spec.Cinder.VolumeID, labels)
|
|
|
|
|
|
|
|
return labels, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// recordOpenstackOperationMetric records openstack operation metrics
|
|
|
|
func recordOpenstackOperationMetric(operation string, timeTaken float64, err error) {
|
|
|
|
if err != nil {
|
2019-12-12 01:27:03 +00:00
|
|
|
openstackAPIRequestErrors.With(metrics.Labels{"request": operation}).Inc()
|
2019-09-27 21:51:53 +00:00
|
|
|
} else {
|
2019-12-12 01:27:03 +00:00
|
|
|
openstackOperationsLatency.With(metrics.Labels{"request": operation}).Observe(timeTaken)
|
2019-09-27 21:51:53 +00:00
|
|
|
}
|
|
|
|
}
|