rclone/fs/fs.go

350 lines
8.7 KiB
Go
Raw Normal View History

// Generic file system interface for rclone object storage systems
2013-06-28 03:13:07 +08:00
package fs
import (
"fmt"
"io"
"log"
"path/filepath"
2013-06-28 03:13:07 +08:00
"regexp"
"time"
)
// Constants
const (
// User agent for Fs which can set it
UserAgent = "rclone/" + Version
// Very large precision value to show mod time isn't supported
ModTimeNotSupported = 100 * 365 * 24 * time.Hour
)
2013-06-28 03:13:07 +08:00
// Globals
var (
// Filesystem registry
fsRegistry []*FsInfo
// Error returned by NewFs if not found in config file
NotFoundInConfigFile = fmt.Errorf("Didn't find section in config file")
ErrorCantCopy = fmt.Errorf("Can't copy object - incompatible remotes")
ErrorCantMove = fmt.Errorf("Can't copy object - incompatible remotes")
ErrorCantDirMove = fmt.Errorf("Can't copy directory - incompatible remotes")
ErrorDirExists = fmt.Errorf("Can't copy directory - destination already exists")
2013-06-28 03:13:07 +08:00
)
// Filesystem info
type FsInfo struct {
// Name of this fs
Name string
// Create a new file system. If root refers to an existing
// object, then it should return a Fs which only returns that
// object.
NewFs func(name string, root string) (Fs, error)
// Function to call to help with config
Config func(string)
// Options for the Fs configuration
Options []Option
2013-06-28 03:13:07 +08:00
}
// An options for a Fs
type Option struct {
Name string
Help string
Optional bool
Examples []OptionExample
}
// An example for an option
type OptionExample struct {
Value string
Help string
}
2013-06-28 03:13:07 +08:00
// Register a filesystem
//
// Fs modules should use this in an init() function
func Register(info *FsInfo) {
fsRegistry = append(fsRegistry, info)
2013-06-28 03:13:07 +08:00
}
// A Filesystem, describes the local filesystem and the remote object store
type Fs interface {
// The name of the remote (as passed into NewFs)
Name() string
2015-09-02 03:45:27 +08:00
// The root of the remote (as passed into NewFs)
Root() string
2013-01-19 02:54:19 +08:00
// String returns a description of the FS
String() string
2013-01-19 02:54:19 +08:00
// List the Fs into a channel
2013-06-28 15:57:32 +08:00
List() ObjectsChan
2013-01-19 02:54:19 +08:00
2013-01-24 06:43:20 +08:00
// List the Fs directories/buckets/containers into a channel
2013-06-28 15:57:32 +08:00
ListDir() DirChan
2013-01-24 06:43:20 +08:00
2013-06-28 15:57:32 +08:00
// Find the Object at remote. Returns nil if can't be found
NewFsObject(remote string) Object
2013-01-19 02:54:19 +08:00
// Put in to the remote path with the modTime given of the given size
//
// May create the object even if it returns an error - if so
// will return the object and the error, otherwise will return
// nil and the error
2013-06-28 15:57:32 +08:00
Put(in io.Reader, remote string, modTime time.Time, size int64) (Object, error)
2013-01-19 02:54:19 +08:00
// Make the directory (container, bucket)
//
// Shouldn't return an error if it already exists
Mkdir() error
2013-01-19 02:54:19 +08:00
// Remove the directory (container, bucket) if empty
//
// Return an error if it doesn't exist or isn't empty
Rmdir() error
// Precision of the ModTimes in this Fs
Precision() time.Duration
}
// A filesystem like object which can either be a remote object or a
// local file/directory
2013-06-28 15:57:32 +08:00
type Object interface {
// String returns a description of the Object
String() string
// Fs returns the Fs that this object is part of
Fs() Fs
2013-01-19 02:54:19 +08:00
// Remote returns the remote path
Remote() string
2013-01-19 02:54:19 +08:00
// Md5sum returns the md5 checksum of the file
// If no Md5sum is available it returns ""
Md5sum() (string, error)
2013-01-19 02:54:19 +08:00
// ModTime returns the modification date of the file
// It should return a best guess if one isn't available
ModTime() time.Time
2013-01-19 02:54:19 +08:00
// SetModTime sets the metadata on the object to set the modification date
SetModTime(time.Time)
2013-01-19 02:54:19 +08:00
// Size returns the size of the file
Size() int64
2013-01-19 02:54:19 +08:00
// Open opens the file for read. Call Close() on the returned io.ReadCloser
Open() (io.ReadCloser, error)
2013-01-19 02:54:19 +08:00
// Update in to the object with the modTime given of the given size
Update(in io.Reader, modTime time.Time, size int64) error
2013-01-19 02:54:19 +08:00
// Storable says whether this object can be stored
Storable() bool
2013-01-19 02:54:19 +08:00
// Removes this object
Remove() error
}
// Optional interfaces
type Purger interface {
// Purge all files in the root and the root directory
//
// Implement this if you have a way of deleting all the files
// quicker than just running Remove() on the result of List()
//
// Return an error if it doesn't exist
Purge() error
}
type Copier interface {
// Copy src to this remote using server side copy operations.
//
// This is stored with the remote path given
//
// It returns the destination Object and a possible error
//
// Will only be called if src.Fs().Name() == f.Name()
//
// If it isn't possible then return fs.ErrorCantCopy
Copy(src Object, remote string) (Object, error)
}
type Mover interface {
// Move src to this remote using server side move operations.
//
// This is stored with the remote path given
//
// It returns the destination Object and a possible error
//
// Will only be called if src.Fs().Name() == f.Name()
//
// If it isn't possible then return fs.ErrorCantMove
Move(src Object, remote string) (Object, error)
}
type DirMover interface {
// Move src to this remote using server side move operations.
//
// Will only be called if src.Fs().Name() == f.Name()
//
// If it isn't possible then return fs.ErrorCantDirMove
//
// If destination exists then return fs.ErrorDirExists
DirMove(src Fs) error
}
// An optional interface for error as to whether the operation should be retried
//
// This should be returned from Update or Put methods as required
type Retry interface {
error
Retry() bool
}
// A type of error
type retryError string
// Error interface
func (r retryError) Error() string {
return string(r)
}
// Retry interface
func (r retryError) Retry() bool {
return true
}
// Check interface
var _ Retry = retryError("")
// RetryErrorf makes an error which indicates it would like to be retried
func RetryErrorf(format string, a ...interface{}) error {
return retryError(fmt.Sprintf(format, a...))
}
// PlainRetryError is an error wrapped so it will retry
type plainRetryError struct {
error
}
// Retry interface
func (_ plainRetryError) Retry() bool {
return true
}
// Check interface
var _ Retry = plainRetryError{(error)(nil)}
// RetryError makes an error which indicates it would like to be retried
func RetryError(err error) error {
return plainRetryError{err}
}
2013-06-28 15:57:32 +08:00
// A channel of Objects
type ObjectsChan chan Object
2013-06-28 15:57:32 +08:00
// A slice of Objects
type Objects []Object
// A pair of Objects
type ObjectPair struct {
src, dst Object
}
// A channel of ObjectPair
type ObjectPairChan chan ObjectPair
2013-01-24 06:43:20 +08:00
// A structure of directory/container/bucket lists
2013-06-28 15:57:32 +08:00
type Dir struct {
2013-01-24 06:43:20 +08:00
Name string // name of the directory
When time.Time // modification or creation time - IsZero for unknown
Bytes int64 // size of directory and contents -1 for unknown
Count int64 // number of objects -1 for unknown
}
2013-06-28 15:57:32 +08:00
// A channel of Dir objects
type DirChan chan *Dir
2013-01-24 06:43:20 +08:00
// Finds a FsInfo object for the name passed in
//
// Services are looked up in the config file
func Find(name string) (*FsInfo, error) {
for _, item := range fsRegistry {
if item.Name == name {
return item, nil
}
}
return nil, fmt.Errorf("Didn't find filing system for %q", name)
}
// Pattern to match an rclone url
var matcher = regexp.MustCompile(`^([\w_-]+):(.*)$`)
// NewFs makes a new Fs object from the path
//
// The path is of the form remote:path
//
// Remotes are looked up in the config file. If the remote isn't
// found then NotFoundInConfigFile will be returned.
//
// On Windows avoid single character remote names as they can be mixed
// up with drive letters.
func NewFs(path string) (Fs, error) {
parts := matcher.FindStringSubmatch(path)
fsName, configName, fsPath := "local", "local", path
if parts != nil && !isDriveLetter(parts[1]) {
configName, fsPath = parts[1], parts[2]
var err error
fsName, err = ConfigFile.GetValue(configName, "type")
if err != nil {
return nil, NotFoundInConfigFile
2013-06-28 03:13:07 +08:00
}
}
fs, err := Find(fsName)
if err != nil {
return nil, err
}
// change native directory separators to / if there are any
fsPath = filepath.ToSlash(fsPath)
return fs.NewFs(configName, fsPath)
}
// Outputs log for object
func OutputLog(o interface{}, text string, args ...interface{}) {
description := ""
2015-03-09 16:11:38 +08:00
if o != nil {
description = fmt.Sprintf("%v: ", o)
}
out := fmt.Sprintf(text, args...)
log.Print(description + out)
}
// Write debuging output for this Object or Fs
func Debug(o interface{}, text string, args ...interface{}) {
2013-06-28 03:13:07 +08:00
if Config.Verbose {
OutputLog(o, text, args...)
}
}
// Write log output for this Object or Fs
func Log(o interface{}, text string, args ...interface{}) {
2013-06-28 03:13:07 +08:00
if !Config.Quiet {
OutputLog(o, text, args...)
}
}
// Write error log output for this Object or Fs
// Unconditionally logs a message regardless of Config.Quiet or Config.Verbose
func ErrorLog(o interface{}, text string, args ...interface{}) {
OutputLog(o, text, args...)
}
// checkClose is a utility function used to check the return from
// Close in a defer statement.
func checkClose(c io.Closer, err *error) {
cerr := c.Close()
if *err == nil {
*err = cerr
}
}