diff --git a/pkg/driver/mocks/mock_stats.go b/pkg/driver/mocks/mock_stats.go new file mode 100644 index 000000000..0174efe96 --- /dev/null +++ b/pkg/driver/mocks/mock_stats.go @@ -0,0 +1,98 @@ +// Code generated by MockGen. DO NOT EDIT. +// Source: sigs.k8s.io/ibm-powervs-block-csi-driver/pkg/driver (interfaces: StatsUtils) + +// Package mocks is a generated GoMock package. +package mocks + +import ( + reflect "reflect" + + gomock "github.com/golang/mock/gomock" +) + +// MockStatsUtils is a mock of StatsUtils interface. +type MockStatsUtils struct { + ctrl *gomock.Controller + recorder *MockStatsUtilsMockRecorder +} + +// MockStatsUtilsMockRecorder is the mock recorder for MockStatsUtils. +type MockStatsUtilsMockRecorder struct { + mock *MockStatsUtils +} + +// NewMockStatsUtils creates a new mock instance. +func NewMockStatsUtils(ctrl *gomock.Controller) *MockStatsUtils { + mock := &MockStatsUtils{ctrl: ctrl} + mock.recorder = &MockStatsUtilsMockRecorder{mock} + return mock +} + +// EXPECT returns an object that allows the caller to indicate expected use. +func (m *MockStatsUtils) EXPECT() *MockStatsUtilsMockRecorder { + return m.recorder +} + +// DeviceInfo mocks base method. +func (m *MockStatsUtils) DeviceInfo(arg0 string) (int64, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "DeviceInfo", arg0) + ret0, _ := ret[0].(int64) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// DeviceInfo indicates an expected call of DeviceInfo. +func (mr *MockStatsUtilsMockRecorder) DeviceInfo(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "DeviceInfo", reflect.TypeOf((*MockStatsUtils)(nil).DeviceInfo), arg0) +} + +// FSInfo mocks base method. +func (m *MockStatsUtils) FSInfo(arg0 string) (int64, int64, int64, int64, int64, int64, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "FSInfo", arg0) + ret0, _ := ret[0].(int64) + ret1, _ := ret[1].(int64) + ret2, _ := ret[2].(int64) + ret3, _ := ret[3].(int64) + ret4, _ := ret[4].(int64) + ret5, _ := ret[5].(int64) + ret6, _ := ret[6].(error) + return ret0, ret1, ret2, ret3, ret4, ret5, ret6 +} + +// FSInfo indicates an expected call of FSInfo. +func (mr *MockStatsUtilsMockRecorder) FSInfo(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "FSInfo", reflect.TypeOf((*MockStatsUtils)(nil).FSInfo), arg0) +} + +// IsBlockDevice mocks base method. +func (m *MockStatsUtils) IsBlockDevice(arg0 string) (bool, error) { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "IsBlockDevice", arg0) + ret0, _ := ret[0].(bool) + ret1, _ := ret[1].(error) + return ret0, ret1 +} + +// IsBlockDevice indicates an expected call of IsBlockDevice. +func (mr *MockStatsUtilsMockRecorder) IsBlockDevice(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "IsBlockDevice", reflect.TypeOf((*MockStatsUtils)(nil).IsBlockDevice), arg0) +} + +// IsPathNotExist mocks base method. +func (m *MockStatsUtils) IsPathNotExist(arg0 string) bool { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "IsPathNotExist", arg0) + ret0, _ := ret[0].(bool) + return ret0 +} + +// IsPathNotExist indicates an expected call of IsPathNotExist. +func (mr *MockStatsUtilsMockRecorder) IsPathNotExist(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "IsPathNotExist", reflect.TypeOf((*MockStatsUtils)(nil).IsPathNotExist), arg0) +} diff --git a/pkg/driver/node.go b/pkg/driver/node.go index d791a3542..3205b65bf 100644 --- a/pkg/driver/node.go +++ b/pkg/driver/node.go @@ -56,6 +56,7 @@ var ( nodeCaps = []csi.NodeServiceCapability_RPC_Type{ csi.NodeServiceCapability_RPC_STAGE_UNSTAGE_VOLUME, csi.NodeServiceCapability_RPC_EXPAND_VOLUME, + csi.NodeServiceCapability_RPC_GET_VOLUME_STATS, } ) @@ -66,6 +67,7 @@ type nodeService struct { driverOptions *Options pvmInstanceId string volumeLocks *util.VolumeLocks + stats StatsUtils } // newNodeService creates a new node service @@ -88,6 +90,7 @@ func newNodeService(driverOptions *Options) nodeService { driverOptions: driverOptions, pvmInstanceId: metadata.GetPvmInstanceId(), volumeLocks: util.NewVolumeLocks(), + stats: &VolumeStatUtils{}, } } @@ -380,7 +383,72 @@ func (d *nodeService) NodeUnpublishVolume(ctx context.Context, req *csi.NodeUnpu } func (d *nodeService) NodeGetVolumeStats(ctx context.Context, req *csi.NodeGetVolumeStatsRequest) (*csi.NodeGetVolumeStatsResponse, error) { - return nil, status.Error(codes.Unimplemented, "NodeGetVolumeStats is not implemented yet") + klog.V(4).Infof("NodeGetVolumeStats: called with args %+v", *req) + var resp *csi.NodeGetVolumeStatsResponse + + if req == nil || req.VolumeId == "" { + return nil, status.Error(codes.InvalidArgument, "Volume ID not provided") + } + + if req.VolumePath == "" { + return nil, status.Error(codes.InvalidArgument, "VolumePath not provided") + } + + volumePath := req.VolumePath + // return if path does not exist + if d.stats.IsPathNotExist(volumePath) { + return nil, status.Error(codes.NotFound, "VolumePath not exist") + } + + // check if volume mode is raw volume mode + isBlock, err := d.stats.IsBlockDevice(volumePath) + if err != nil { + return nil, status.Error(codes.Internal, fmt.Sprintf("Failed to check volume %s is block device or not: %v", req.VolumeId, err)) + } + // if block device, get deviceStats + if isBlock { + capacity, err := d.stats.DeviceInfo(volumePath) + if err != nil { + return nil, status.Error(codes.Internal, fmt.Sprintf("Failed to collect block device info: %v", err)) + } + + resp = &csi.NodeGetVolumeStatsResponse{ + Usage: []*csi.VolumeUsage{ + { + Total: capacity, + Unit: csi.VolumeUsage_BYTES, + }, + }, + } + + klog.V(4).Infof("Block Device Volume stats collected: %+v\n", resp) + return resp, nil + } + + // else get the file system stats + available, capacity, usage, inodes, inodesFree, inodesUsed, err := d.stats.FSInfo(volumePath) + if err != nil { + return nil, status.Error(codes.Internal, fmt.Sprintf("Failed to collect FSInfo: %v", err)) + } + resp = &csi.NodeGetVolumeStatsResponse{ + Usage: []*csi.VolumeUsage{ + { + Available: available, + Total: capacity, + Used: usage, + Unit: csi.VolumeUsage_BYTES, + }, + { + Available: inodesFree, + Total: inodes, + Used: inodesUsed, + Unit: csi.VolumeUsage_INODES, + }, + }, + } + + klog.V(4).Infof("FS Volume stats collected: %+v\n", resp) + return resp, nil } func (d *nodeService) NodeGetCapabilities(ctx context.Context, req *csi.NodeGetCapabilitiesRequest) (*csi.NodeGetCapabilitiesResponse, error) { diff --git a/pkg/driver/node_test.go b/pkg/driver/node_test.go index d8f513a21..6d9ebc781 100644 --- a/pkg/driver/node_test.go +++ b/pkg/driver/node_test.go @@ -16,6 +16,7 @@ package driver import ( "context" "errors" + "fmt" "reflect" "testing" @@ -540,6 +541,117 @@ func TestNodeGetInfo(t *testing.T) { } } +func TestNodeGetVolumeStats(t *testing.T) { + testCases := []struct { + name string + testFunc func(t *testing.T) + }{ + {name: "success block device volume", + testFunc: func(t *testing.T) { + mockCtl := gomock.NewController(t) + defer mockCtl.Finish() + mockStatUtils := mocks.NewMockStatsUtils(mockCtl) + + volumePath := "./test" + var mockCapacity int64 = 100 + mockStatUtils.EXPECT().IsPathNotExist(volumePath).Return(false) + mockStatUtils.EXPECT().IsBlockDevice(volumePath).Return(true, nil) + mockStatUtils.EXPECT().DeviceInfo(volumePath).Return(mockCapacity, nil) + driver := &nodeService{stats: mockStatUtils} + + req := csi.NodeGetVolumeStatsRequest{VolumeId: volumeID, VolumePath: volumePath} + + resp, err := driver.NodeGetVolumeStats(context.TODO(), &req) + if err != nil { + t.Fatalf("Expect no error but got: %v", err) + } + + if resp.Usage[0].Total != mockCapacity { + t.Fatalf("Expected total capacity as %d, got %d", mockCapacity, resp.Usage[0].Total) + } + }, + }, { + name: "failure path not exist", + testFunc: func(t *testing.T) { + mockCtl := gomock.NewController(t) + defer mockCtl.Finish() + mockStatUtils := mocks.NewMockStatsUtils(mockCtl) + + volumePath := "./test" + mockStatUtils.EXPECT().IsPathNotExist(volumePath).Return(true) + driver := &nodeService{stats: mockStatUtils} + + req := csi.NodeGetVolumeStatsRequest{VolumeId: volumeID, VolumePath: volumePath} + + _, err := driver.NodeGetVolumeStats(context.TODO(), &req) + expectErr(t, err, codes.NotFound) + }, + }, { + name: "failure checking for block device", + testFunc: func(t *testing.T) { + mockCtl := gomock.NewController(t) + defer mockCtl.Finish() + mockStatUtils := mocks.NewMockStatsUtils(mockCtl) + + volumePath := "./test" + mockStatUtils.EXPECT().IsPathNotExist(volumePath).Return(false) + mockStatUtils.EXPECT().IsBlockDevice(volumePath).Return(false, errors.New("Error checking for block device")) + driver := &nodeService{stats: mockStatUtils} + + req := csi.NodeGetVolumeStatsRequest{VolumeId: volumeID, VolumePath: volumePath} + + _, err := driver.NodeGetVolumeStats(context.TODO(), &req) + expectErr(t, err, codes.Internal) + }, + }, { + name: "failure collecting block device info", + testFunc: func(t *testing.T) { + mockCtl := gomock.NewController(t) + defer mockCtl.Finish() + mockStatUtils := mocks.NewMockStatsUtils(mockCtl) + + volumePath := "./test" + mockStatUtils.EXPECT().IsPathNotExist(volumePath).Return(false) + mockStatUtils.EXPECT().IsBlockDevice(volumePath).Return(true, nil) + mockStatUtils.EXPECT().DeviceInfo(volumePath).Return(int64(0), errors.New("Error collecting block device info")) + + driver := &nodeService{stats: mockStatUtils} + + req := csi.NodeGetVolumeStatsRequest{VolumeId: volumeID, VolumePath: volumePath} + + _, err := driver.NodeGetVolumeStats(context.TODO(), &req) + fmt.Println(err) + expectErr(t, err, codes.Internal) + }, + }, + { + name: "failure collecting fs info", + testFunc: func(t *testing.T) { + mockCtl := gomock.NewController(t) + defer mockCtl.Finish() + mockStatUtils := mocks.NewMockStatsUtils(mockCtl) + + volumePath := "./test" + mockStatUtils.EXPECT().IsPathNotExist(volumePath).Return(false) + mockStatUtils.EXPECT().IsBlockDevice(volumePath).Return(false, nil) + var statUnit int64 = 0 + mockStatUtils.EXPECT().FSInfo(volumePath).Return(statUnit, statUnit, statUnit, statUnit, statUnit, statUnit, errors.New("Error collecting FS Info")) + driver := &nodeService{stats: mockStatUtils} + + req := csi.NodeGetVolumeStatsRequest{VolumeId: volumeID, VolumePath: volumePath} + + _, err := driver.NodeGetVolumeStats(context.TODO(), &req) + fmt.Println(err) + + expectErr(t, err, codes.Internal) + }, + }, + } + for _, tc := range testCases { + t.Run(tc.name, tc.testFunc) + } +} + func expectErr(t *testing.T, actualErr error, expectedCode codes.Code) { if actualErr == nil { t.Fatalf("Expect error but got no error") diff --git a/pkg/driver/stats.go b/pkg/driver/stats.go new file mode 100644 index 000000000..428e16430 --- /dev/null +++ b/pkg/driver/stats.go @@ -0,0 +1,67 @@ +package driver + +import ( + "fmt" + "os" + "os/exec" + "strconv" + "strings" + + "golang.org/x/sys/unix" + "k8s.io/kubernetes/pkg/volume/util/fs" +) + +// StatsUtils ... +type StatsUtils interface { + FSInfo(path string) (int64, int64, int64, int64, int64, int64, error) + IsBlockDevice(devicePath string) (bool, error) + DeviceInfo(devicePath string) (int64, error) + IsPathNotExist(path string) bool +} + +// VolumeStatUtils ... +type VolumeStatUtils struct { +} + +// IsDevicePathNotExist ... +func (su *VolumeStatUtils) IsPathNotExist(path string) bool { + var stat unix.Stat_t + err := unix.Stat(path, &stat) + if err != nil { + if os.IsNotExist(err) { + return true + } + } + return false +} + +// IsBlockDevice ... +func (su *VolumeStatUtils) IsBlockDevice(devicePath string) (bool, error) { + var stat unix.Stat_t + err := unix.Stat(devicePath, &stat) + if err != nil { + return false, err + } + + return (stat.Mode & unix.S_IFMT) == unix.S_IFBLK, nil +} + +// DeviceInfo ... +func (su *VolumeStatUtils) DeviceInfo(devicePath string) (int64, error) { + output, err := exec.Command("blockdev", "--getsize64", devicePath).CombinedOutput() + if err != nil { + return 0, fmt.Errorf("failed to get size of block volume at path %s: output: %s, err: %v", devicePath, string(output), err) + } + strOut := strings.TrimSpace(string(output)) + gotSizeBytes, err := strconv.ParseInt(strOut, 10, 64) + if err != nil { + return 0, fmt.Errorf("failed to parse size '%s' into int", strOut) + } + + return gotSizeBytes, nil +} + +//FSInfo ... +func (su *VolumeStatUtils) FSInfo(path string) (int64, int64, int64, int64, int64, int64, error) { + return fs.Info(path) +}