Skip to content

Commit

Permalink
Implement GetNodeVolumeStats
Browse files Browse the repository at this point in the history
  • Loading branch information
flofourcade committed Jan 28, 2020

Verified

This commit was created on GitHub.com and signed with GitHub’s verified signature.
1 parent 5b197c4 commit fb13c61
Showing 1 changed file with 105 additions and 1 deletion.
106 changes: 105 additions & 1 deletion pkg/csi/service/node.go
Original file line number Diff line number Diff line change
@@ -32,6 +32,12 @@ import (
"google.golang.org/grpc/codes"
"google.golang.org/grpc/status"
"k8s.io/klog"

csipbv1 "github.com/container-storage-interface/spec/lib/go/csi"
"k8s.io/apimachinery/pkg/api/resource"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
k8svol "k8s.io/kubernetes/pkg/volume"
"k8s.io/kubernetes/pkg/volume/util/fs"
cnsvsphere "sigs.k8s.io/vsphere-csi-driver/pkg/common/cns-lib/vsphere"
cnsconfig "sigs.k8s.io/vsphere-csi-driver/pkg/common/config"
"sigs.k8s.io/vsphere-csi-driver/pkg/csi/service/common"
@@ -327,7 +333,98 @@ func (s *service) NodeGetVolumeStats(
req *csi.NodeGetVolumeStatsRequest) (
*csi.NodeGetVolumeStatsResponse, error) {

return nil, nil
var err error
targetPath := req.GetVolumePath()
if targetPath == "" {
err = fmt.Errorf("targetpath %v is empty", targetPath)
return nil, status.Error(codes.InvalidArgument, err.Error())
}

dev, err := getDevFromMount(targetPath)
if err != nil {
err = fmt.Errorf("unable to get targetpath %v device", targetPath)
return nil, status.Error(codes.Internal, err.Error())
}
if dev == nil {
err = fmt.Errorf("could not find device mounted on targetpath %v", targetPath)
return nil, status.Error(codes.Internal, err.Error())
}

//TODO Check that the matching device is a vSphere volume, and that the volID matches the mount point

volMetrics, err := getMetrics(targetPath)
if err != nil {
return nil, status.Error(codes.Internal, err.Error())
}

available, ok := (*(volMetrics.Available)).AsInt64()
if !ok {
klog.Errorf("failed to fetch available bytes")
}
capacity, ok := (*(volMetrics.Capacity)).AsInt64()
if !ok {
klog.Errorf("failed to fetch capacity bytes")
return nil, status.Error(codes.Unknown, "failed to fetch capacity bytes")
}
used, ok := (*(volMetrics.Used)).AsInt64()
if !ok {
klog.Errorf("failed to fetch used bytes")
}
inodes, ok := (*(volMetrics.Inodes)).AsInt64()
if !ok {
klog.Errorf("failed to fetch available inodes")
return nil, status.Error(codes.Unknown, "failed to fetch available inodes")

}
inodesFree, ok := (*(volMetrics.InodesFree)).AsInt64()
if !ok {
klog.Errorf("failed to fetch free inodes")
}

inodesUsed, ok := (*(volMetrics.InodesUsed)).AsInt64()
if !ok {
klog.Errorf("failed to fetch used inodes")
}
return &csi.NodeGetVolumeStatsResponse{
Usage: []*csi.VolumeUsage{
{
Available: available,
Total: capacity,
Used: used,
Unit: csipbv1.VolumeUsage_BYTES,
},
{
Available: inodesFree,
Total: inodes,
Used: inodesUsed,
Unit: csipbv1.VolumeUsage_INODES,
},
},
}, nil
}

//Get volume metrics using k8s fsInfo strategy
func getMetrics(path string) (*k8svol.Metrics, error) {
if path == "" {
return nil, fmt.Errorf("No path given")
}

available, capacity, usage, inodes, inodesFree, inodesUsed, err := fs.FsInfo(path)
metrics := &k8svol.Metrics{Time: metav1.Now()}
if err != nil {
return nil, err
}
metrics.Available = resource.NewQuantity(available, resource.BinarySI)
metrics.Capacity = resource.NewQuantity(capacity, resource.BinarySI)
metrics.Used = resource.NewQuantity(usage, resource.BinarySI)
metrics.Inodes = resource.NewQuantity(inodes, resource.BinarySI)
metrics.InodesFree = resource.NewQuantity(inodesFree, resource.BinarySI)
metrics.InodesUsed = resource.NewQuantity(inodesUsed, resource.BinarySI)
if err != nil {
return metrics, err
}

return metrics, nil
}

func (s *service) NodeGetCapabilities(
@@ -344,6 +441,13 @@ func (s *service) NodeGetCapabilities(
},
},
},
{
Type: &csi.NodeServiceCapability_Rpc{
Rpc: &csi.NodeServiceCapability_RPC{
Type: csi.NodeServiceCapability_RPC_GET_VOLUME_STATS,
},
},
},
},
}, nil
}

0 comments on commit fb13c61

Please sign in to comment.