Skip to content

Commit

Permalink
Add support for checking if a device is being used by a filesystem
Browse files Browse the repository at this point in the history
  • Loading branch information
pwschuurman committed Oct 10, 2024
1 parent eaf27e4 commit fa5ac2b
Show file tree
Hide file tree
Showing 6 changed files with 158 additions and 9 deletions.
8 changes: 8 additions & 0 deletions pkg/deviceutils/device-utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import (

"k8s.io/apimachinery/pkg/util/wait"
"k8s.io/klog/v2"
"k8s.io/mount-utils"
pathutils "k8s.io/utils/path"
"sigs.k8s.io/gcp-compute-persistent-disk-csi-driver/pkg/resizefs"
)
Expand Down Expand Up @@ -91,6 +92,13 @@ type DeviceUtils interface {

// Resize returns whether or not a device needs resizing.
Resize(resizer resizefs.Resizefs, devicePath string, deviceMountPath string) (bool, error)

// IsDeviceFilesystemInUse returns if a device path with the specified fstype
// TODO: Mounter is passed in in order to call GetDiskFormat()
// This is currently only implemented in mounter_linux, not mounter_windows.
// Refactor this interface and function call up the stack to the caller once it is
// implemented in mounter_windows.
IsDeviceFilesystemInUse(mounter *mount.SafeFormatAndMount, devicePath, devFsPath string) (bool, error)
}

