From 0159da9f37dedb7a6b5430e092067bd35318e2c7 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sun, 6 Jul 2014 09:57:02 +0100 Subject: [PATCH 01/10] dropbox: graphics for the Dropbox app (used in auth process) --- graphics/rclone-256x256.png | Bin 0 -> 52437 bytes graphics/rclone-64x64.png | Bin 0 -> 7865 bytes 2 files changed, 0 insertions(+), 0 deletions(-) create mode 100644 graphics/rclone-256x256.png create mode 100644 graphics/rclone-64x64.png diff --git a/graphics/rclone-256x256.png b/graphics/rclone-256x256.png new file mode 100644 index 0000000000000000000000000000000000000000..45556aea27533549a7853d91e719e187e683a770 GIT binary patch literal 52437 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%0AXk` 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 Date: Tue, 8 Jul 2014 21:59:30 +0100 Subject: [PATCH 02/10] dropbox: Initial support of full Fs interface Still missing metadata support (eg SetModTime) --- dropbox/dropbox.go | 449 +++++++++++++++++++++++++++++++++++++++ rclone.go | 1 + rclonetest/rclonetest.go | 1 + 3 files changed, 451 insertions(+) create mode 100644 dropbox/dropbox.go diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go new file mode 100644 index 000000000..44ee7a556 --- /dev/null +++ b/dropbox/dropbox.go @@ -0,0 +1,449 @@ +// Dropbox interface +package dropbox + +/* +Limitations of dropbox + +File system is case insensitive! + +Can only have 25,000 objects in a directory + +/delta might be more efficient than recursing with Metadata + +Setting metadata is problematic! Might have to use a database + +Md5sum has to download the file +*/ + +import ( + "crypto/md5" + "errors" + "fmt" + "io" + "log" + "strings" + "time" + + "github.com/ncw/rclone/fs" + "github.com/stacktic/dropbox" +) + +// Constants +const ( + rcloneAppKey = "5jcck7diasz0rqy" + rcloneAppSecret = "1n9m04y2zx7bf26" + uploadChunkSize = 64 * 1024 // chunk size for upload + metadataLimit = dropbox.MetadataLimitDefault // max items to fetch at once +) + +// Register with Fs +func init() { + fs.Register(&fs.FsInfo{ + Name: "dropbox", + NewFs: NewFs, + Config: Config, + Options: []fs.Option{{ + Name: "app_key", + Help: "Dropbox App Key - leave blank to use rclone's.", + }, { + Name: "app_secret", + Help: "Dropbox App Secret - leave blank to use rclone's.", + }}, + }) +} + +// Configuration helper - called after the user has put in the defaults +func Config(name string) { + // See if already have a token + token := fs.ConfigFile.MustValue(name, "token") + if token != "" { + fmt.Printf("Already have a dropbox token - refresh?\n") + if !fs.Confirm() { + return + } + } + + // Get a dropbox + db := newDropbox(name) + + // This method will ask the user to visit an URL and paste the generated code. + if err := db.Auth(); err != nil { + log.Fatalf("Failed to authorize: %v", err) + } + + // Get the token + token = db.AccessToken() + + // Stuff it in the config file if it has changed + old := fs.ConfigFile.MustValue(name, "token") + if token != old { + fs.ConfigFile.SetValue(name, "token", token) + fs.SaveConfig() + } +} + +// FsDropbox represents a remote dropbox server +type FsDropbox struct { + db *dropbox.Dropbox // the connection to the dropbox server + root string // the path we are working on + slashRoot string // root with "/" prefix and postix +} + +// FsObjectDropbox describes a dropbox object +type FsObjectDropbox struct { + dropbox *FsDropbox // what this object is part of + remote string // The remote path + md5sum string // md5sum of the object + bytes int64 // size of the object + modTime time.Time // time it was last modified +} + +// ------------------------------------------------------------ + +// String converts this FsDropbox to a string +func (f *FsDropbox) String() string { + return fmt.Sprintf("Dropbox root '%s'", f.root) +} + +// parseParse parses a dropbox 'url' +func parseDropboxPath(path string) (root string, err error) { + root = strings.Trim(path, "/") + return +} + +// Makes a new dropbox from the config +func newDropbox(name string) *dropbox.Dropbox { + db := dropbox.NewDropbox() + + appKey := fs.ConfigFile.MustValue(name, "app_key") + if appKey == "" { + appKey = rcloneAppKey + } + appSecret := fs.ConfigFile.MustValue(name, "app_secret") + if appSecret == "" { + appSecret = rcloneAppSecret + } + + db.SetAppInfo(appKey, appSecret) + + return db +} + +// NewFs contstructs an FsDropbox from the path, container:path +func NewFs(name, path string) (fs.Fs, error) { + db := newDropbox(name) + + root, err := parseDropboxPath(path) + if err != nil { + return nil, err + } + slashRoot := "/" + root + if root != "" { + slashRoot += "/" + } + f := &FsDropbox{ + root: root, + slashRoot: slashRoot, + db: db, + } + + // Read the token from the config file + token := fs.ConfigFile.MustValue(name, "token") + + // Authorize the client + db.SetAccessToken(token) + + return f, nil +} + +// Return an FsObject from a path +func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.Object, error) { + fs := &FsObjectDropbox{ + dropbox: f, + remote: remote, + } + if info != nil { + fs.setMetaData(info) + } else { + err := fs.readMetaData() // reads info and meta, returning an error + if err != nil { + // logged already fs.Debug("Failed to read info: %s", err) + return nil, err + } + } + return fs, nil +} + +// Return an FsObject from a path +// +// May return nil if an error occurred +func (f *FsDropbox) NewFsObjectWithInfo(remote string, info *dropbox.Entry) fs.Object { + fs, _ := f.newFsObjectWithInfo(remote, info) + // Errors have already been logged + return fs +} + +// Return an FsObject from a path +// +// May return nil if an error occurred +func (f *FsDropbox) NewFsObject(remote string) fs.Object { + return f.NewFsObjectWithInfo(remote, nil) +} + +// Strips the root off entry and returns it +func (f *FsDropbox) stripRoot(entry *dropbox.Entry) string { + path := entry.Path + if strings.HasPrefix(path, f.slashRoot) { + path = path[len(f.slashRoot):] + } + return path +} + +// Walk the path returning a channel of FsObjects +// +// FIXME could do this in parallel but needs to be limited to Checkers +func (f *FsDropbox) list(path string, out fs.ObjectsChan) { + entry, err := f.db.Metadata(f.slashRoot+path, true, false, "", "", metadataLimit) + if err != nil { + fs.Stats.Error() + fs.Log(f, "Couldn't list %q: %s", path, err) + } else { + for i := range entry.Contents { + entry := &entry.Contents[i] + path = f.stripRoot(entry) + if entry.IsDir { + f.list(path, out) + } else { + out <- f.NewFsObjectWithInfo(path, entry) + } + } + } +} + +// Walk the path returning a channel of FsObjects +func (f *FsDropbox) List() fs.ObjectsChan { + out := make(fs.ObjectsChan, fs.Config.Checkers) + go func() { + defer close(out) + f.list("", out) + }() + return out +} + +// Walk the path returning a channel of FsObjects +func (f *FsDropbox) ListDir() fs.DirChan { + out := make(fs.DirChan, fs.Config.Checkers) + go func() { + defer close(out) + entry, err := f.db.Metadata(f.root, true, false, "", "", metadataLimit) + if err != nil { + fs.Stats.Error() + fs.Log(f, "Couldn't list directories in root: %s", err) + } else { + for i := range entry.Contents { + entry := &entry.Contents[i] + if entry.IsDir { + out <- &fs.Dir{ + Name: f.stripRoot(entry), + When: time.Time(entry.ClientMtime), + Bytes: int64(entry.Bytes), + Count: -1, + } + } + } + } + }() + return out +} + +// A read closer which doesn't close the input +type readCloser struct { + in io.Reader +} + +// Read bytes from the object - see io.Reader +func (rc *readCloser) Read(p []byte) (n int, err error) { + return rc.in.Read(p) +} + +// Dummy close function +func (rc *readCloser) Close() error { + return nil +} + +// Put the object +// +// Copy the reader in to the new object which is returned +// +// The new object may have been created if an error is returned +func (f *FsDropbox) Put(in io.Reader, remote string, modTime time.Time, size int64) (fs.Object, error) { + // Temporary FsObject under construction + o := &FsObjectDropbox{dropbox: f, remote: remote} + return o, o.Update(in, modTime, size) +} + +// Mkdir creates the container if it doesn't exist +func (f *FsDropbox) Mkdir() error { + _, err := f.db.CreateFolder(f.slashRoot) + return err +} + +// Rmdir deletes the container +// +// Returns an error if it isn't empty +func (f *FsDropbox) Rmdir() error { + entry, err := f.db.Metadata(f.slashRoot, true, false, "", "", metadataLimit) + if err != nil { + return err + } + if len(entry.Contents) != 0 { + return errors.New("Directory not empty") + } + return f.Purge() +} + +// Return the precision +func (fs *FsDropbox) Precision() time.Duration { + return time.Second +} + +// Purge deletes all the files and the container +// +// Returns an error if it isn't empty +func (f *FsDropbox) Purge() error { + _, err := f.db.Delete(f.slashRoot) + return err +} + +// ------------------------------------------------------------ + +// Return the parent Fs +func (o *FsObjectDropbox) Fs() fs.Fs { + return o.dropbox +} + +// Return a string version +func (o *FsObjectDropbox) String() string { + if o == nil { + return "" + } + return o.remote +} + +// Return the remote path +func (o *FsObjectDropbox) Remote() string { + return o.remote +} + +// Md5sum returns the Md5sum of an object returning a lowercase hex string +// +// FIXME has to download the file! +func (o *FsObjectDropbox) Md5sum() (string, error) { + if o.md5sum != "" { + return o.md5sum, nil + } + in, err := o.Open() + if err != nil { + return "", err + } + defer in.Close() + hash := md5.New() + _, err = io.Copy(hash, in) + if err != nil { + return "", err + } + o.md5sum = fmt.Sprintf("%x", hash.Sum(nil)) + return o.md5sum, nil +} + +// Size returns the size of an object in bytes +func (o *FsObjectDropbox) Size() int64 { + return o.bytes +} + +// setMetaData sets the fs data from a dropbox.Entry +func (o *FsObjectDropbox) setMetaData(info *dropbox.Entry) { + o.bytes = int64(info.Bytes) + o.modTime = time.Time(info.ClientMtime) +} + +// Returns the remote path for the object +func (o *FsObjectDropbox) remotePath() string { + return o.dropbox.slashRoot + o.remote +} + +// readMetaData gets the info if it hasn't already been fetched +func (o *FsObjectDropbox) readMetaData() (err error) { + if !o.modTime.IsZero() { + return nil + } + entry, err := o.dropbox.db.Metadata(o.remotePath(), false, false, "", "", metadataLimit) + if err != nil { + fs.Debug(o, "Couldn't find directory: %s", err) + return fmt.Errorf("Couldn't find directory: %s", err) + } + o.setMetaData(entry) + return nil +} + +// ModTime returns the modification time of the object +// +// It attempts to read the objects mtime and if that isn't present the +// LastModified returned in the http headers +func (o *FsObjectDropbox) ModTime() time.Time { + err := o.readMetaData() + if err != nil { + fs.Log(o, "Failed to read metadata: %s", err) + return time.Now() + } + return o.modTime +} + +// Sets the modification time of the local fs object +func (o *FsObjectDropbox) SetModTime(modTime time.Time) { + err := o.readMetaData() + if err != nil { + fs.Stats.Error() + fs.Log(o, "Failed to read metadata: %s", err) + return + } + // fs.Stats.Error() + fs.Log(o, "FIXME can't update dropbox mtime") +} + +// Is this object storable +func (o *FsObjectDropbox) Storable() bool { + return true +} + +// Open an object for read +func (o *FsObjectDropbox) Open() (in io.ReadCloser, err error) { + in, _, err = o.dropbox.db.Download(o.remotePath(), "", 0) + return +} + +// Update the already existing object +// +// Copy the reader into the object updating modTime and size +// +// The new object may have been created if an error is returned +func (o *FsObjectDropbox) Update(in io.Reader, modTime time.Time, size int64) error { + rc := &readCloser{in: in} + entry, err := o.dropbox.db.UploadByChunk(rc, uploadChunkSize, o.remotePath(), true, "") + if err != nil { + return fmt.Errorf("Upload failed: %s", err) + } + o.setMetaData(entry) + return nil +} + +// Remove an object +func (o *FsObjectDropbox) Remove() error { + _, err := o.dropbox.db.Delete(o.remotePath()) + return err +} + +// Check the interfaces are satisfied +var _ fs.Fs = &FsDropbox{} +var _ fs.Purger = &FsDropbox{} +var _ fs.Object = &FsObjectDropbox{} diff --git a/rclone.go b/rclone.go index 6d89578fd..b6a3f477f 100644 --- a/rclone.go +++ b/rclone.go @@ -17,6 +17,7 @@ import ( "github.com/ncw/rclone/fs" // Active file systems _ "github.com/ncw/rclone/drive" + _ "github.com/ncw/rclone/dropbox" _ "github.com/ncw/rclone/local" _ "github.com/ncw/rclone/s3" _ "github.com/ncw/rclone/swift" diff --git a/rclonetest/rclonetest.go b/rclonetest/rclonetest.go index 19a93fff9..8e322b365 100644 --- a/rclonetest/rclonetest.go +++ b/rclonetest/rclonetest.go @@ -18,6 +18,7 @@ import ( // Active file systems _ "github.com/ncw/rclone/drive" + _ "github.com/ncw/rclone/dropbox" _ "github.com/ncw/rclone/local" _ "github.com/ncw/rclone/s3" _ "github.com/ncw/rclone/swift" From 2b0911531c01496d942e5a6281eba74dad710bc0 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Thu, 10 Jul 2014 00:17:40 +0100 Subject: [PATCH 03/10] dropbox: basics of metadata in Dropbox datastore working --- dropbox/dropbox.go | 209 ++++++++++++++++++++++++++++++++++++++------- 1 file changed, 176 insertions(+), 33 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index 44ee7a556..ad0cd60a2 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -13,6 +13,10 @@ Can only have 25,000 objects in a directory Setting metadata is problematic! Might have to use a database Md5sum has to download the file + +FIXME do we need synchronisation for any of the dropbox calls? + +FIXME need to delete metadata when we delete files! */ import ( @@ -25,6 +29,7 @@ import ( "time" "github.com/ncw/rclone/fs" + "github.com/ncw/swift" "github.com/stacktic/dropbox" ) @@ -34,6 +39,10 @@ const ( rcloneAppSecret = "1n9m04y2zx7bf26" uploadChunkSize = 64 * 1024 // chunk size for upload metadataLimit = dropbox.MetadataLimitDefault // max items to fetch at once + datastoreName = "rclone" + tableName = "metadata" + md5sumField = "md5sum" + mtimeField = "mtime" ) // Register with Fs @@ -84,9 +93,12 @@ func Config(name string) { // FsDropbox represents a remote dropbox server type FsDropbox struct { - db *dropbox.Dropbox // the connection to the dropbox server - root string // the path we are working on - slashRoot string // root with "/" prefix and postix + db *dropbox.Dropbox // the connection to the dropbox server + root string // the path we are working on + slashRoot string // root with "/" prefix and postix + datastoreManager *dropbox.DatastoreManager + datastore *dropbox.Datastore + table *dropbox.Table } // FsObjectDropbox describes a dropbox object @@ -153,25 +165,40 @@ func NewFs(name, path string) (fs.Fs, error) { // Authorize the client db.SetAccessToken(token) + // Make a db to store rclone metadata in + f.datastoreManager = db.NewDatastoreManager() + + // Open the rclone datastore + f.datastore, err = f.datastoreManager.OpenDatastore(datastoreName) + if err != nil { + return nil, err + } + + // Get the table we are using + f.table, err = f.datastore.GetTable(tableName) + if err != nil { + return nil, err + } + return f, nil } // Return an FsObject from a path func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.Object, error) { - fs := &FsObjectDropbox{ + o := &FsObjectDropbox{ dropbox: f, remote: remote, } - if info != nil { - fs.setMetaData(info) + if info == nil { + o.setMetadataFromEntry(info) } else { - err := fs.readMetaData() // reads info and meta, returning an error + err := o.readEntryAndSetMetadata() if err != nil { // logged already fs.Debug("Failed to read info: %s", err) return nil, err } } - return fs, nil + return o, nil } // Return an FsObject from a path @@ -304,7 +331,7 @@ func (f *FsDropbox) Rmdir() error { // Return the precision func (fs *FsDropbox) Precision() time.Duration { - return time.Second + return time.Nanosecond } // Purge deletes all the files and the container @@ -342,17 +369,26 @@ func (o *FsObjectDropbox) Md5sum() (string, error) { if o.md5sum != "" { return o.md5sum, nil } - in, err := o.Open() + err := o.readMetaData() if err != nil { - return "", err + fs.Log(o, "Failed to read metadata: %s", err) + return "", fmt.Errorf("Failed to read metadata: %s", err) + } - defer in.Close() - hash := md5.New() - _, err = io.Copy(hash, in) - if err != nil { - return "", err - } - o.md5sum = fmt.Sprintf("%x", hash.Sum(nil)) + + // For pre-existing files which have no md5sum can read it and set it? + + // in, err := o.Open() + // if err != nil { + // return "", err + // } + // defer in.Close() + // hash := md5.New() + // _, err = io.Copy(hash, in) + // if err != nil { + // return "", err + // } + // o.md5sum = fmt.Sprintf("%x", hash.Sum(nil)) return o.md5sum, nil } @@ -361,28 +397,102 @@ func (o *FsObjectDropbox) Size() int64 { return o.bytes } -// setMetaData sets the fs data from a dropbox.Entry -func (o *FsObjectDropbox) setMetaData(info *dropbox.Entry) { +// setMetadataFromEntry sets the fs data from a dropbox.Entry +// +// This isn't a complete set of metadata and has an inacurate date +func (o *FsObjectDropbox) setMetadataFromEntry(info *dropbox.Entry) { o.bytes = int64(info.Bytes) o.modTime = time.Time(info.ClientMtime) } +// Reads the entry from dropbox +func (o *FsObjectDropbox) readEntry() (*dropbox.Entry, error) { + entry, err := o.dropbox.db.Metadata(o.remotePath(), false, false, "", "", metadataLimit) + if err != nil { + fs.Debug(o, "Error reading file: %s", err) + return nil, fmt.Errorf("Error reading file: %s", err) + } + return entry, nil +} + +// Read entry if not set and set metadata from it +func (o *FsObjectDropbox) readEntryAndSetMetadata() error { + // Last resort set time from client + if !o.modTime.IsZero() { + return nil + } + entry, err := o.readEntry() + if err != nil { + return err + } + o.setMetadataFromEntry(entry) + return nil +} + // Returns the remote path for the object func (o *FsObjectDropbox) remotePath() string { return o.dropbox.slashRoot + o.remote } +// Returns the key for the metadata database +func (o *FsObjectDropbox) metadataKey() string { + // FIXME lower case it? + key := o.dropbox.slashRoot + o.remote + return fmt.Sprintf("%x", md5.Sum([]byte(key))) +} + // readMetaData gets the info if it hasn't already been fetched func (o *FsObjectDropbox) readMetaData() (err error) { - if !o.modTime.IsZero() { + if o.md5sum != "" { return nil } - entry, err := o.dropbox.db.Metadata(o.remotePath(), false, false, "", "", metadataLimit) + + record, err := o.dropbox.table.Get(o.metadataKey()) if err != nil { - fs.Debug(o, "Couldn't find directory: %s", err) - return fmt.Errorf("Couldn't find directory: %s", err) + fs.Debug(o, "Couldn't read metadata: %s", err) + record = nil } - o.setMetaData(entry) + + if record != nil { + // Read md5sum + md5sumInterface, ok, err := record.Get(md5sumField) + if err != nil { + return err + } + if !ok { + fs.Debug(o, "Couldn't find md5sum in record") + } else { + md5sum, ok := md5sumInterface.(string) + if !ok { + fs.Debug(o, "md5sum not a string") + } else { + o.md5sum = md5sum + } + } + + // read mtime + mtimeInterface, ok, err := record.Get(mtimeField) + if err != nil { + return err + } + if !ok { + fs.Debug(o, "Couldn't find mtime in record") + } else { + mtime, ok := mtimeInterface.(string) + if !ok { + fs.Debug(o, "mtime not a string") + } else { + modTime, err := swift.FloatStringToTime(mtime) + if err != nil { + return err + } + o.modTime = modTime + } + } + } + + // Last resort + o.readEntryAndSetMetadata() return nil } @@ -399,16 +509,45 @@ func (o *FsObjectDropbox) ModTime() time.Time { return o.modTime } +// Sets the modification time of the local fs object into the record +// FIXME if we don't set md5sum what will that do? +func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) error { + record, err := o.dropbox.table.GetOrInsert(o.metadataKey()) + if err != nil { + return fmt.Errorf("Couldn't read record: %s", err) + } + + if md5sum != "" { + err = record.Set(md5sumField, md5sum) + if err != nil { + return fmt.Errorf("Couldn't set md5sum record: %s", err) + } + } + + if !modTime.IsZero() { + mtime := swift.TimeToFloatString(modTime) + err := record.Set(mtimeField, mtime) + if err != nil { + return fmt.Errorf("Couldn't set mtime record: %s", err) + } + } + + err = o.dropbox.datastore.Commit() + if err != nil { + return fmt.Errorf("Failed to commit metadata changes: %s", err) + } + return nil +} + // Sets the modification time of the local fs object +// +// Commits the datastore func (o *FsObjectDropbox) SetModTime(modTime time.Time) { - err := o.readMetaData() + err := o.setModTimeAndMd5sum(modTime, "") if err != nil { fs.Stats.Error() - fs.Log(o, "Failed to read metadata: %s", err) - return + fs.Log(o, err.Error()) } - // fs.Stats.Error() - fs.Log(o, "FIXME can't update dropbox mtime") } // Is this object storable @@ -428,13 +567,17 @@ func (o *FsObjectDropbox) Open() (in io.ReadCloser, err error) { // // The new object may have been created if an error is returned func (o *FsObjectDropbox) Update(in io.Reader, modTime time.Time, size int64) error { - rc := &readCloser{in: in} + // Calculate md5sum as we upload it + hash := md5.New() + rc := &readCloser{in: io.TeeReader(in, hash)} entry, err := o.dropbox.db.UploadByChunk(rc, uploadChunkSize, o.remotePath(), true, "") if err != nil { return fmt.Errorf("Upload failed: %s", err) } - o.setMetaData(entry) - return nil + o.setMetadataFromEntry(entry) + + md5sum := fmt.Sprintf("%x", hash.Sum(nil)) + return o.setModTimeAndMd5sum(modTime, md5sum) } // Remove an object From c9aca330303af28a26b27e482d4c6851b77be6a4 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Fri, 11 Jul 2014 17:21:23 +0100 Subject: [PATCH 04/10] dropbox: Fix concurrent access to Dropbox datastore and Lower case keys in datastore --- dropbox/dropbox.go | 103 ++++++++++++++++++++++++++++++--------------- 1 file changed, 68 insertions(+), 35 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index ad0cd60a2..281650c1b 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -8,15 +8,19 @@ File system is case insensitive! Can only have 25,000 objects in a directory -/delta might be more efficient than recursing with Metadata - -Setting metadata is problematic! Might have to use a database - -Md5sum has to download the file +FIXME /delta might be more efficient than recursing with Metadata FIXME do we need synchronisation for any of the dropbox calls? +- added locking for datastore +- fixes concurrency there FIXME need to delete metadata when we delete files! + +FIXME Getting this sometimes +Failed to copy: Upload failed: invalid character '<' looking for beginning of value +This is a JSON decode error - from Update / UploadByChunk +- Caused by 500 error from dropbox +- See https://github.com/stacktic/dropbox/issues/1 */ import ( @@ -26,6 +30,7 @@ import ( "io" "log" "strings" + "sync" "time" "github.com/ncw/rclone/fs" @@ -35,14 +40,15 @@ import ( // Constants const ( - rcloneAppKey = "5jcck7diasz0rqy" - rcloneAppSecret = "1n9m04y2zx7bf26" - uploadChunkSize = 64 * 1024 // chunk size for upload - metadataLimit = dropbox.MetadataLimitDefault // max items to fetch at once - datastoreName = "rclone" - tableName = "metadata" - md5sumField = "md5sum" - mtimeField = "mtime" + rcloneAppKey = "5jcck7diasz0rqy" + rcloneAppSecret = "1n9m04y2zx7bf26" + uploadChunkSize = 64 * 1024 // chunk size for upload + metadataLimit = dropbox.MetadataLimitDefault // max items to fetch at once + datastoreName = "rclone" + tableName = "metadata" + md5sumField = "md5sum" + mtimeField = "mtime" + maxCommitRetries = 5 ) // Register with Fs @@ -99,6 +105,7 @@ type FsDropbox struct { datastoreManager *dropbox.DatastoreManager datastore *dropbox.Datastore table *dropbox.Table + datastoreMutex sync.Mutex // lock this when using the datastore } // FsObjectDropbox describes a dropbox object @@ -342,6 +349,31 @@ func (f *FsDropbox) Purge() error { return err } +// Tries the transaction in fn then calls commit, repeating until retry limit +// +// Holds datastore mutex while in progress +func (f *FsDropbox) transaction(fn func() error) error { + f.datastoreMutex.Lock() + defer f.datastoreMutex.Unlock() + var err error + for i := 1; i <= maxCommitRetries; i++ { + err = fn() + if err != nil { + return err + } + + err = f.datastore.Commit() + if err == nil { + break + } + fs.Debug(f, "Retrying transaction %d/%d", i, maxCommitRetries) + } + if err != nil { + return fmt.Errorf("Failed to commit metadata changes: %s", err) + } + return nil +} + // ------------------------------------------------------------ // Return the parent Fs @@ -436,8 +468,8 @@ func (o *FsObjectDropbox) remotePath() string { // Returns the key for the metadata database func (o *FsObjectDropbox) metadataKey() string { - // FIXME lower case it? - key := o.dropbox.slashRoot + o.remote + // NB File system is case insensitive + key := strings.ToLower(o.dropbox.slashRoot + o.remote) return fmt.Sprintf("%x", md5.Sum([]byte(key))) } @@ -447,7 +479,9 @@ func (o *FsObjectDropbox) readMetaData() (err error) { return nil } + o.dropbox.datastoreMutex.Lock() record, err := o.dropbox.table.Get(o.metadataKey()) + o.dropbox.datastoreMutex.Unlock() if err != nil { fs.Debug(o, "Couldn't read metadata: %s", err) record = nil @@ -512,31 +546,30 @@ func (o *FsObjectDropbox) ModTime() time.Time { // Sets the modification time of the local fs object into the record // FIXME if we don't set md5sum what will that do? func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) error { - record, err := o.dropbox.table.GetOrInsert(o.metadataKey()) - if err != nil { - return fmt.Errorf("Couldn't read record: %s", err) - } - - if md5sum != "" { - err = record.Set(md5sumField, md5sum) + key := o.metadataKey() + return o.dropbox.transaction(func() error { + record, err := o.dropbox.table.GetOrInsert(key) if err != nil { - return fmt.Errorf("Couldn't set md5sum record: %s", err) + return fmt.Errorf("Couldn't read record: %s", err) } - } - if !modTime.IsZero() { - mtime := swift.TimeToFloatString(modTime) - err := record.Set(mtimeField, mtime) - if err != nil { - return fmt.Errorf("Couldn't set mtime record: %s", err) + if md5sum != "" { + err = record.Set(md5sumField, md5sum) + if err != nil { + return fmt.Errorf("Couldn't set md5sum record: %s", err) + } } - } - err = o.dropbox.datastore.Commit() - if err != nil { - return fmt.Errorf("Failed to commit metadata changes: %s", err) - } - return nil + if !modTime.IsZero() { + mtime := swift.TimeToFloatString(modTime) + err := record.Set(mtimeField, mtime) + if err != nil { + return fmt.Errorf("Couldn't set mtime record: %s", err) + } + } + + return nil + }) } // Sets the modification time of the local fs object From d2f187e1a13938e637dd6fa50c830f835a4dea26 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sat, 12 Jul 2014 11:46:45 +0100 Subject: [PATCH 05/10] dropbox: Use /delta to list objects - much quicker Also fix major performance problem - re-reading entry each time! --- dropbox/dropbox.go | 89 +++++++++++++++++++++++++++------------------- 1 file changed, 52 insertions(+), 37 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index 281650c1b..d7cb450e4 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -4,15 +4,7 @@ package dropbox /* Limitations of dropbox -File system is case insensitive! - -Can only have 25,000 objects in a directory - -FIXME /delta might be more efficient than recursing with Metadata - -FIXME do we need synchronisation for any of the dropbox calls? -- added locking for datastore -- fixes concurrency there +File system is case insensitive FIXME need to delete metadata when we delete files! @@ -21,6 +13,7 @@ Failed to copy: Upload failed: invalid character '<' looking for beginning of va This is a JSON decode error - from Update / UploadByChunk - Caused by 500 error from dropbox - See https://github.com/stacktic/dropbox/issues/1 +- Possibly confusing dropbox with excess concurrency? */ import ( @@ -101,7 +94,8 @@ func Config(name string) { type FsDropbox struct { db *dropbox.Dropbox // the connection to the dropbox server root string // the path we are working on - slashRoot string // root with "/" prefix and postix + slashRoot string // root with "/" prefix + slashRootSlash string // root with "/" prefix and postix datastoreManager *dropbox.DatastoreManager datastore *dropbox.Datastore table *dropbox.Table @@ -157,13 +151,15 @@ func NewFs(name, path string) (fs.Fs, error) { return nil, err } slashRoot := "/" + root + slashRootSlash := slashRoot if root != "" { - slashRoot += "/" + slashRootSlash += "/" } f := &FsDropbox{ - root: root, - slashRoot: slashRoot, - db: db, + root: root, + slashRoot: slashRoot, + slashRootSlash: slashRootSlash, + db: db, } // Read the token from the config file @@ -196,7 +192,7 @@ func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs. dropbox: f, remote: remote, } - if info == nil { + if info != nil { o.setMetadataFromEntry(info) } else { err := o.readEntryAndSetMetadata() @@ -227,29 +223,46 @@ func (f *FsDropbox) NewFsObject(remote string) fs.Object { // Strips the root off entry and returns it func (f *FsDropbox) stripRoot(entry *dropbox.Entry) string { path := entry.Path - if strings.HasPrefix(path, f.slashRoot) { - path = path[len(f.slashRoot):] + if strings.HasPrefix(path, f.slashRootSlash) { + path = path[len(f.slashRootSlash):] } return path } -// Walk the path returning a channel of FsObjects -// -// FIXME could do this in parallel but needs to be limited to Checkers -func (f *FsDropbox) list(path string, out fs.ObjectsChan) { - entry, err := f.db.Metadata(f.slashRoot+path, true, false, "", "", metadataLimit) - if err != nil { - fs.Stats.Error() - fs.Log(f, "Couldn't list %q: %s", path, err) - } else { - for i := range entry.Contents { - entry := &entry.Contents[i] - path = f.stripRoot(entry) - if entry.IsDir { - f.list(path, out) - } else { - out <- f.NewFsObjectWithInfo(path, entry) +// Walk the root returning a channel of FsObjects +func (f *FsDropbox) list(out fs.ObjectsChan) { + cursor := "" + for { + deltaPage, err := f.db.Delta(cursor, f.slashRoot) + if err != nil { + fs.Stats.Error() + fs.Log(f, "Couldn't list: %s", err) + break + } else { + if deltaPage.Reset && cursor != "" { + fs.Log(f, "Unexpected reset during listing - try again") + fs.Stats.Error() + break } + fs.Debug(f, "%d delta entries received", len(deltaPage.Entries)) + for i := range deltaPage.Entries { + deltaEntry := &deltaPage.Entries[i] + entry := deltaEntry.Entry + if entry == nil { + // This notifies of a deleted object which we ignore + continue + } + if entry.IsDir { + // ignore directories + } else { + path := f.stripRoot(entry) + out <- f.NewFsObjectWithInfo(path, entry) + } + } + if !deltaPage.HasMore { + break + } + cursor = deltaPage.Cursor } } } @@ -259,7 +272,7 @@ func (f *FsDropbox) List() fs.ObjectsChan { out := make(fs.ObjectsChan, fs.Config.Checkers) go func() { defer close(out) - f.list("", out) + f.list(out) }() return out } @@ -326,7 +339,7 @@ func (f *FsDropbox) Mkdir() error { // // Returns an error if it isn't empty func (f *FsDropbox) Rmdir() error { - entry, err := f.db.Metadata(f.slashRoot, true, false, "", "", metadataLimit) + entry, err := f.db.Metadata(f.slashRoot, true, false, "", "", 16) if err != nil { return err } @@ -463,13 +476,13 @@ func (o *FsObjectDropbox) readEntryAndSetMetadata() error { // Returns the remote path for the object func (o *FsObjectDropbox) remotePath() string { - return o.dropbox.slashRoot + o.remote + return o.dropbox.slashRootSlash + o.remote } // Returns the key for the metadata database func (o *FsObjectDropbox) metadataKey() string { // NB File system is case insensitive - key := strings.ToLower(o.dropbox.slashRoot + o.remote) + key := strings.ToLower(o.remotePath()) return fmt.Sprintf("%x", md5.Sum([]byte(key))) } @@ -479,6 +492,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) { return nil } + // fs.Debug(o, "Reading metadata from datastore") o.dropbox.datastoreMutex.Lock() record, err := o.dropbox.table.Get(o.metadataKey()) o.dropbox.datastoreMutex.Unlock() @@ -547,6 +561,7 @@ func (o *FsObjectDropbox) ModTime() time.Time { // FIXME if we don't set md5sum what will that do? func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) error { key := o.metadataKey() + // fs.Debug(o, "Writing metadata to datastore") return o.dropbox.transaction(func() error { record, err := o.dropbox.table.GetOrInsert(key) if err != nil { From e57a4c7c0cc64879d8af6582c4c0cb07295b42fd Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sat, 12 Jul 2014 12:38:30 +0100 Subject: [PATCH 06/10] dropbox: open the datastore in the background --- dropbox/dropbox.go | 51 ++++++++++++++++++++++++++++++++++------------ 1 file changed, 38 insertions(+), 13 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index d7cb450e4..a9ab7171b 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -100,6 +100,7 @@ type FsDropbox struct { datastore *dropbox.Datastore table *dropbox.Table datastoreMutex sync.Mutex // lock this when using the datastore + datastoreErr error // pending errors on the datastore } // FsObjectDropbox describes a dropbox object @@ -171,17 +172,28 @@ func NewFs(name, path string) (fs.Fs, error) { // Make a db to store rclone metadata in f.datastoreManager = db.NewDatastoreManager() - // Open the rclone datastore - f.datastore, err = f.datastoreManager.OpenDatastore(datastoreName) - if err != nil { - return nil, err - } + // Open the datastore in the background + go func() { + f.datastoreMutex.Lock() + defer f.datastoreMutex.Unlock() + fs.Debug(f, "Open rclone datastore") + // Open the rclone datastore + f.datastore, err = f.datastoreManager.OpenDatastore(datastoreName) + if err != nil { + fs.Log(f, "Failed to open datastore: %v", err) + f.datastoreErr = err + return + } - // Get the table we are using - f.table, err = f.datastore.GetTable(tableName) - if err != nil { - return nil, err - } + // Get the table we are using + f.table, err = f.datastore.GetTable(tableName) + if err != nil { + fs.Log(f, "Failed to open datastore table: %v", err) + f.datastoreErr = err + return + } + fs.Debug(f, "Open rclone datastore finished") + }() return f, nil } @@ -368,6 +380,9 @@ func (f *FsDropbox) Purge() error { func (f *FsDropbox) transaction(fn func() error) error { f.datastoreMutex.Lock() defer f.datastoreMutex.Unlock() + if f.datastoreErr != nil { + return f.datastoreErr + } var err error for i := 1; i <= maxCommitRetries; i++ { err = fn() @@ -387,6 +402,18 @@ func (f *FsDropbox) transaction(fn func() error) error { return nil } +// Reads the record attached to key +// +// Holds datastore mutex while in progress +func (f *FsDropbox) readRecord(key string) (*dropbox.Record, error) { + f.datastoreMutex.Lock() + defer f.datastoreMutex.Unlock() + if f.datastoreErr != nil { + return nil, f.datastoreErr + } + return f.table.Get(key) +} + // ------------------------------------------------------------ // Return the parent Fs @@ -493,9 +520,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) { } // fs.Debug(o, "Reading metadata from datastore") - o.dropbox.datastoreMutex.Lock() - record, err := o.dropbox.table.Get(o.metadataKey()) - o.dropbox.datastoreMutex.Unlock() + record, err := o.dropbox.readRecord(o.metadataKey()) if err != nil { fs.Debug(o, "Couldn't read metadata: %s", err) record = nil From b185e104edd26a2386d13c2a09435c8ff96d357d Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sun, 13 Jul 2014 10:51:47 +0100 Subject: [PATCH 07/10] dropbox: Fix mkdir on already created directory --- dropbox/dropbox.go | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index a9ab7171b..1e58db398 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -343,7 +343,14 @@ func (f *FsDropbox) Put(in io.Reader, remote string, modTime time.Time, size int // Mkdir creates the container if it doesn't exist func (f *FsDropbox) Mkdir() error { - _, err := f.db.CreateFolder(f.slashRoot) + entry, err := f.db.Metadata(f.slashRoot, false, false, "", "", metadataLimit) + if err == nil { + if entry.IsDir { + return nil + } + return fmt.Errorf("%q already exists as file", f.root) + } + _, err = f.db.CreateFolder(f.slashRoot) return err } From f8bb0d9cc8f321b9d53719d1cf9a674c2be6e146 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sun, 13 Jul 2014 10:53:53 +0100 Subject: [PATCH 08/10] dropbox: remove metadata when we remove files --- dropbox/dropbox.go | 86 +++++++++++++++++++++++++++++++++++++++------- 1 file changed, 74 insertions(+), 12 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index 1e58db398..c05028c92 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -6,7 +6,10 @@ Limitations of dropbox File system is case insensitive -FIXME need to delete metadata when we delete files! +The datastore is limited to 100,000 records which therefore is the +limit of the number of files that rclone can use on dropbox. + +FIXME only open datastore if we need it? FIXME Getting this sometimes Failed to copy: Upload failed: invalid character '<' looking for beginning of value @@ -261,14 +264,22 @@ func (f *FsDropbox) list(out fs.ObjectsChan) { deltaEntry := &deltaPage.Entries[i] entry := deltaEntry.Entry if entry == nil { - // This notifies of a deleted object which we ignore - continue - } - if entry.IsDir { - // ignore directories + // This notifies of a deleted object + fs.Debug(f, "Deleting metadata for %q", deltaEntry.Path) + key := metadataKey(deltaEntry.Path) // Path is lowercased + err := f.deleteMetadata(key) + if err != nil { + fs.Debug(f, "Failed to delete metadata for %q", deltaEntry.Path) + // Don't accumulate Error here + } + } else { - path := f.stripRoot(entry) - out <- f.NewFsObjectWithInfo(path, entry) + if entry.IsDir { + // ignore directories + } else { + path := f.stripRoot(entry) + out <- f.NewFsObjectWithInfo(path, entry) + } } } if !deltaPage.HasMore { @@ -375,8 +386,26 @@ func (fs *FsDropbox) Precision() time.Duration { // Purge deletes all the files and the container // -// Returns an error if it isn't empty +// Optional interface: Only implement this if you have a way of +// deleting all the files quicker than just running Remove() on the +// result of List() func (f *FsDropbox) Purge() error { + // Delete metadata first + var wg sync.WaitGroup + to_be_deleted := f.List() + wg.Add(fs.Config.Transfers) + for i := 0; i < fs.Config.Transfers; i++ { + go func() { + defer wg.Done() + for dst := range to_be_deleted { + o := dst.(*FsObjectDropbox) + o.deleteMetadata() + } + }() + } + wg.Wait() + + // Let dropbox delete the filesystem tree _, err := f.db.Delete(f.slashRoot) return err } @@ -409,6 +438,21 @@ func (f *FsDropbox) transaction(fn func() error) error { return nil } +// Deletes the medadata associated with this key +func (f *FsDropbox) deleteMetadata(key string) error { + return f.transaction(func() error { + record, err := f.table.Get(key) + if err != nil { + return fmt.Errorf("Couldn't get record: %s", err) + } + if record == nil { + return nil + } + record.DeleteRecord() + return nil + }) +} + // Reads the record attached to key // // Holds datastore mutex while in progress @@ -513,11 +557,16 @@ func (o *FsObjectDropbox) remotePath() string { return o.dropbox.slashRootSlash + o.remote } +// Returns the key for the metadata database for a given path +func metadataKey(path string) string { + // NB File system is case insensitive + path = strings.ToLower(path) + return fmt.Sprintf("%x", md5.Sum([]byte(path))) +} + // Returns the key for the metadata database func (o *FsObjectDropbox) metadataKey() string { - // NB File system is case insensitive - key := strings.ToLower(o.remotePath()) - return fmt.Sprintf("%x", md5.Sum([]byte(key))) + return metadataKey(o.remotePath()) } // readMetaData gets the info if it hasn't already been fetched @@ -619,6 +668,18 @@ func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) }) } +// Deletes the medadata associated with this file +// +// It logs any errors +func (o *FsObjectDropbox) deleteMetadata() { + fs.Debug(o, "Deleting metadata from datastore") + err := o.dropbox.deleteMetadata(o.metadataKey()) + if err != nil { + fs.Log(o, "Error deleting metadata: %v", err) + fs.Stats.Error() + } +} + // Sets the modification time of the local fs object // // Commits the datastore @@ -662,6 +723,7 @@ func (o *FsObjectDropbox) Update(in io.Reader, modTime time.Time, size int64) er // Remove an object func (o *FsObjectDropbox) Remove() error { + o.deleteMetadata() _, err := o.dropbox.db.Delete(o.remotePath()) return err } From 7c9bdb4b7a2f7d3b64ab3f22934ce01dbe75fff3 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Mon, 14 Jul 2014 11:24:04 +0100 Subject: [PATCH 09/10] dropbox: make limited fs work (copy single file) --- dropbox/dropbox.go | 91 ++++++++++++++++++++++++++-------------------- 1 file changed, 51 insertions(+), 40 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index c05028c92..226269063 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -25,6 +25,7 @@ import ( "fmt" "io" "log" + "path" "strings" "sync" "time" @@ -122,12 +123,6 @@ func (f *FsDropbox) String() string { return fmt.Sprintf("Dropbox root '%s'", f.root) } -// parseParse parses a dropbox 'url' -func parseDropboxPath(path string) (root string, err error) { - root = strings.Trim(path, "/") - return -} - // Makes a new dropbox from the config func newDropbox(name string) *dropbox.Dropbox { db := dropbox.NewDropbox() @@ -147,24 +142,12 @@ func newDropbox(name string) *dropbox.Dropbox { } // NewFs contstructs an FsDropbox from the path, container:path -func NewFs(name, path string) (fs.Fs, error) { +func NewFs(name, root string) (fs.Fs, error) { db := newDropbox(name) - - root, err := parseDropboxPath(path) - if err != nil { - return nil, err - } - slashRoot := "/" + root - slashRootSlash := slashRoot - if root != "" { - slashRootSlash += "/" - } f := &FsDropbox{ - root: root, - slashRoot: slashRoot, - slashRootSlash: slashRootSlash, - db: db, + db: db, } + f.setRoot(root) // Read the token from the config file token := fs.ConfigFile.MustValue(name, "token") @@ -176,31 +159,59 @@ func NewFs(name, path string) (fs.Fs, error) { f.datastoreManager = db.NewDatastoreManager() // Open the datastore in the background - go func() { - f.datastoreMutex.Lock() - defer f.datastoreMutex.Unlock() - fs.Debug(f, "Open rclone datastore") - // Open the rclone datastore - f.datastore, err = f.datastoreManager.OpenDatastore(datastoreName) - if err != nil { - fs.Log(f, "Failed to open datastore: %v", err) - f.datastoreErr = err - return - } + go f.openDataStore() - // Get the table we are using - f.table, err = f.datastore.GetTable(tableName) - if err != nil { - fs.Log(f, "Failed to open datastore table: %v", err) - f.datastoreErr = err - return + // See if the root is actually an object + entry, err := f.db.Metadata(f.slashRoot, false, false, "", "", metadataLimit) + if err == nil && !entry.IsDir { + remote := path.Base(f.root) + newRoot := path.Dir(f.root) + if newRoot == "." { + newRoot = "" } - fs.Debug(f, "Open rclone datastore finished") - }() + f.setRoot(newRoot) + obj := f.NewFsObject(remote) + // return a Fs Limited to this object + return fs.NewLimited(f, obj), nil + } return f, nil } +// Sets root in f +func (f *FsDropbox) setRoot(root string) { + f.root = strings.Trim(root, "/") + f.slashRoot = "/" + f.root + f.slashRootSlash = f.slashRoot + if f.root != "" { + f.slashRootSlash += "/" + } +} + +// Opens the datastore in f +func (f *FsDropbox) openDataStore() { + f.datastoreMutex.Lock() + defer f.datastoreMutex.Unlock() + fs.Debug(f, "Open rclone datastore") + // Open the rclone datastore + var err error + f.datastore, err = f.datastoreManager.OpenDatastore(datastoreName) + if err != nil { + fs.Log(f, "Failed to open datastore: %v", err) + f.datastoreErr = err + return + } + + // Get the table we are using + f.table, err = f.datastore.GetTable(tableName) + if err != nil { + fs.Log(f, "Failed to open datastore table: %v", err) + f.datastoreErr = err + return + } + fs.Debug(f, "Open rclone datastore finished") +} + // Return an FsObject from a path func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.Object, error) { o := &FsObjectDropbox{ From dfc8a375f6b9f92e9f4c4bc4fcfdd3c49b9da2ca Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Mon, 14 Jul 2014 12:00:21 +0100 Subject: [PATCH 10/10] dropbox: Switch to using RFC3339 for time metadata --- dropbox/dropbox.go | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/dropbox/dropbox.go b/dropbox/dropbox.go index 226269063..24cbe98b3 100644 --- a/dropbox/dropbox.go +++ b/dropbox/dropbox.go @@ -31,7 +31,6 @@ import ( "time" "github.com/ncw/rclone/fs" - "github.com/ncw/swift" "github.com/stacktic/dropbox" ) @@ -46,6 +45,8 @@ const ( md5sumField = "md5sum" mtimeField = "mtime" maxCommitRetries = 5 + RFC3339In = time.RFC3339 + RFC3339Out = "2006-01-02T15:04:05.000000000Z07:00" ) // Register with Fs @@ -622,7 +623,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) { if !ok { fs.Debug(o, "mtime not a string") } else { - modTime, err := swift.FloatStringToTime(mtime) + modTime, err := time.Parse(RFC3339In, mtime) if err != nil { return err } @@ -668,7 +669,7 @@ func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) } if !modTime.IsZero() { - mtime := swift.TimeToFloatString(modTime) + mtime := modTime.Format(RFC3339Out) err := record.Set(mtimeField, mtime) if err != nil { return fmt.Errorf("Couldn't set mtime record: %s", err)