From 267a09001d091de33de5b7f8444bb071089e0c47 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Thu, 19 Jan 2023 15:54:10 +0000 Subject: [PATCH] mount: fix check for empty mount point on Linux #3562 --- cmd/mountlib/check_linux.go | 14 +++++++++++--- cmd/mountlib/check_other.go | 22 +--------------------- cmd/mountlib/utils.go | 22 ++++++++++++++++++++++ 3 files changed, 34 insertions(+), 24 deletions(-) diff --git a/cmd/mountlib/check_linux.go b/cmd/mountlib/check_linux.go index 1f241af87..1c32a6d8a 100644 --- a/cmd/mountlib/check_linux.go +++ b/cmd/mountlib/check_linux.go @@ -33,12 +33,20 @@ func CheckMountEmpty(mountpoint string) error { if err != nil { return fmt.Errorf("cannot read %s: %w", mtabPath, err) } + foundAutofs := false for _, entry := range entries { - if entry.Dir == mountpointAbs && entry.Type != "autofs" { - return fmt.Errorf(msg, mountpointAbs) + if entry.Dir == mountpointAbs { + if entry.Type != "autofs" { + return fmt.Errorf(msg, mountpointAbs) + } + foundAutofs = true } } - return nil + // It isn't safe to list an autofs in the middle of mounting + if foundAutofs { + return nil + } + return checkMountEmpty(mountpoint) } // CheckMountReady checks whether mountpoint is mounted by rclone. diff --git a/cmd/mountlib/check_other.go b/cmd/mountlib/check_other.go index c8876632c..328f83bca 100644 --- a/cmd/mountlib/check_other.go +++ b/cmd/mountlib/check_other.go @@ -4,33 +4,13 @@ package mountlib import ( - "fmt" - "io" - "os" "time" - - "github.com/rclone/rclone/fs" ) // CheckMountEmpty checks if mountpoint folder is empty. // On non-Linux unixes we list directory to ensure that. func CheckMountEmpty(mountpoint string) error { - fp, err := os.Open(mountpoint) - if err != nil { - return fmt.Errorf("cannot open: %s: %w", mountpoint, err) - } - defer fs.CheckClose(fp, &err) - - _, err = fp.Readdirnames(1) - if err == io.EOF { - return nil - } - - const msg = "directory is not empty, use --allow-non-empty to mount anyway: %s" - if err == nil { - return fmt.Errorf(msg, mountpoint) - } - return fmt.Errorf(msg+": %w", mountpoint, err) + return checkMountEmpty(mountpoint) } // CheckMountReady should check if mountpoint is mounted by rclone. diff --git a/cmd/mountlib/utils.go b/cmd/mountlib/utils.go index 099e3a767..5148f818b 100644 --- a/cmd/mountlib/utils.go +++ b/cmd/mountlib/utils.go @@ -2,6 +2,8 @@ package mountlib import ( "fmt" + "io" + "os" "path/filepath" "runtime" "strings" @@ -84,6 +86,26 @@ func (m *MountPoint) CheckAllowed() error { return nil } +// checkMountEmpty checks if mountpoint folder is empty by listing it. +func checkMountEmpty(mountpoint string) error { + fp, err := os.Open(mountpoint) + if err != nil { + return fmt.Errorf("cannot open: %s: %w", mountpoint, err) + } + defer fs.CheckClose(fp, &err) + + _, err = fp.Readdirnames(1) + if err == io.EOF { + return nil + } + + const msg = "%q is not empty, use --allow-non-empty to mount anyway" + if err == nil { + return fmt.Errorf(msg, mountpoint) + } + return fmt.Errorf(msg+": %w", mountpoint, err) +} + // SetVolumeName with sensible default func (m *MountPoint) SetVolumeName(vol string) { if vol == "" {