type deviceUtils struct {
Expand Down
22 changes: 22 additions & 0 deletions pkg/deviceutils/device-utils_linux.go
Original file line number Diff line number Diff line change
Expand Up @@ -20,9 +20,31 @@ import (
"fmt"
"os"
"path/filepath"

"k8s.io/mount-utils"
)

func (_ *deviceUtils) DisableDevice(devicePath string) error {
deviceName := filepath.Base(devicePath)
return os.WriteFile(fmt.Sprintf("/sys/block/%s/device/state", deviceName), []byte("offline"), 0644)
}

func (_ *deviceUtils) IsDeviceFilesystemInUse(mounter *mount.SafeFormatAndMount, devicePath, devFsPath string) (bool, error) {
fstype, err := mounter.GetDiskFormat(devicePath)
if err != nil {
return false, fmt.Errorf("failed to get disk format for %s (aka %s): %v", devicePath, devFsPath, err)
}

devFsName := filepath.Base(devFsPath)
sysFsTypePath := fmt.Sprintf("/sys/fs/%s/%s", fstype, devFsName)
stat, err := os.Stat(sysFsTypePath)
if err != nil {
if os.IsNotExist(err) {
// Path doesn't exist, indicating the device is NOT in use
return false, nil
}
return false, err
}

return stat.IsDir(), nil
}
8 changes: 8 additions & 0 deletions pkg/deviceutils/device-utils_windows.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,15 @@ limitations under the License.

package deviceutils

import "k8s.io/mount-utils"

func (_ *deviceUtils) DisableDevice(devicePath string) error {
// No disabling is necessary on windows.
return nil
}

func (_ *deviceUtils) IsDeviceFilesystemInUse(mounter *mount.SafeFormatAndMount, devicePath, devFsPath string) (bool, error) {
// We don't support checking if a device filesystem is captured elsewhere by the system
// Return false, to skip this check. Assume the filesystem is not in use.
return false, nil
}
11 changes: 10 additions & 1 deletion pkg/deviceutils/fake-device-utils.go
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,10 @@ limitations under the License.

package deviceutils

import "sigs.k8s.io/gcp-compute-persistent-disk-csi-driver/pkg/resizefs"
import (
"k8s.io/mount-utils"
"sigs.k8s.io/gcp-compute-persistent-disk-csi-driver/pkg/resizefs"
)

type fakeDeviceUtils struct {
skipResize bool
Expand Down Expand Up @@ -50,3 +53,9 @@ func (du *fakeDeviceUtils) Resize(resizer resizefs.Resizefs, devicePath string,
}
return resizer.Resize(devicePath, deviceMountPath)
}

func (_ *fakeDeviceUtils) IsDeviceFilesystemInUse(mounter *mount.SafeFormatAndMount, devicePath, devFsPath string) (bool, error) {
// We don't support checking if a device filesystem is captured elsewhere by the system
// Return false, to skip this check.
return false, nil
}
42 changes: 34 additions & 8 deletions pkg/gce-pd-csi-driver/node.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ package gceGCEDriver

import (
"context"
"errors"
"fmt"
"os"
"path/filepath"
Expand Down Expand Up @@ -392,21 +393,46 @@ func (ns *GCENodeServer) NodeUnstageVolume(ctx context.Context, req *csi.NodeUns
return nil, status.Error(codes.Internal, fmt.Sprintf("NodeUnstageVolume failed: %v\nUnmounting arguments: %s\n", err.Error(), stagingTargetPath))
}

devicePath, err := getDevicePath(ns, volumeID, "" /* partition, which is unused */)
if err != nil {
klog.Errorf("Failed to find device path for volume %s. Device may not be detached cleanly (error is ignored and unstaging is continuing): %v", volumeID, err.Error())
} else {
if devFsPath, err := filepath.EvalSymlinks(devicePath); err != nil {
klog.Warningf("filepath.EvalSymlinks(%q) failed when trying to disable device: %w (ignored, unstaging continues)", devicePath, err)
} else if err := ns.DeviceUtils.DisableDevice(devFsPath); err != nil {
klog.Warningf("Failed to disabled device %s (aka %s) for volume %s. Device may not be detached cleanly (ignored, unstaging continues): %w", devicePath, devFsPath, volumeID, err)
if err := ns.safelyDisableDevice(volumeID); err != nil {
var targetErr *ignoreableError
if errors.As(err, &targetErr) {
klog.Warningf("Device may not be detached cleanly for volume %s (ignored, unstaging continues): %v", volumeID, err)
} else {
return nil, status.Errorf(codes.Internal, "NodeUnstageVolume for volume %s failed: %v", volumeID, err)
}
}

klog.V(4).Infof("NodeUnstageVolume succeeded on %v from %s", volumeID, stagingTargetPath)
return &csi.NodeUnstageVolumeResponse{}, nil
}

// A private error wrapper used to pass control flow decisions up to the caller
type ignoreableError struct{ error }

func (ns *GCENodeServer) safelyDisableDevice(volumeID string) error {
devicePath, err := getDevicePath(ns, volumeID, "" /* partition, which is unused */)
if err != nil {
return &ignoreableError{fmt.Errorf("failed to find device path for volume %s: %v", volumeID, err.Error())}
}

devFsPath, err := filepath.EvalSymlinks(devicePath)
if err != nil {
return &ignoreableError{fmt.Errorf("filepath.EvalSymlinks(%q) failed: %v", devicePath, err)}
}

if inUse, err := ns.DeviceUtils.IsDeviceFilesystemInUse(ns.Mounter, devicePath, devFsPath); err != nil {
return &ignoreableError{fmt.Errorf("failed to check if device %s (aka %s) is in use: %v", devicePath, devFsPath, err)}
} else if inUse {
return fmt.Errorf("device %s (aka %s) is still in use", devicePath, devFsPath)
}

if err := ns.DeviceUtils.DisableDevice(devFsPath); err != nil {
return &ignoreableError{fmt.Errorf("failed to disable device %s (aka %s): %v", devicePath, devFsPath, err)}
}

return nil
}

func (ns *GCENodeServer) NodeGetCapabilities(ctx context.Context, req *csi.NodeGetCapabilitiesRequest) (*csi.NodeGetCapabilitiesResponse, error) {
return &csi.NodeGetCapabilitiesResponse{
Capabilities: ns.Driver.nscap,
Expand Down
76 changes: 76 additions & 0 deletions test/e2e/tests/single_zone_e2e_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -1341,6 +1341,82 @@ var _ = Describe("GCE PD CSI Driver", func() {

Expect(err).To(BeNil(), "no error expected when passed valid compute url")
})

It("Should block unstage if filesystem mounted", func() {
testContext := getRandomTestContext()

p, z, _ := testContext.Instance.GetIdentity()
client := testContext.Client
instance := testContext.Instance

// Create Disk
volName, volID := createAndValidateUniqueZonalDisk(client, p, z, standardDiskType)

defer func() {
// Delete Disk
err := client.DeleteVolume(volID)
Expect(err).To(BeNil(), "DeleteVolume failed")

// Validate Disk Deleted
_, err = computeService.Disks.Get(p, z, volName).Do()
Expect(gce.IsGCEError(err, "notFound")).To(BeTrue(), "Expected disk to not be found")
}()

// Attach Disk
err := client.ControllerPublishVolume(volID, instance.GetNodeID(), false /* forceAttach */)
Expect(err).To(BeNil(), "ControllerPublishVolume failed with error for disk %v on node %v: %v", volID, instance.GetNodeID(), err)

defer func() {
// Detach Disk
err = client.ControllerUnpublishVolume(volID, instance.GetNodeID())
if err != nil {
klog.Errorf("Failed to detach disk: %v", err)
}
}()

// Stage Disk
stageDir := filepath.Join("/tmp/", volName, "stage")
err = client.NodeStageExt4Volume(volID, stageDir)
Expect(err).To(BeNil(), "failed to stage volume: %v", err)

// Create private bind mount
boundMountStageDir := filepath.Join("/tmp/bindmount", volName, "bindmount")
boundMountStageMkdirOutput, err := instance.SSH("mkdir", "-p", boundMountStageDir)
Expect(err).To(BeNil(), "mkdir failed on instance %v: output: %v: %v", instance.GetNodeID(), boundMountStageMkdirOutput, err)
bindMountOutput, err := instance.SSH("mount", "--rbind", "--make-private", stageDir, boundMountStageDir)
Expect(err).To(BeNil(), "Bind mount failed on instance %v: output: %v: %v", instance.GetNodeID(), bindMountOutput, err)

privateBindMountRemoved := false
unmountAndRmPrivateBindMount := func() {
if !privateBindMountRemoved {
// Umount and delete private mount staging directory
bindUmountOutput, err := instance.SSH("umount", boundMountStageDir)
Expect(err).To(BeNil(), "Bind mount failed on instance %v: output: %v: %v", instance.GetNodeID(), bindUmountOutput, err)
err = testutils.RmAll(instance, boundMountStageDir)
Expect(err).To(BeNil(), "Failed to rm mount stage dir %s: %v", boundMountStageDir, err)
}
privateBindMountRemoved = true
}

defer func() {
unmountAndRmPrivateBindMount()
}()

// Unstage Disk
err = client.NodeUnstageVolume(volID, stageDir)
Expect(err).ToNot(BeNil(), "Expected failure during unstage")
Expect(err).To(MatchError(ContainSubstring(("is still in use"))))

// Unmount private bind mount and try again
unmountAndRmPrivateBindMount()

// Unstage Disk
err = client.NodeUnstageVolume(volID, stageDir)
Expect(err).To(BeNil(), "Failed to unstage volume: %v", err)
fp := filepath.Join("/tmp/", volName)
err = testutils.RmAll(instance, fp)
Expect(err).To(BeNil(), "Failed to rm file path %s: %v", fp, err)
})
})

func equalWithinEpsilon(a, b, epsiolon int64) bool {
Expand Down

0 comments on commit fa5ac2b

Please sign in to comment.