2016-07-18 00:03:23 +02:00
// Package mount implents a FUSE mounting system for rclone remotes.
// +build linux darwin freebsd
package mount
import (
2016-09-09 09:39:19 +02:00
"log"
"os"
2016-10-18 15:44:16 +02:00
"time"
2016-09-09 09:39:19 +02:00
2016-07-18 00:03:23 +02:00
"bazil.org/fuse"
"github.com/ncw/rclone/cmd"
"github.com/ncw/rclone/fs"
"github.com/pkg/errors"
"github.com/spf13/cobra"
2016-09-09 09:39:19 +02:00
"golang.org/x/sys/unix"
2016-07-18 00:03:23 +02:00
)
// Globals
var (
2016-10-18 15:44:16 +02:00
noModTime = false
debugFUSE = false
noSeek = false
dirCacheTime = 5 * 60 * time . Second
2016-09-09 09:39:19 +02:00
// mount options
readOnly = false
allowNonEmpty = false
allowRoot = false
allowOther = false
defaultPermissions = false
writebackCache = false
maxReadAhead fs . SizeSuffix = 128 * 1024
umask = 0
uid = uint32 ( unix . Geteuid ( ) )
gid = uint32 ( unix . Getegid ( ) )
// foreground = false
// default permissions for directories - modified by umask in Mount
dirPerms = os . FileMode ( 0777 )
filePerms = os . FileMode ( 0666 )
2016-07-18 00:03:23 +02:00
)
func init ( ) {
2016-09-09 09:39:19 +02:00
umask = unix . Umask ( 0 ) // read the umask
unix . Umask ( umask ) // set it back to what it was
2016-07-18 00:03:23 +02:00
cmd . Root . AddCommand ( mountCmd )
2016-09-09 09:39:19 +02:00
mountCmd . Flags ( ) . BoolVarP ( & noModTime , "no-modtime" , "" , noModTime , "Don't read the modification time (can speed things up)." )
mountCmd . Flags ( ) . BoolVarP ( & debugFUSE , "debug-fuse" , "" , debugFUSE , "Debug the FUSE internals - needs -v." )
2016-10-06 14:37:45 +02:00
mountCmd . Flags ( ) . BoolVarP ( & noSeek , "no-seek" , "" , noSeek , "Don't allow seeking in files." )
2016-10-18 15:44:16 +02:00
mountCmd . Flags ( ) . DurationVarP ( & dirCacheTime , "dir-cache-time" , "" , dirCacheTime , "Time to cache directory entries for." )
2016-09-09 09:39:19 +02:00
// mount options
mountCmd . Flags ( ) . BoolVarP ( & readOnly , "read-only" , "" , readOnly , "Mount read-only." )
mountCmd . Flags ( ) . BoolVarP ( & allowNonEmpty , "allow-non-empty" , "" , allowNonEmpty , "Allow mounting over a non-empty directory." )
mountCmd . Flags ( ) . BoolVarP ( & allowRoot , "allow-root" , "" , allowRoot , "Allow access to root user." )
mountCmd . Flags ( ) . BoolVarP ( & allowOther , "allow-other" , "" , allowOther , "Allow access to other users." )
mountCmd . Flags ( ) . BoolVarP ( & defaultPermissions , "default-permissions" , "" , defaultPermissions , "Makes kernel enforce access control based on the file mode." )
mountCmd . Flags ( ) . BoolVarP ( & writebackCache , "write-back-cache" , "" , writebackCache , "Makes kernel buffer writes before sending them to rclone. Without this, writethrough caching is used." )
mountCmd . Flags ( ) . VarP ( & maxReadAhead , "max-read-ahead" , "" , "The number of bytes that can be prefetched for sequential reads." )
mountCmd . Flags ( ) . IntVarP ( & umask , "umask" , "" , umask , "Override the permission bits set by the filesystem." )
mountCmd . Flags ( ) . Uint32VarP ( & uid , "uid" , "" , uid , "Override the uid field set by the filesystem." )
mountCmd . Flags ( ) . Uint32VarP ( & gid , "gid" , "" , gid , "Override the gid field set by the filesystem." )
//mountCmd.Flags().BoolVarP(&foreground, "foreground", "", foreground, "Do not detach.")
2016-07-18 00:03:23 +02:00
}
var mountCmd = & cobra . Command {
Use : "mount remote:path /path/to/mountpoint" ,
Short : ` Mount the remote as a mountpoint. **EXPERIMENTAL** ` ,
Long : `
2016-08-22 17:46:08 +02:00
rclone mount allows Linux , FreeBSD and macOS to mount any of Rclone ' s
cloud storage systems as a file system with FUSE .
2016-07-18 00:03:23 +02:00
This is * * EXPERIMENTAL * * - use with care .
First set up your remote using ` + " ` rclone config ` " + ` . Check it works with ` + " ` rclone ls ` " + ` etc .
Start the mount like this
rclone mount remote : path / to / files / path / to / local / mount &
Stop the mount with
fusermount - u / path / to / local / mount
Or with OS X
umount - u / path / to / local / mount
# # # Limitations # # #
2016-09-10 23:25:26 +02:00
This can only write files seqentially , it can only seek when reading .
2016-07-18 00:03:23 +02:00
2016-09-10 23:25:26 +02:00
Rclone mount inherits rclone ' s directory handling . In rclone ' s world
2016-07-18 00:03:23 +02:00
directories don ' t really exist . This means that empty directories
will have a tendency to disappear once they fall out of the directory
cache .
The bucket based FSes ( eg swift , s3 , google compute storage , b2 ) won ' t
work from the root - you will need to specify a bucket , or a path
within the bucket . So ` + " ` swift : ` " + ` won ' t work whereas ` + " ` swift : bucket ` " + ` will
as will ` + " ` swift : bucket / path ` " + ` .
2016-08-22 17:46:08 +02:00
Only supported on Linux , FreeBSD and OS X at the moment .
2016-07-18 00:03:23 +02:00
# # # rclone mount vs rclone sync / copy # #
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 mount
can ' t use retries in the same way without making local copies of the
uploads . This might happen in the future , but for the moment rclone
mount won ' t do that , so will be less reliable than the rclone command .
# # # Bugs # # #
* All the remotes should work for read , but some may not for write
2016-10-23 22:46:48 +02:00
* those which need to know the size in advance won ' t - eg B2 , Amazon Drive
2016-07-18 00:03:23 +02:00
* maybe should pass in size as - 1 to mean work it out
2016-10-23 22:46:48 +02:00
* Or put in an an upload cache to cache the files on disk first
2016-07-18 00:03:23 +02:00
# # # TODO # # #
* Check hashes on upload / download
* Preserve timestamps
* Move directories
` ,
2016-09-09 09:39:19 +02:00
Run : func ( command * cobra . Command , args [ ] string ) {
2016-07-18 00:03:23 +02:00
cmd . CheckArgs ( 2 , 2 , command , args )
fdst := cmd . NewFsDst ( args )
2016-09-09 09:39:19 +02:00
err := Mount ( fdst , args [ 1 ] )
if err != nil {
log . Fatalf ( "Fatal error: %v" , err )
}
2016-07-18 00:03:23 +02:00
} ,
}
// Mount mounts the remote at mountpoint.
//
// If noModTime is set then it
func Mount ( f fs . Fs , mountpoint string ) error {
if debugFUSE {
fuse . Debug = func ( msg interface { } ) {
fs . Debug ( "fuse" , "%v" , msg )
}
}
2016-09-09 09:39:19 +02:00
// Set permissions
dirPerms = 0777 &^ os . FileMode ( umask )
filePerms = 0666 &^ os . FileMode ( umask )
2016-07-18 00:03:23 +02:00
// Mount it
errChan , err := mount ( f , mountpoint )
if err != nil {
return errors . Wrap ( err , "failed to mount FUSE fs" )
}
// Wait for umount
err = <- errChan
if err != nil {
return errors . Wrap ( err , "failed to umount FUSE fs" )
}
return nil
}