dropbox: Use /delta to list objects - much quicker

Also fix major performance problem - re-reading entry each time!
This commit is contained in:
Nick Craig-Wood 2014-07-12 11:46:45 +01:00
parent c9aca33030
commit d2f187e1a1

View file

@ -4,15 +4,7 @@ package dropbox
/* /*
Limitations of dropbox Limitations of dropbox
File system is case insensitive! 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
FIXME need to delete metadata when we delete files! 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 This is a JSON decode error - from Update / UploadByChunk
- Caused by 500 error from dropbox - Caused by 500 error from dropbox
- See https://github.com/stacktic/dropbox/issues/1 - See https://github.com/stacktic/dropbox/issues/1
- Possibly confusing dropbox with excess concurrency?
*/ */
import ( import (
@ -101,7 +94,8 @@ func Config(name string) {
type FsDropbox struct { type FsDropbox struct {
db *dropbox.Dropbox // the connection to the dropbox server db *dropbox.Dropbox // the connection to the dropbox server
root string // the path we are working on 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 datastoreManager *dropbox.DatastoreManager
datastore *dropbox.Datastore datastore *dropbox.Datastore
table *dropbox.Table table *dropbox.Table
@ -157,13 +151,15 @@ func NewFs(name, path string) (fs.Fs, error) {
return nil, err return nil, err
} }
slashRoot := "/" + root slashRoot := "/" + root
slashRootSlash := slashRoot
if root != "" { if root != "" {
slashRoot += "/" slashRootSlash += "/"
} }
f := &FsDropbox{ f := &FsDropbox{
root: root, root: root,
slashRoot: slashRoot, slashRoot: slashRoot,
db: db, slashRootSlash: slashRootSlash,
db: db,
} }
// Read the token from the config file // Read the token from the config file
@ -196,7 +192,7 @@ func (f *FsDropbox) newFsObjectWithInfo(remote string, info *dropbox.Entry) (fs.
dropbox: f, dropbox: f,
remote: remote, remote: remote,
} }
if info == nil { if info != nil {
o.setMetadataFromEntry(info) o.setMetadataFromEntry(info)
} else { } else {
err := o.readEntryAndSetMetadata() err := o.readEntryAndSetMetadata()
@ -227,29 +223,46 @@ func (f *FsDropbox) NewFsObject(remote string) fs.Object {
// Strips the root off entry and returns it // Strips the root off entry and returns it
func (f *FsDropbox) stripRoot(entry *dropbox.Entry) string { func (f *FsDropbox) stripRoot(entry *dropbox.Entry) string {
path := entry.Path path := entry.Path
if strings.HasPrefix(path, f.slashRoot) { if strings.HasPrefix(path, f.slashRootSlash) {
path = path[len(f.slashRoot):] path = path[len(f.slashRootSlash):]
} }
return path return path
} }
// Walk the path returning a channel of FsObjects // Walk the root returning a channel of FsObjects
// func (f *FsDropbox) list(out fs.ObjectsChan) {
// FIXME could do this in parallel but needs to be limited to Checkers cursor := ""
func (f *FsDropbox) list(path string, out fs.ObjectsChan) { for {
entry, err := f.db.Metadata(f.slashRoot+path, true, false, "", "", metadataLimit) deltaPage, err := f.db.Delta(cursor, f.slashRoot)
if err != nil { if err != nil {
fs.Stats.Error() fs.Stats.Error()
fs.Log(f, "Couldn't list %q: %s", path, err) fs.Log(f, "Couldn't list: %s", err)
} else { break
for i := range entry.Contents { } else {
entry := &entry.Contents[i] if deltaPage.Reset && cursor != "" {
path = f.stripRoot(entry) fs.Log(f, "Unexpected reset during listing - try again")
if entry.IsDir { fs.Stats.Error()
f.list(path, out) break
} else {
out <- f.NewFsObjectWithInfo(path, entry)
} }
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) out := make(fs.ObjectsChan, fs.Config.Checkers)
go func() { go func() {
defer close(out) defer close(out)
f.list("", out) f.list(out)
}() }()
return out return out
} }
@ -326,7 +339,7 @@ func (f *FsDropbox) Mkdir() error {
// //
// Returns an error if it isn't empty // Returns an error if it isn't empty
func (f *FsDropbox) Rmdir() error { 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 { if err != nil {
return err return err
} }
@ -463,13 +476,13 @@ func (o *FsObjectDropbox) readEntryAndSetMetadata() error {
// Returns the remote path for the object // Returns the remote path for the object
func (o *FsObjectDropbox) remotePath() string { func (o *FsObjectDropbox) remotePath() string {
return o.dropbox.slashRoot + o.remote return o.dropbox.slashRootSlash + o.remote
} }
// Returns the key for the metadata database // Returns the key for the metadata database
func (o *FsObjectDropbox) metadataKey() string { func (o *FsObjectDropbox) metadataKey() string {
// NB File system is case insensitive // 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))) return fmt.Sprintf("%x", md5.Sum([]byte(key)))
} }
@ -479,6 +492,7 @@ func (o *FsObjectDropbox) readMetaData() (err error) {
return nil return nil
} }
// fs.Debug(o, "Reading metadata from datastore")
o.dropbox.datastoreMutex.Lock() o.dropbox.datastoreMutex.Lock()
record, err := o.dropbox.table.Get(o.metadataKey()) record, err := o.dropbox.table.Get(o.metadataKey())
o.dropbox.datastoreMutex.Unlock() 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? // FIXME if we don't set md5sum what will that do?
func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) error { func (o *FsObjectDropbox) setModTimeAndMd5sum(modTime time.Time, md5sum string) error {
key := o.metadataKey() key := o.metadataKey()
// fs.Debug(o, "Writing metadata to datastore")
return o.dropbox.transaction(func() error { return o.dropbox.transaction(func() error {
record, err := o.dropbox.table.GetOrInsert(key) record, err := o.dropbox.table.GetOrInsert(key)
if err != nil { if err != nil {