2021-07-24 15:27:48 +02:00
|
|
|
//go:build linux
|
|
|
|
// +build linux
|
|
|
|
|
|
|
|
package mountlib
|
|
|
|
|
|
|
|
import (
|
2021-11-04 11:12:57 +01:00
|
|
|
"errors"
|
|
|
|
"fmt"
|
2021-07-24 15:27:48 +02:00
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/artyom/mtab"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
mtabPath = "/proc/mounts"
|
|
|
|
pollInterval = 100 * time.Millisecond
|
|
|
|
)
|
|
|
|
|
|
|
|
// CheckMountEmpty checks if folder is not already a mountpoint.
|
|
|
|
// On Linux we use the OS-specific /proc/mount API so the check won't access the path.
|
|
|
|
// Directories marked as "mounted" by autofs are considered not mounted.
|
|
|
|
func CheckMountEmpty(mountpoint string) error {
|
2022-06-08 22:54:39 +02:00
|
|
|
const msg = "directory already mounted, use --allow-non-empty to mount anyway: %s"
|
2021-07-24 15:27:48 +02:00
|
|
|
|
|
|
|
mountpointAbs, err := filepath.Abs(mountpoint)
|
|
|
|
if err != nil {
|
2021-11-04 11:12:57 +01:00
|
|
|
return fmt.Errorf("cannot get absolute path: %s: %w", mountpoint, err)
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
entries, err := mtab.Entries(mtabPath)
|
|
|
|
if err != nil {
|
2021-11-04 11:12:57 +01:00
|
|
|
return fmt.Errorf("cannot read %s: %w", mtabPath, err)
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
2023-01-19 16:54:10 +01:00
|
|
|
foundAutofs := false
|
2021-07-24 15:27:48 +02:00
|
|
|
for _, entry := range entries {
|
2023-01-19 16:54:10 +01:00
|
|
|
if entry.Dir == mountpointAbs {
|
|
|
|
if entry.Type != "autofs" {
|
|
|
|
return fmt.Errorf(msg, mountpointAbs)
|
|
|
|
}
|
|
|
|
foundAutofs = true
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
|
|
|
}
|
2023-01-19 16:54:10 +01:00
|
|
|
// It isn't safe to list an autofs in the middle of mounting
|
|
|
|
if foundAutofs {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return checkMountEmpty(mountpoint)
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// CheckMountReady checks whether mountpoint is mounted by rclone.
|
|
|
|
// Only mounts with type "rclone" or "fuse.rclone" count.
|
|
|
|
func CheckMountReady(mountpoint string) error {
|
|
|
|
mountpointAbs, err := filepath.Abs(mountpoint)
|
|
|
|
if err != nil {
|
2021-11-04 11:12:57 +01:00
|
|
|
return fmt.Errorf("cannot get absolute path: %s: %w", mountpoint, err)
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
|
|
|
entries, err := mtab.Entries(mtabPath)
|
|
|
|
if err != nil {
|
2021-11-04 11:12:57 +01:00
|
|
|
return fmt.Errorf("cannot read %s: %w", mtabPath, err)
|
2021-07-24 15:27:48 +02:00
|
|
|
}
|
|
|
|
for _, entry := range entries {
|
|
|
|
if entry.Dir == mountpointAbs && strings.Contains(entry.Type, "rclone") {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return errors.New("mount not ready")
|
|
|
|
}
|
|
|
|
|
|
|
|
// WaitMountReady waits until mountpoint is mounted by rclone.
|
|
|
|
func WaitMountReady(mountpoint string, timeout time.Duration) (err error) {
|
|
|
|
endTime := time.Now().Add(timeout)
|
|
|
|
for {
|
|
|
|
err = CheckMountReady(mountpoint)
|
|
|
|
delay := time.Until(endTime)
|
|
|
|
if err == nil || delay <= 0 {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
if delay > pollInterval {
|
|
|
|
delay = pollInterval
|
|
|
|
}
|
|
|
|
time.Sleep(delay)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|