2019-02-19 08:14:10 +00:00
|
|
|
/*
|
2019-04-03 08:46:15 +00:00
|
|
|
Copyright 2019 The Ceph-CSI Authors.
|
2019-02-19 08:14:10 +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 util
|
|
|
|
|
|
|
|
import (
|
2019-08-22 17:19:06 +00:00
|
|
|
"context"
|
2020-06-25 11:30:04 +00:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2019-09-25 08:35:33 +00:00
|
|
|
"math"
|
2019-02-19 08:14:10 +00:00
|
|
|
"os"
|
2020-06-22 07:09:45 +00:00
|
|
|
"strconv"
|
2019-03-13 05:09:58 +00:00
|
|
|
"strings"
|
2019-08-14 05:57:45 +00:00
|
|
|
"time"
|
2019-02-19 08:14:10 +00:00
|
|
|
|
2020-06-17 18:30:09 +00:00
|
|
|
"golang.org/x/sys/unix"
|
2019-07-03 10:02:36 +00:00
|
|
|
"google.golang.org/grpc/codes"
|
|
|
|
"google.golang.org/grpc/status"
|
2019-03-13 05:09:58 +00:00
|
|
|
"k8s.io/apimachinery/pkg/util/validation"
|
2020-01-17 15:44:06 +00:00
|
|
|
"k8s.io/cloud-provider/volume/helpers"
|
2020-07-09 05:14:10 +00:00
|
|
|
klog "k8s.io/klog/v2"
|
2020-01-14 10:38:55 +00:00
|
|
|
"k8s.io/utils/mount"
|
2019-02-19 08:14:10 +00:00
|
|
|
)
|
|
|
|
|
2020-07-09 14:48:24 +00:00
|
|
|
// enum defining logging levels.
|
|
|
|
const (
|
|
|
|
Default klog.Level = iota + 1
|
|
|
|
Useful
|
|
|
|
Extended
|
|
|
|
Debug
|
|
|
|
Trace
|
|
|
|
)
|
|
|
|
|
2019-09-25 08:35:33 +00:00
|
|
|
// RoundOffVolSize rounds up given quantity upto chunks of MiB/GiB
|
|
|
|
func RoundOffVolSize(size int64) int64 {
|
2019-10-11 08:26:10 +00:00
|
|
|
size = RoundOffBytes(size)
|
2019-09-25 08:35:33 +00:00
|
|
|
// convert size back to MiB for rbd CLI
|
2020-01-17 15:44:06 +00:00
|
|
|
return size / helpers.MiB
|
2019-09-25 08:35:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// RoundOffBytes converts roundoff the size
|
|
|
|
// 1.1Mib will be round off to 2Mib same for GiB
|
2019-10-11 08:26:10 +00:00
|
|
|
// size less than 1MiB will be round off to 1MiB
|
2019-09-25 08:35:33 +00:00
|
|
|
func RoundOffBytes(bytes int64) int64 {
|
|
|
|
var num int64
|
|
|
|
floatBytes := float64(bytes)
|
|
|
|
// round off the value if its in decimal
|
2020-01-17 15:44:06 +00:00
|
|
|
if floatBytes < helpers.GiB {
|
|
|
|
num = int64(math.Ceil(floatBytes / helpers.MiB))
|
|
|
|
num *= helpers.MiB
|
2019-09-25 08:35:33 +00:00
|
|
|
} else {
|
2020-01-17 15:44:06 +00:00
|
|
|
num = int64(math.Ceil(floatBytes / helpers.GiB))
|
|
|
|
num *= helpers.GiB
|
2019-09-25 08:35:33 +00:00
|
|
|
}
|
|
|
|
return num
|
2019-03-01 12:08:17 +00:00
|
|
|
}
|
|
|
|
|
2019-07-12 10:18:00 +00:00
|
|
|
// variables which will be set during the build time
|
|
|
|
var (
|
|
|
|
// GitCommit tell the latest git commit image is built from
|
|
|
|
GitCommit string
|
|
|
|
// DriverVersion which will be driver version
|
|
|
|
DriverVersion string
|
|
|
|
)
|
|
|
|
|
2019-08-14 05:57:45 +00:00
|
|
|
// Config holds the parameters list which can be configured
|
|
|
|
type Config struct {
|
2020-07-10 10:44:59 +00:00
|
|
|
Vtype string // driver type [rbd|cephfs|liveness]
|
|
|
|
Endpoint string // CSI endpoint
|
|
|
|
DriverName string // name of the driver
|
|
|
|
NodeID string // node id
|
|
|
|
InstanceID string // unique ID distinguishing this instance of Ceph CSI
|
|
|
|
PluginPath string // location of cephcsi plugin
|
|
|
|
DomainLabels string // list of domain labels to read from the node
|
2019-08-14 05:57:45 +00:00
|
|
|
|
2019-08-21 09:28:02 +00:00
|
|
|
// metrics related flags
|
|
|
|
MetricsPath string // path of prometheus endpoint where metrics will be available
|
|
|
|
HistogramOption string // Histogram option for grpc metrics, should be comma separated value, ex:= "0.5,2,6" where start=0.5 factor=2, count=6
|
|
|
|
MetricsIP string // TCP port for liveness/ metrics requests
|
|
|
|
PidLimit int // PID limit to configure through cgroups")
|
|
|
|
MetricsPort int // TCP port for liveness/grpc metrics requests
|
|
|
|
PollTime time.Duration // time interval in seconds between each poll
|
|
|
|
PoolTimeout time.Duration // probe timeout in seconds
|
|
|
|
EnableGRPCMetrics bool // option to enable grpc metrics
|
|
|
|
|
|
|
|
IsControllerServer bool // if set to true start provisoner server
|
|
|
|
IsNodeServer bool // if set to true start node server
|
2019-11-06 04:52:07 +00:00
|
|
|
Version bool // cephcsi version
|
2019-10-10 10:15:44 +00:00
|
|
|
|
2020-06-24 08:12:12 +00:00
|
|
|
// SkipForceFlatten is set to false if the kernel supports mounting of
|
|
|
|
// rbd image or the image chain has the deep-flatten feature.
|
|
|
|
SkipForceFlatten bool
|
|
|
|
|
2019-10-10 10:15:44 +00:00
|
|
|
// cephfs related flags
|
|
|
|
ForceKernelCephFS bool // force to use the ceph kernel client even if the kernel is < 4.17
|
|
|
|
|
2020-06-24 06:44:02 +00:00
|
|
|
// RbdHardMaxCloneDepth is the hard limit for maximum number of nested volume clones that are taken before a flatten occurs
|
|
|
|
RbdHardMaxCloneDepth uint
|
|
|
|
|
|
|
|
// RbdSoftMaxCloneDepth is the soft limit for maximum number of nested volume clones that are taken before a flatten occurs
|
|
|
|
RbdSoftMaxCloneDepth uint
|
2020-07-01 05:27:11 +00:00
|
|
|
|
|
|
|
// MaxSnapshotsOnImage represents the maximum number of snapshots allowed
|
|
|
|
// on rbd image without flattening, once the limit is reached cephcsi will
|
|
|
|
// start flattening the older rbd images to allow more snapshots
|
|
|
|
MaxSnapshotsOnImage uint
|
2019-08-14 05:57:45 +00:00
|
|
|
}
|
|
|
|
|
2019-03-13 05:09:58 +00:00
|
|
|
// ValidateDriverName validates the driver name
|
|
|
|
func ValidateDriverName(driverName string) error {
|
2019-06-10 06:48:41 +00:00
|
|
|
if driverName == "" {
|
2019-03-13 05:09:58 +00:00
|
|
|
return errors.New("driver name is empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(driverName) > 63 {
|
|
|
|
return errors.New("driver name length should be less than 63 chars")
|
|
|
|
}
|
|
|
|
var err error
|
|
|
|
for _, msg := range validation.IsDNS1123Subdomain(strings.ToLower(driverName)) {
|
|
|
|
if err == nil {
|
|
|
|
err = errors.New(msg)
|
|
|
|
continue
|
|
|
|
}
|
2020-06-25 11:30:04 +00:00
|
|
|
err = fmt.Errorf("%s: %w", msg, err)
|
2019-03-13 05:09:58 +00:00
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
2019-05-14 19:15:01 +00:00
|
|
|
|
2020-06-22 06:58:47 +00:00
|
|
|
// GetKernelVersion returns the version of the running Unix (like) system from the
|
2020-06-17 18:30:09 +00:00
|
|
|
// 'utsname' structs 'release' component.
|
2020-06-22 06:58:47 +00:00
|
|
|
func GetKernelVersion() (string, error) {
|
2020-06-17 18:30:09 +00:00
|
|
|
utsname := unix.Utsname{}
|
|
|
|
err := unix.Uname(&utsname)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2020-06-17 18:34:12 +00:00
|
|
|
return strings.TrimRight(string(utsname.Release[:]), "\x00"), nil
|
2020-06-17 18:30:09 +00:00
|
|
|
}
|
|
|
|
|
2020-06-22 07:09:45 +00:00
|
|
|
// KernelVersion holds kernel related informations
|
|
|
|
type KernelVersion struct {
|
|
|
|
Version int
|
|
|
|
PatchLevel int
|
|
|
|
SubLevel int
|
|
|
|
ExtraVersion int // prefix of the part after the first "-"
|
|
|
|
Distribution string // component of full extraversion
|
|
|
|
Backport bool // backports have a fixed version/patchlevel/sublevel
|
|
|
|
}
|
|
|
|
|
|
|
|
// CheckKernelSupport checks the running kernel and comparing it to known
|
|
|
|
// versions that have support for required features . Distributors of
|
|
|
|
// enterprise Linux have backported quota support to previous versions. This
|
|
|
|
// function checks if the running kernel is one of the versions that have the
|
|
|
|
// feature/fixes backported.
|
|
|
|
//
|
|
|
|
// `uname -r` (or Uname().Utsname.Release has a format like 1.2.3-rc.vendor
|
|
|
|
// This can be slit up in the following components: - version (1) - patchlevel
|
|
|
|
// (2) - sublevel (3) - optional, defaults to 0 - extraversion (rc) - optional,
|
|
|
|
// matching integers only - distribution (.vendor) - optional, match against
|
|
|
|
// whole `uname -r` string
|
|
|
|
//
|
|
|
|
// For matching multiple versions, the kernelSupport type contains a backport
|
|
|
|
// bool, which will cause matching
|
|
|
|
// version+patchlevel+sublevel+(>=extraversion)+(~distribution)
|
|
|
|
//
|
|
|
|
// In case the backport bool is false, a simple check for higher versions than
|
|
|
|
// version+patchlevel+sublevel is done.
|
|
|
|
func CheckKernelSupport(release string, supportedVersions []KernelVersion) bool {
|
|
|
|
vers := strings.Split(strings.SplitN(release, "-", 2)[0], ".")
|
|
|
|
version, err := strconv.Atoi(vers[0])
|
|
|
|
if err != nil {
|
|
|
|
klog.Errorf("failed to parse version from %s: %v", release, err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
patchlevel, err := strconv.Atoi(vers[1])
|
|
|
|
if err != nil {
|
|
|
|
klog.Errorf("failed to parse patchlevel from %s: %v", release, err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
sublevel := 0
|
|
|
|
if len(vers) >= 3 {
|
|
|
|
sublevel, err = strconv.Atoi(vers[2])
|
|
|
|
if err != nil {
|
|
|
|
klog.Errorf("failed to parse sublevel from %s: %v", release, err)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
extra := strings.SplitN(release, "-", 2)
|
|
|
|
extraversion := 0
|
|
|
|
if len(extra) == 2 {
|
|
|
|
// ignore errors, 1st component of extraversion does not need to be an int
|
|
|
|
extraversion, err = strconv.Atoi(strings.Split(extra[1], ".")[0])
|
|
|
|
if err != nil {
|
|
|
|
// "go lint" wants err to be checked...
|
|
|
|
extraversion = 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// compare running kernel against known versions
|
|
|
|
for _, kernel := range supportedVersions {
|
|
|
|
if !kernel.Backport {
|
|
|
|
// deal with the default case(s), find >= match for version, patchlevel, sublevel
|
|
|
|
if version > kernel.Version || (version == kernel.Version && patchlevel > kernel.PatchLevel) ||
|
|
|
|
(version == kernel.Version && patchlevel == kernel.PatchLevel && sublevel >= kernel.SubLevel) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// specific backport, match distribution initially
|
|
|
|
if !strings.Contains(release, kernel.Distribution) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// strict match version, patchlevel, sublevel, and >= match extraversion
|
|
|
|
if version == kernel.Version && patchlevel == kernel.PatchLevel &&
|
|
|
|
sublevel == kernel.SubLevel && extraversion >= kernel.ExtraVersion {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
klog.Errorf("kernel %s does not support required features", release)
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2019-05-14 19:15:01 +00:00
|
|
|
// GenerateVolID generates a volume ID based on passed in parameters and version, to be returned
|
|
|
|
// to the CO system
|
2020-01-24 16:26:56 +00:00
|
|
|
func GenerateVolID(ctx context.Context, monitors string, cr *Credentials, locationID int64, pool, clusterID, objUUID string, volIDVersion uint16) (string, error) {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
if locationID == InvalidPoolID {
|
2020-05-14 11:23:55 +00:00
|
|
|
locationID, err = GetPoolID(monitors, cr, pool)
|
2020-01-24 16:26:56 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2019-05-14 19:15:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// generate the volume ID to return to the CO system
|
|
|
|
vi := CSIIdentifier{
|
2020-01-24 16:26:56 +00:00
|
|
|
LocationID: locationID,
|
2019-05-14 19:15:01 +00:00
|
|
|
EncodingVersion: volIDVersion,
|
|
|
|
ClusterID: clusterID,
|
|
|
|
ObjectUUID: objUUID,
|
|
|
|
}
|
|
|
|
|
|
|
|
volID, err := vi.ComposeCSIID()
|
|
|
|
|
|
|
|
return volID, err
|
|
|
|
}
|
2019-07-03 10:02:36 +00:00
|
|
|
|
|
|
|
// CreateMountPoint creates the directory with given path
|
|
|
|
func CreateMountPoint(mountPath string) error {
|
|
|
|
return os.MkdirAll(mountPath, 0750)
|
|
|
|
}
|
|
|
|
|
2019-07-25 09:01:10 +00:00
|
|
|
// checkDirExists checks directory exists or not
|
|
|
|
func checkDirExists(p string) bool {
|
|
|
|
if _, err := os.Stat(p); os.IsNotExist(err) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2019-07-03 10:02:36 +00:00
|
|
|
// IsMountPoint checks if the given path is mountpoint or not
|
|
|
|
func IsMountPoint(p string) (bool, error) {
|
|
|
|
dummyMount := mount.New("")
|
|
|
|
notMnt, err := dummyMount.IsLikelyNotMountPoint(p)
|
|
|
|
if err != nil {
|
|
|
|
return false, status.Error(codes.Internal, err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return !notMnt, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Mount mounts the source to target path
|
|
|
|
func Mount(source, target, fstype string, options []string) error {
|
|
|
|
dummyMount := mount.New("")
|
|
|
|
return dummyMount.Mount(source, target, fstype, options)
|
|
|
|
}
|
2020-06-17 09:00:55 +00:00
|
|
|
|
|
|
|
// MountOptionsAdd adds the `add` mount options to the `options` and returns a
|
|
|
|
// new string. In case `add` is already present in the `options`, `add` is not
|
|
|
|
// added again.
|
|
|
|
func MountOptionsAdd(options string, add ...string) string {
|
|
|
|
opts := strings.Split(options, ",")
|
|
|
|
newOpts := []string{}
|
|
|
|
// clean original options from empty strings
|
|
|
|
for _, opt := range opts {
|
|
|
|
if opt != "" {
|
|
|
|
newOpts = append(newOpts, opt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, opt := range add {
|
|
|
|
if opt != "" && !contains(newOpts, opt) {
|
|
|
|
newOpts = append(newOpts, opt)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.Join(newOpts, ",")
|
|
|
|
}
|
|
|
|
|
|
|
|
func contains(s []string, key string) bool {
|
|
|
|
for _, v := range s {
|
|
|
|
if v == key {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
2020-07-09 14:48:24 +00:00
|
|
|
|
|
|
|
// DefaultLog helps in logging with klog.level 1
|
|
|
|
func DefaultLog(message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(message, args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Default).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// UsefulLog helps in logging with klog.level 2
|
|
|
|
func UsefulLog(ctx context.Context, message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(Log(ctx, message), args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Useful).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExtendedLogMsg helps in logging a message with klog.level 3
|
|
|
|
func ExtendedLogMsg(message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(message, args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Extended).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ExtendedLog helps in logging with klog.level 3
|
|
|
|
func ExtendedLog(ctx context.Context, message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(Log(ctx, message), args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Extended).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DebugLogMsg helps in logging a message with klog.level 4
|
|
|
|
func DebugLogMsg(message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(message, args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Debug).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DebugLog helps in logging with klog.level 4
|
|
|
|
func DebugLog(ctx context.Context, message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(Log(ctx, message), args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Debug).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TraceLogMsg helps in logging a message with klog.level 5
|
|
|
|
func TraceLogMsg(message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(message, args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Trace).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TraceLog helps in logging with klog.level 5
|
|
|
|
func TraceLog(ctx context.Context, message string, args ...interface{}) {
|
|
|
|
logMessage := fmt.Sprintf(Log(ctx, message), args...)
|
|
|
|
// If logging is disabled, don't evaluate the arguments
|
|
|
|
if klog.V(Trace).Enabled() {
|
|
|
|
klog.InfoDepth(1, logMessage)
|
|
|
|
}
|
|
|
|
}
|