2017-06-19 12:44:49 +00:00
package mountlib
import (
2021-08-13 17:42:33 +00:00
"context"
2021-11-04 10:12:57 +00:00
"fmt"
2017-06-19 12:44:49 +00:00
"log"
2017-11-09 00:37:27 +00:00
"os"
2017-11-16 12:20:53 +00:00
"runtime"
2018-05-03 08:34:07 +00:00
"strings"
2020-11-27 10:50:10 +00:00
"sync"
2018-03-02 16:39:42 +00:00
"time"
2017-06-19 12:44:49 +00:00
2019-07-28 17:47:38 +00:00
"github.com/rclone/rclone/cmd"
"github.com/rclone/rclone/fs"
"github.com/rclone/rclone/fs/config"
"github.com/rclone/rclone/fs/config/flags"
2020-07-23 16:17:01 +00:00
"github.com/rclone/rclone/fs/rc"
2020-07-23 12:08:38 +00:00
"github.com/rclone/rclone/lib/atexit"
2021-08-18 11:07:09 +00:00
"github.com/rclone/rclone/lib/daemonize"
2019-07-28 17:47:38 +00:00
"github.com/rclone/rclone/vfs"
2021-01-03 00:05:52 +00:00
"github.com/rclone/rclone/vfs/vfscommon"
2019-07-28 17:47:38 +00:00
"github.com/rclone/rclone/vfs/vfsflags"
2021-01-03 00:05:52 +00:00
sysdnotify "github.com/iguanesolutions/go-systemd/v5/notify"
2017-06-19 12:44:49 +00:00
"github.com/spf13/cobra"
2020-07-23 16:17:01 +00:00
"github.com/spf13/pflag"
2017-06-19 12:44:49 +00:00
)
2020-07-23 16:17:01 +00:00
// Options for creating the mount
type Options struct {
DebugFUSE bool
AllowNonEmpty bool
AllowRoot bool
AllowOther bool
DefaultPermissions bool
WritebackCache bool
Daemon bool
2021-08-18 11:07:09 +00:00
DaemonWait time . Duration // time to wait for ready mount from daemon, maximum on Linux or constant on macOS/BSD
2020-07-23 16:17:01 +00:00
MaxReadAhead fs . SizeSuffix
2017-11-07 17:09:08 +00:00
ExtraOptions [ ] string
ExtraFlags [ ] string
2020-07-23 16:17:01 +00:00
AttrTimeout time . Duration // how long the kernel caches attribute for
2018-05-03 08:34:07 +00:00
VolumeName string
2020-07-23 16:17:01 +00:00
NoAppleDouble bool
NoAppleXattr bool
2018-07-18 15:21:35 +00:00
DaemonTimeout time . Duration // OSXFUSE only
2020-07-23 16:17:01 +00:00
AsyncRead bool
2020-11-06 13:21:38 +00:00
NetworkMode bool // Windows only
2020-07-23 16:17:01 +00:00
}
// DefaultOpt is the default values for creating the mount
var DefaultOpt = Options {
MaxReadAhead : 128 * 1024 ,
AttrTimeout : 1 * time . Second , // how long the kernel caches attribute for
NoAppleDouble : true , // use noappledouble by default
NoAppleXattr : false , // do not use noapplexattr by default
AsyncRead : true , // do async reads by default
}
2017-06-19 12:44:49 +00:00
2020-04-25 05:03:07 +00:00
type (
// UnmountFn is called to unmount the file system
UnmountFn func ( ) error
// MountFn is called to mount the file system
2020-07-23 16:17:01 +00:00
MountFn func ( VFS * vfs . VFS , mountpoint string , opt * Options ) ( <- chan error , func ( ) error , error )
2020-04-25 05:03:07 +00:00
)
2021-01-03 00:05:52 +00:00
// MountPoint represents a mount with options and runtime state
type MountPoint struct {
MountPoint string
MountedOn time . Time
MountOpt Options
VFSOpt vfscommon . Options
Fs fs . Fs
VFS * vfs . VFS
MountFn MountFn
UnmountFn UnmountFn
ErrChan <- chan error
}
2019-10-18 09:53:07 +00:00
// Global constants
const (
2020-01-19 14:54:55 +00:00
MaxLeafSize = 1024 // don't pass file names longer than this
2019-10-18 09:53:07 +00:00
)
2019-06-24 10:54:38 +00:00
func init ( ) {
2021-08-18 11:07:09 +00:00
switch runtime . GOOS {
case "darwin" :
// DaemonTimeout defaults to non-zero for macOS
// (this is a macOS specific kernel option unrelated to DaemonWait)
2021-02-21 12:56:19 +00:00
DefaultOpt . DaemonTimeout = 10 * time . Minute
2020-07-23 16:17:01 +00:00
}
2021-08-18 11:07:09 +00:00
switch runtime . GOOS {
case "linux" :
// Linux provides /proc/mounts to check mount status
// so --daemon-wait means *maximum* time to wait
DefaultOpt . DaemonWait = 60 * time . Second
case "darwin" , "openbsd" , "freebsd" , "netbsd" :
// On BSD we can't check mount status yet
// so --daemon-wait is just a *constant* delay
DefaultOpt . DaemonWait = 5 * time . Second
}
// Opt must be assigned in the init block to ensure changes really get in
2020-07-23 16:17:01 +00:00
Opt = DefaultOpt
2021-08-18 11:07:09 +00:00
}
// Opt contains options set by command line flags
var Opt Options
2020-07-23 16:17:01 +00:00
// AddFlags adds the non filing system specific flags to the command
func AddFlags ( flagSet * pflag . FlagSet ) {
rc . AddOption ( "mount" , & Opt )
2021-08-16 09:30:01 +00:00
flags . BoolVarP ( flagSet , & Opt . DebugFUSE , "debug-fuse" , "" , Opt . DebugFUSE , "Debug the FUSE internals - needs -v" )
flags . DurationVarP ( flagSet , & Opt . AttrTimeout , "attr-timeout" , "" , Opt . AttrTimeout , "Time for which file/directory attributes are cached" )
flags . StringArrayVarP ( flagSet , & Opt . ExtraOptions , "option" , "o" , [ ] string { } , "Option for libfuse/WinFsp (repeat if required)" )
flags . StringArrayVarP ( flagSet , & Opt . ExtraFlags , "fuse-flag" , "" , [ ] string { } , "Flags or arguments to be passed direct to libfuse/WinFsp (repeat if required)" )
2020-11-10 23:17:25 +00:00
// Non-Windows only
2021-08-16 09:30:01 +00:00
flags . BoolVarP ( flagSet , & Opt . Daemon , "daemon" , "" , Opt . Daemon , "Run mount in background and exit parent process (as background output is suppressed, use --log-file with --log-format=pid,... to monitor) (not supported on Windows)" )
flags . DurationVarP ( flagSet , & Opt . DaemonTimeout , "daemon-timeout" , "" , Opt . DaemonTimeout , "Time limit for rclone to respond to kernel (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . DefaultPermissions , "default-permissions" , "" , Opt . DefaultPermissions , "Makes kernel enforce access control based on the file mode (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . AllowNonEmpty , "allow-non-empty" , "" , Opt . AllowNonEmpty , "Allow mounting over a non-empty directory (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . AllowRoot , "allow-root" , "" , Opt . AllowRoot , "Allow access to root user (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . AllowOther , "allow-other" , "" , Opt . AllowOther , "Allow access to other users (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . AsyncRead , "async-read" , "" , Opt . AsyncRead , "Use asynchronous reads (not supported on Windows)" )
flags . FVarP ( flagSet , & Opt . MaxReadAhead , "max-read-ahead" , "" , "The number of bytes that can be prefetched for sequential reads (not supported on Windows)" )
flags . BoolVarP ( flagSet , & Opt . WritebackCache , "write-back-cache" , "" , Opt . WritebackCache , "Makes kernel buffer writes before sending them to rclone (without this, writethrough caching is used) (not supported on Windows)" )
2020-11-10 23:17:25 +00:00
// Windows and OSX
2021-08-16 09:30:01 +00:00
flags . StringVarP ( flagSet , & Opt . VolumeName , "volname" , "" , Opt . VolumeName , "Set the volume name (supported on Windows and OSX only)" )
2020-11-10 23:17:25 +00:00
// OSX only
2021-08-16 09:30:01 +00:00
flags . BoolVarP ( flagSet , & Opt . NoAppleDouble , "noappledouble" , "" , Opt . NoAppleDouble , "Ignore Apple Double (._) and .DS_Store files (supported on OSX only)" )
flags . BoolVarP ( flagSet , & Opt . NoAppleXattr , "noapplexattr" , "" , Opt . NoAppleXattr , "Ignore all \"com.apple.*\" extended attributes (supported on OSX only)" )
2020-11-10 23:17:25 +00:00
// Windows only
2021-08-16 09:30:01 +00:00
flags . BoolVarP ( flagSet , & Opt . NetworkMode , "network-mode" , "" , Opt . NetworkMode , "Mount as remote network drive, instead of fixed disk drive (supported on Windows only)" )
2021-08-18 11:07:09 +00:00
// Unix only
2021-08-16 09:30:01 +00:00
flags . DurationVarP ( flagSet , & Opt . DaemonWait , "daemon-wait" , "" , Opt . DaemonWait , "Time to wait for ready mount from daemon (maximum time on Linux, constant sleep time on OSX/BSD) (not supported on Windows)" )
2019-06-24 10:54:38 +00:00
}
2017-06-19 12:44:49 +00:00
// NewMountCommand makes a mount command with the given name and Mount function
2020-07-23 12:08:38 +00:00
func NewMountCommand ( commandName string , hidden bool , mount MountFn ) * cobra . Command {
2019-10-11 15:58:11 +00:00
var commandDefinition = & cobra . Command {
2020-02-11 12:05:43 +00:00
Use : commandName + " remote:path /path/to/mountpoint" ,
Hidden : hidden ,
Short : ` Mount the remote as file system on a mountpoint. ` ,
2021-01-03 00:05:52 +00:00
Long : strings . ReplaceAll ( strings . ReplaceAll ( mountHelp , "|" , "`" ) , "@" , commandName ) + vfs . Help ,
2017-06-19 12:44:49 +00:00
Run : func ( command * cobra . Command , args [ ] string ) {
cmd . CheckArgs ( 2 , 2 , command , args )
2018-08-21 08:41:16 +00:00
2021-08-13 17:42:33 +00:00
if fs . GetConfig ( context . Background ( ) ) . UseListR {
fs . Logf ( nil , "--fast-list does nothing on a mount" )
}
2021-01-03 00:05:52 +00:00
if Opt . Daemon {
2018-08-21 08:41:16 +00:00
config . PassConfigKeyForDaemonization = true
}
2021-08-18 11:07:09 +00:00
if os . Getenv ( "PATH" ) == "" && runtime . GOOS != "windows" {
// PATH can be unset when running under Autofs or Systemd mount
fs . Debugf ( nil , "Using fallback PATH to run fusermount" )
_ = os . Setenv ( "PATH" , "/bin:/usr/bin" )
}
2017-06-19 12:44:49 +00:00
// Show stats if the user has specifically requested them
if cmd . ShowStats ( ) {
2018-10-03 20:46:18 +00:00
defer cmd . StartStats ( ) ( )
2017-06-19 12:44:49 +00:00
}
2021-01-03 00:05:52 +00:00
mnt := & MountPoint {
MountFn : mount ,
MountPoint : args [ 1 ] ,
Fs : cmd . NewFsDir ( args ) ,
MountOpt : Opt ,
VFSOpt : vfsflags . Opt ,
2020-03-03 18:59:47 +00:00
}
2018-05-03 08:34:07 +00:00
2021-08-18 11:07:09 +00:00
daemon , err := mnt . Mount ( )
// Wait for foreground mount, if any...
if daemon == nil {
if err == nil {
err = mnt . Wait ( )
}
if err != nil {
log . Fatalf ( "Fatal error: %v" , err )
}
return
}
// Wait for daemon, if any...
killOnce := sync . Once { }
killDaemon := func ( reason string ) {
killOnce . Do ( func ( ) {
if err := daemon . Signal ( os . Interrupt ) ; err != nil {
fs . Errorf ( nil , "%s. Failed to terminate daemon pid %d: %v" , reason , daemon . Pid , err )
return
}
fs . Debugf ( nil , "%s. Terminating daemon pid %d" , reason , daemon . Pid )
} )
}
if err == nil && Opt . DaemonWait > 0 {
handle := atexit . Register ( func ( ) {
killDaemon ( "Got interrupt" )
} )
err = WaitMountReady ( mnt . MountPoint , Opt . DaemonWait )
if err != nil {
killDaemon ( "Daemon timed out" )
}
atexit . Unregister ( handle )
2018-03-02 13:30:04 +00:00
}
2017-06-19 12:44:49 +00:00
if err != nil {
log . Fatalf ( "Fatal error: %v" , err )
}
} ,
}
// Register the command
2019-10-11 15:58:11 +00:00
cmd . Root . AddCommand ( commandDefinition )
2017-06-19 12:44:49 +00:00
// Add flags
2019-10-11 15:55:04 +00:00
cmdFlags := commandDefinition . Flags ( )
2020-07-23 16:17:01 +00:00
AddFlags ( cmdFlags )
2019-10-11 15:55:04 +00:00
vfsflags . AddFlags ( cmdFlags )
2017-10-24 20:06:06 +00:00
2019-10-11 15:58:11 +00:00
return commandDefinition
2017-06-19 12:44:49 +00:00
}
2018-06-26 08:26:34 +00:00
2021-01-03 00:05:52 +00:00
// Mount the remote at mountpoint
2021-08-18 11:07:09 +00:00
func ( m * MountPoint ) Mount ( ) ( daemon * os . Process , err error ) {
2021-01-03 00:05:52 +00:00
if err = m . CheckOverlap ( ) ; err != nil {
2021-08-18 11:07:09 +00:00
return nil , err
2018-06-26 08:26:34 +00:00
}
2021-01-03 00:05:52 +00:00
if err = m . CheckAllowings ( ) ; err != nil {
2021-08-18 11:07:09 +00:00
return nil , err
2018-06-26 08:26:34 +00:00
}
2021-01-03 00:05:52 +00:00
m . SetVolumeName ( m . MountOpt . VolumeName )
2020-07-23 12:08:38 +00:00
2021-07-26 10:44:02 +00:00
// Start background task if --daemon is specified
2021-01-03 00:05:52 +00:00
if m . MountOpt . Daemon {
2021-08-18 11:07:09 +00:00
daemon , err = daemonize . StartDaemon ( os . Args )
if daemon != nil || err != nil {
return daemon , err
2021-01-03 00:05:52 +00:00
}
2020-07-23 16:17:01 +00:00
}
2021-01-03 00:05:52 +00:00
m . VFS = vfs . New ( m . Fs , & m . VFSOpt )
m . ErrChan , m . UnmountFn , err = m . MountFn ( m . VFS , m . MountPoint , & m . MountOpt )
2020-07-23 12:08:38 +00:00
if err != nil {
2021-11-04 10:12:57 +00:00
return nil , fmt . Errorf ( "failed to mount FUSE fs: %w" , err )
2020-07-23 12:08:38 +00:00
}
2021-08-18 11:07:09 +00:00
return nil , nil
2021-01-03 00:05:52 +00:00
}
// Wait for mount end
func ( m * MountPoint ) Wait ( ) error {
2020-07-23 12:08:38 +00:00
// Unmount on exit
2020-11-27 10:50:10 +00:00
var finaliseOnce sync . Once
finalise := func ( ) {
finaliseOnce . Do ( func ( ) {
_ = sysdnotify . Stopping ( )
2021-08-18 11:07:09 +00:00
// Unmount only if directory was mounted by rclone, e.g. don't unmount autofs hooks.
if err := CheckMountReady ( m . MountPoint ) ; err != nil {
fs . Debugf ( m . MountPoint , "Unmounted externally. Just exit now." )
return
}
if err := m . Unmount ( ) ; err != nil {
fs . Errorf ( m . MountPoint , "Failed to unmount: %v" , err )
} else {
fs . Errorf ( m . MountPoint , "Unmounted rclone mount" )
}
2020-11-27 10:50:10 +00:00
} )
}
fnHandle := atexit . Register ( finalise )
2020-07-23 12:08:38 +00:00
defer atexit . Unregister ( fnHandle )
// Notify systemd
2020-09-18 15:37:54 +00:00
if err := sysdnotify . Ready ( ) ; err != nil {
2021-11-04 10:12:57 +00:00
return fmt . Errorf ( "failed to notify systemd: %w" , err )
2020-07-23 12:08:38 +00:00
}
// Reload VFS cache on SIGHUP
sigHup := make ( chan os . Signal , 1 )
2021-01-03 00:05:52 +00:00
NotifyOnSigHup ( sigHup )
var err error
2020-07-23 12:08:38 +00:00
2021-01-03 00:05:52 +00:00
waiting := true
for waiting {
2020-07-23 12:08:38 +00:00
select {
// umount triggered outside the app
2021-01-03 00:05:52 +00:00
case err = <- m . ErrChan :
waiting = false
2020-07-23 12:08:38 +00:00
// user sent SIGHUP to clear the cache
case <- sigHup :
2021-01-03 00:05:52 +00:00
root , err := m . VFS . Root ( )
2020-07-23 12:08:38 +00:00
if err != nil {
2021-01-03 00:05:52 +00:00
fs . Errorf ( m . VFS . Fs ( ) , "Error reading root: %v" , err )
2020-07-23 12:08:38 +00:00
} else {
root . ForgetAll ( )
}
}
}
2020-11-27 10:50:10 +00:00
finalise ( )
2020-07-23 12:08:38 +00:00
if err != nil {
2021-11-04 10:12:57 +00:00
return fmt . Errorf ( "failed to umount FUSE fs: %w" , err )
2020-07-23 12:08:38 +00:00
}
return nil
}
2021-01-03 00:05:52 +00:00
// Unmount the specified mountpoint
func ( m * MountPoint ) Unmount ( ) ( err error ) {
return m . UnmountFn ( )
}