2017-06-19 12:44:49 +00:00
package mountlib
import (
2017-11-09 00:37:27 +00:00
"io"
2017-06-19 12:44:49 +00:00
"log"
2017-11-09 00:37:27 +00:00
"os"
2020-07-23 12:08:38 +00:00
"os/signal"
2019-01-10 14:18:00 +00:00
"path/filepath"
2017-11-16 12:20:53 +00:00
"runtime"
2018-05-03 08:34:07 +00:00
"strings"
2020-07-23 12:08:38 +00:00
"syscall"
2018-03-02 16:39:42 +00:00
"time"
2017-06-19 12:44:49 +00:00
2020-07-23 12:08:38 +00:00
"github.com/okzk/sdnotify"
2017-11-09 00:37:27 +00:00
"github.com/pkg/errors"
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 12:08:38 +00:00
"github.com/rclone/rclone/lib/atexit"
2019-07-28 17:47:38 +00:00
"github.com/rclone/rclone/vfs"
"github.com/rclone/rclone/vfs/vfsflags"
2017-06-19 12:44:49 +00:00
"github.com/spf13/cobra"
)
// Options set by command line flags
var (
2017-10-28 19:01:34 +00:00
DebugFUSE = false
2017-06-19 12:44:49 +00:00
AllowNonEmpty = false
AllowRoot = false
AllowOther = false
DefaultPermissions = false
WritebackCache = false
2018-03-02 13:30:04 +00:00
Daemon = false
2017-06-19 12:44:49 +00:00
MaxReadAhead fs . SizeSuffix = 128 * 1024
2017-11-07 17:09:08 +00:00
ExtraOptions [ ] string
ExtraFlags [ ] string
2018-03-23 22:42:51 +00:00
AttrTimeout = 1 * time . Second // how long the kernel caches attribute for
2018-05-03 08:34:07 +00:00
VolumeName string
2018-07-18 15:21:35 +00:00
NoAppleDouble = true // use noappledouble by default
NoAppleXattr = false // do not use noapplexattr by default
DaemonTimeout time . Duration // OSXFUSE only
2020-02-04 10:21:07 +00:00
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-22 16:58:49 +00:00
MountFn func ( VFS * vfs . VFS , mountpoint string ) ( <- chan error , func ( ) error , error )
2020-04-25 05:03:07 +00:00
)
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 ( ) {
2019-10-12 11:41:36 +00:00
// DaemonTimeout defaults to non zero for macOS
if runtime . GOOS == "darwin" {
2019-06-24 10:54:38 +00:00
DaemonTimeout = 15 * time . Minute
}
}
2017-11-09 00:37:27 +00:00
// Check is folder is empty
func checkMountEmpty ( mountpoint string ) error {
fp , fpErr := os . Open ( mountpoint )
if fpErr != nil {
return errors . Wrap ( fpErr , "Can not open: " + mountpoint )
}
defer fs . CheckClose ( fp , & fpErr )
_ , fpErr = fp . Readdirnames ( 1 )
// directory is not empty
if fpErr != io . EOF {
var e error
var errorMsg = "Directory is not empty: " + mountpoint + " If you want to mount it anyway use: --allow-non-empty option"
if fpErr == nil {
e = errors . New ( errorMsg )
} else {
e = errors . Wrap ( fpErr , errorMsg )
}
return e
}
return nil
}
2019-01-10 14:18:00 +00:00
// Check the root doesn't overlap the mountpoint
func checkMountpointOverlap ( root , mountpoint string ) error {
abs := func ( x string ) string {
if absX , err := filepath . EvalSymlinks ( x ) ; err == nil {
x = absX
}
if absX , err := filepath . Abs ( x ) ; err == nil {
x = absX
}
x = filepath . ToSlash ( x )
if ! strings . HasSuffix ( x , "/" ) {
x += "/"
}
return x
}
rootAbs , mountpointAbs := abs ( root ) , abs ( mountpoint )
if strings . HasPrefix ( rootAbs , mountpointAbs ) || strings . HasPrefix ( mountpointAbs , rootAbs ) {
return errors . Errorf ( "mount point %q and directory to be mounted %q mustn't overlap" , mountpoint , root )
}
return nil
}
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. ` ,
2017-06-19 12:44:49 +00:00
Long : `
rclone ` + commandName + ` allows Linux , FreeBSD , macOS and Windows to
mount any of Rclone ' s cloud storage systems as a file system with
FUSE .
First set up your remote using ` + " ` rclone config ` " + ` . Check it works with ` + " ` rclone ls ` " + ` etc .
2020-05-25 23:26:20 +00:00
You can either run mount in foreground mode or background ( daemon ) mode . Mount runs in
2020-02-10 15:21:22 +00:00
foreground mode by default , use the -- daemon flag to specify background mode mode .
Background mode is only supported on Linux and OSX , you can only run mount in
foreground mode on Windows .
2020-03-31 11:16:03 +00:00
On Linux / macOS / FreeBSD Start the mount like this where ` + " ` / path / to / local / mount ` " + `
is an * * empty * * * * existing * * directory .
2017-06-19 12:44:49 +00:00
rclone ` + commandName + ` remote : path / to / files / path / to / local / mount
2020-03-31 11:16:03 +00:00
Or on Windows like this where ` + " ` X : ` " + ` is an unused drive letter
or use a path to * * non - existent * * directory .
2017-06-19 12:44:49 +00:00
rclone ` + commandName + ` remote : path / to / files X :
2020-03-31 11:16:03 +00:00
rclone ` + commandName + ` remote : path / to / files C : \ path \ to \ nonexistent \ directory
2017-06-19 12:44:49 +00:00
2020-02-10 15:21:22 +00:00
When running in background mode the user will have to stop the mount manually ( specified below ) .
When the program ends while in foreground mode , either via Ctrl + C or receiving
a SIGINT or SIGTERM signal , the mount is automatically stopped .
2017-06-19 12:44:49 +00:00
The umount operation can fail , for example when the mountpoint is busy .
2020-02-10 15:21:22 +00:00
When that happens , it is the user ' s responsibility to stop the mount manually .
Stopping the mount manually :
2017-06-19 12:44:49 +00:00
# Linux
fusermount - u / path / to / local / mount
# OS X
umount / path / to / local / mount
2018-01-18 19:16:21 +00:00
# # # Installing on Windows
2017-07-23 11:13:29 +00:00
To run rclone ` + commandName + ` on Windows , you will need to
download and install [ WinFsp ] ( http : //www.secfs.net/winfsp/).
2020-05-25 08:08:16 +00:00
[ WinFsp ] ( https : //github.com/billziss-gh/winfsp) is an open source
2017-07-23 11:13:29 +00:00
Windows File System Proxy which makes it easy to write user space file
systems for Windows . It provides a FUSE emulation layer which rclone
uses combination with
[ cgofuse ] ( https : //github.com/billziss-gh/cgofuse). Both of these
packages are by Bill Zissimopoulos who was very helpful during the
implementation of rclone ` + commandName + ` for Windows .
2018-01-18 19:16:21 +00:00
# # # # Windows caveats
2017-07-26 20:08:24 +00:00
Note that drives created as Administrator are not visible by other
accounts ( including the account that was elevated as
Administrator ) . So if you start a Windows drive from an Administrative
Command Prompt and then try to access the same drive from Explorer
( which does not run as Administrator ) , you will not be able to see the
new drive .
The easiest way around this is to start the drive from a normal
command prompt . It is also possible to start a drive from the SYSTEM
account ( using [ the WinFsp . Launcher
infrastructure ] ( https : //github.com/billziss-gh/winfsp/wiki/WinFsp-Service-Architecture))
2018-02-11 17:37:47 +00:00
which creates drives accessible for everyone on the system or
alternatively using [ the nssm service manager ] ( https : //nssm.cc/usage).
2017-07-26 20:08:24 +00:00
2020-02-10 14:42:09 +00:00
# # # # Mount as a network drive
By default , rclone will mount the remote as a normal drive . However ,
you can also mount it as a * * Network Drive * * ( or * * Network Share * * , as
mentioned in some places )
Unlike other systems , Windows provides a different filesystem type for
network drives . Windows and other programs treat the network drives
and fixed / removable drives differently : In network drives , many I / O
operations are optimized , as the high latency and low reliability
( compared to a normal drive ) of a network is expected .
Although many people prefer network shares to be mounted as normal
system drives , this might cause some issues , such as programs not
working as expected or freezes and errors while operating with the
mounted remote in Windows Explorer . If you experience any of those ,
consider mounting rclone remotes as network shares , as Windows expects
normal drives to be fast and reliable , while cloud storage is far from
that . See also [ Limitations ] ( # limitations ) section below for more
info
Add "--fuse-flag --VolumePrefix=\server\share" to your "mount"
command , * * replacing "share" with any other name of your choice if you
are mounting more than one remote * * . Otherwise , the mountpoints will
conflict and your mounted filesystems will overlap .
[ Read more about drive mapping ] ( https : //en.wikipedia.org/wiki/Drive_mapping)
2018-01-18 19:16:21 +00:00
# # # Limitations
2017-06-19 12:44:49 +00:00
2018-02-10 09:28:20 +00:00
Without the use of "--vfs-cache-mode" this can only write files
sequentially , it can only seek when reading . This means that many
applications won ' t work with their files on an rclone mount without
"--vfs-cache-mode writes" or "--vfs-cache-mode full" . See the [ File
Caching ] ( # file - caching ) section for more info .
2017-06-19 12:44:49 +00:00
The bucket based remotes ( eg Swift , S3 , Google Compute Storage , B2 ,
2019-08-16 14:36:32 +00:00
Hubic ) do not support the concept of empty directories , so empty
2017-06-19 12:44:49 +00:00
directories will have a tendency to disappear once they fall out of
the directory cache .
Only supported on Linux , FreeBSD , OS X and Windows at the moment .
2018-01-18 19:16:21 +00:00
# # # rclone ` + commandName + ` vs rclone sync / copy
2017-06-19 12:44:49 +00:00
File systems expect things to be 100 % reliable , whereas cloud storage
systems are a long way from 100 % reliable . The rclone sync / copy
commands cope with this with lots of retries . However rclone ` + commandName + `
can ' t use retries in the same way without making local copies of the
2018-10-04 10:37:42 +00:00
uploads . Look at the [ file caching ] ( # file - caching )
2018-11-14 16:11:58 +00:00
for solutions to make ` + commandName + ` more reliable .
2017-06-19 12:44:49 +00:00
2018-03-02 16:39:42 +00:00
# # # Attribute caching
You can use the flag -- attr - timeout to set the time the kernel caches
the attributes ( size , modification time etc ) for directory entries .
2018-03-23 22:42:51 +00:00
The default is "1s" which caches files just long enough to avoid
too many callbacks to rclone from the kernel .
In theory 0 s should be the correct value for filesystems which can
change outside the control of the kernel . However this causes quite a
few problems such as
2019-07-28 17:47:38 +00:00
[ rclone using too much memory ] ( https : //github.com/rclone/rclone/issues/2157),
2018-03-23 22:42:51 +00:00
[ rclone not serving files to samba ] ( https : //forum.rclone.org/t/rclone-1-39-vs-1-40-mount-issue/5112)
2019-07-28 17:47:38 +00:00
and [ excessive time listing directories ] ( https : //github.com/rclone/rclone/issues/2095#issuecomment-371141147).
2018-03-23 22:42:51 +00:00
The kernel can cache the info about a file for the time given by
"--attr-timeout" . You may see corruption if the remote file changes
length during this window . It will show up as either a truncated file
or a file with garbage on the end . With "--attr-timeout 1s" this is
very unlikely but not impossible . The higher you set "--attr-timeout"
the more likely it is . The default setting of "1s" is the lowest
setting which mitigates the problems above .
If you set it higher ( ' 10 s ' or ' 1 m ' say ) then the kernel will call
back to rclone less often making it more efficient , however there is
more chance of the corruption issue above .
If files don ' t change on the remote outside of the control of rclone
then there is no chance of corruption .
2018-03-02 16:39:42 +00:00
This is the same as setting the attr_timeout option in mount . fuse .
2018-01-18 19:16:21 +00:00
# # # Filters
2017-06-19 12:44:49 +00:00
Note that all the rclone filters can be used to select a subset of the
files to be visible in the mount .
2018-01-18 19:16:21 +00:00
# # # systemd
2017-11-19 22:03:49 +00:00
When running rclone ` + commandName + ` as a systemd service , it is possible
2018-01-18 19:16:21 +00:00
to use Type = notify . In this case the service will enter the started state
2017-11-19 22:03:49 +00:00
after the mountpoint has been successfully set up .
Units having the rclone ` + commandName + ` service specified as a requirement
will see all files and folders immediately in this mode .
2018-02-18 14:18:12 +00:00
# # # chunked reading # # #
-- vfs - read - chunk - size will enable reading the source objects in parts .
This can reduce the used download quota for some remotes by requesting only chunks
from the remote that are actually read at the cost of an increased number of requests .
When -- vfs - read - chunk - size - limit is also specified and greater than -- vfs - read - chunk - size ,
the chunk size for each open file will get doubled for each chunk read , until the
specified value is reached . A value of - 1 will disable the limit and the chunk size will
grow indefinitely .
With -- vfs - read - chunk - size 100 M and -- vfs - read - chunk - size - limit 0 the following
parts will be downloaded : 0 - 100 M , 100 M - 200 M , 200 M - 300 M , 300 M - 400 M and so on .
When -- vfs - read - chunk - size - limit 500 M is specified , the result would be
0 - 100 M , 100 M - 300 M , 300 M - 700 M , 700 M - 1200 M , 1200 M - 1700 M and so on .
Chunked reading will only work with -- vfs - cache - mode < full , as the file will always
be copied to the vfs cache before opening with -- vfs - cache - mode full .
2017-11-13 17:54:21 +00:00
` + 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
if Daemon {
config . PassConfigKeyForDaemonization = true
}
2019-01-10 14:18:00 +00:00
mountpoint := args [ 1 ]
2018-05-07 16:58:16 +00:00
fdst := cmd . NewFsDir ( args )
2019-01-10 14:18:00 +00:00
if fdst . Name ( ) == "" || fdst . Name ( ) == "local" {
err := checkMountpointOverlap ( fdst . Root ( ) , mountpoint )
if err != nil {
log . Fatalf ( "Fatal error: %v" , err )
}
}
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
}
2017-11-16 12:20:53 +00:00
// Skip checkMountEmpty if --allow-non-empty flag is used or if
// the Operating System is Windows
if ! AllowNonEmpty && runtime . GOOS != "windows" {
2019-01-10 14:18:00 +00:00
err := checkMountEmpty ( mountpoint )
2017-11-09 00:37:27 +00:00
if err != nil {
log . Fatalf ( "Fatal error: %v" , err )
}
2020-03-31 11:16:03 +00:00
} else if AllowNonEmpty && runtime . GOOS == "windows" {
fs . Logf ( nil , "--allow-non-empty flag does nothing on Windows" )
2017-11-09 00:37:27 +00:00
}
2018-05-03 08:34:07 +00:00
// Work out the volume name, removing special
// characters from it if necessary
if VolumeName == "" {
VolumeName = fdst . Name ( ) + ":" + fdst . Root ( )
}
VolumeName = strings . Replace ( VolumeName , ":" , " " , - 1 )
VolumeName = strings . Replace ( VolumeName , "/" , " " , - 1 )
VolumeName = strings . TrimSpace ( VolumeName )
2020-03-03 18:59:47 +00:00
if runtime . GOOS == "windows" && len ( VolumeName ) > 32 {
VolumeName = VolumeName [ : 32 ]
}
2018-05-03 08:34:07 +00:00
2018-03-02 13:30:04 +00:00
// Start background task if --background is specified
if Daemon {
daemonized := startBackgroundMode ( )
if daemonized {
return
}
}
2020-07-22 16:58:49 +00:00
VFS := vfs . New ( fdst , & vfsflags . Opt )
2020-07-23 12:08:38 +00:00
err := Mount ( VFS , mountpoint , mount )
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 ( )
flags . BoolVarP ( cmdFlags , & DebugFUSE , "debug-fuse" , "" , DebugFUSE , "Debug the FUSE internals - needs -v." )
2017-06-19 12:44:49 +00:00
// mount options
2020-03-31 11:16:03 +00:00
flags . BoolVarP ( cmdFlags , & AllowNonEmpty , "allow-non-empty" , "" , AllowNonEmpty , "Allow mounting over a non-empty directory (not Windows)." )
2019-10-11 15:55:04 +00:00
flags . BoolVarP ( cmdFlags , & AllowRoot , "allow-root" , "" , AllowRoot , "Allow access to root user." )
flags . BoolVarP ( cmdFlags , & AllowOther , "allow-other" , "" , AllowOther , "Allow access to other users." )
flags . BoolVarP ( cmdFlags , & DefaultPermissions , "default-permissions" , "" , DefaultPermissions , "Makes kernel enforce access control based on the file mode." )
flags . BoolVarP ( cmdFlags , & WritebackCache , "write-back-cache" , "" , WritebackCache , "Makes kernel buffer writes before sending them to rclone. Without this, writethrough caching is used." )
flags . FVarP ( cmdFlags , & MaxReadAhead , "max-read-ahead" , "" , "The number of bytes that can be prefetched for sequential reads." )
flags . DurationVarP ( cmdFlags , & AttrTimeout , "attr-timeout" , "" , AttrTimeout , "Time for which file/directory attributes are cached." )
flags . StringArrayVarP ( cmdFlags , & ExtraOptions , "option" , "o" , [ ] string { } , "Option for libfuse/WinFsp. Repeat if required." )
flags . StringArrayVarP ( cmdFlags , & ExtraFlags , "fuse-flag" , "" , [ ] string { } , "Flags or arguments to be passed direct to libfuse/WinFsp. Repeat if required." )
flags . BoolVarP ( cmdFlags , & Daemon , "daemon" , "" , Daemon , "Run mount as a daemon (background mode)." )
flags . StringVarP ( cmdFlags , & VolumeName , "volname" , "" , VolumeName , "Set the volume name (not supported by all OSes)." )
flags . DurationVarP ( cmdFlags , & DaemonTimeout , "daemon-timeout" , "" , DaemonTimeout , "Time limit for rclone to respond to kernel (not supported by all OSes)." )
2020-02-04 10:21:07 +00:00
flags . BoolVarP ( cmdFlags , & AsyncRead , "async-read" , "" , AsyncRead , "Use asynchronous reads." )
2017-06-19 12:44:49 +00:00
2018-05-03 08:45:24 +00:00
if runtime . GOOS == "darwin" {
2019-10-11 15:55:04 +00:00
flags . BoolVarP ( cmdFlags , & NoAppleDouble , "noappledouble" , "" , NoAppleDouble , "Sets the OSXFUSE option noappledouble." )
flags . BoolVarP ( cmdFlags , & NoAppleXattr , "noapplexattr" , "" , NoAppleXattr , "Sets the OSXFUSE option noapplexattr." )
2018-05-03 08:45:24 +00:00
}
2017-10-24 20:06:06 +00:00
// Add in the generic flags
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
2020-05-25 06:05:53 +00:00
// ClipBlocks clips the blocks pointed to the OS max
2018-06-26 08:26:34 +00:00
func ClipBlocks ( b * uint64 ) {
var max uint64
switch runtime . GOOS {
case "windows" :
2019-01-25 17:31:56 +00:00
if runtime . GOARCH == "386" {
max = ( 1 << 32 ) - 1
} else {
max = ( 1 << 43 ) - 1
}
2018-06-26 08:26:34 +00:00
case "darwin" :
// OSX FUSE only supports 32 bit number of blocks
// https://github.com/osxfuse/osxfuse/issues/396
max = ( 1 << 32 ) - 1
default :
// no clipping
return
}
if * b > max {
* b = max
}
}
2020-07-23 12:08:38 +00:00
// Mount mounts the remote at mountpoint.
//
// If noModTime is set then it
func Mount ( VFS * vfs . VFS , mountpoint string , mount MountFn ) error {
// Mount it
errChan , unmount , err := mount ( VFS , mountpoint )
if err != nil {
return errors . Wrap ( err , "failed to mount FUSE fs" )
}
// Unmount on exit
fnHandle := atexit . Register ( func ( ) {
_ = unmount ( )
_ = sdnotify . Stopping ( )
} )
defer atexit . Unregister ( fnHandle )
// Notify systemd
if err := sdnotify . Ready ( ) ; err != nil && err != sdnotify . ErrSdNotifyNoSocket {
return errors . Wrap ( err , "failed to notify systemd" )
}
// Reload VFS cache on SIGHUP
sigHup := make ( chan os . Signal , 1 )
signal . Notify ( sigHup , syscall . SIGHUP )
waitloop :
for {
select {
// umount triggered outside the app
case err = <- errChan :
break waitloop
// user sent SIGHUP to clear the cache
case <- sigHup :
root , err := VFS . Root ( )
if err != nil {
fs . Errorf ( VFS . Fs ( ) , "Error reading root: %v" , err )
} else {
root . ForgetAll ( )
}
}
}
_ = unmount ( )
_ = sdnotify . Stopping ( )
if err != nil {
return errors . Wrap ( err , "failed to umount FUSE fs" )
}
return nil
}