drive: refactor listDir to take a client function

This commit is contained in:
Nick Craig-Wood 2013-01-20 11:56:56 +00:00
parent 98108caac2
commit 351829e9fe

View file

@ -21,6 +21,7 @@ package main
// * multiple files with the same name // * multiple files with the same name
// * files can be in multiple directories // * files can be in multiple directories
// * can have directory loops // * can have directory loops
// * files with / in name
import ( import (
"code.google.com/p/goauth2/oauth" "code.google.com/p/goauth2/oauth"
@ -133,10 +134,17 @@ func parseDrivePath(path string) (root string, err error) {
return return
} }
// Lists the directory required // User function to process a File item from listAll
//
// Should return true to finish processing
type listAllFn func(*drive.File) bool
// Lists the directory required calling the user function on each item found
//
// If the user fn ever returns true then it early exits with found = true
// //
// Search params: https://developers.google.com/drive/search-parameters // Search params: https://developers.google.com/drive/search-parameters
func (f *FsDrive) listAll(dirId string, title string, directoriesOnly bool, filesOnly bool) (items []*drive.File, err error) { func (f *FsDrive) listAll(dirId string, title string, directoriesOnly bool, filesOnly bool, fn listAllFn) (found bool, err error) {
query := fmt.Sprintf("trashed=false and '%s' in parents", dirId) query := fmt.Sprintf("trashed=false and '%s' in parents", dirId)
if title != "" { if title != "" {
// Escaping the backslash isn't documented but seems to work // Escaping the backslash isn't documented but seems to work
@ -151,12 +159,18 @@ func (f *FsDrive) listAll(dirId string, title string, directoriesOnly bool, file
query += fmt.Sprintf(" and mimeType!='%s'", driveFolderType) query += fmt.Sprintf(" and mimeType!='%s'", driveFolderType)
} }
list := f.svc.Files.List().Q(query) list := f.svc.Files.List().Q(query)
OUTER:
for { for {
files, err := list.Do() files, err := list.Do()
if err != nil { if err != nil {
return nil, fmt.Errorf("Couldn't list directory: %s", err) return false, fmt.Errorf("Couldn't list directory: %s", err)
}
for _, item := range files.Items {
if fn(item) {
found = true
break OUTER
}
} }
items = append(items, files.Items...)
if files.NextPageToken == "" { if files.NextPageToken == "" {
break break
} }
@ -280,30 +294,33 @@ func (f *FsDrive) NewFsObject(remote string) FsObject {
// Path should be directory path either "" or "path/" // Path should be directory path either "" or "path/"
func (f *FsDrive) listDir(dirId string, path string, out FsObjectsChan) error { func (f *FsDrive) listDir(dirId string, path string, out FsObjectsChan) error {
var subError error
// Make the API request // Make the API request
items, err := f.listAll(dirId, "", false, false) _, err := f.listAll(dirId, "", false, false, func(item *drive.File) bool {
if err != nil {
return err
}
for _, item := range items {
// Recurse on directories // Recurse on directories
// FIXME should do this in parallel // FIXME should do this in parallel
// use a wg to sync then collect error // use a wg to sync then collect error
if item.MimeType == driveFolderType { if item.MimeType == driveFolderType {
err := f.listDir(item.Id, path+item.Title+"/", out) subError = f.listDir(item.Id, path+item.Title+"/", out)
if err != nil { if subError != nil {
return err return true
} }
} else { } else {
// If item has no MD5 sum it isn't stored on drive, so ignore it // If item has no MD5 sum it isn't stored on drive, so ignore it
if item.Md5Checksum == "" { if item.Md5Checksum != "" {
continue
}
if fs := f.NewFsObjectWithInfo(path+item.Title, item); fs != nil { if fs := f.NewFsObjectWithInfo(path+item.Title, item); fs != nil {
out <- fs out <- fs
} }
} }
} }
return false
})
if err != nil {
return err
}
if subError != nil {
return subError
}
return nil return nil
} }
@ -380,18 +397,16 @@ func (f *FsDrive) _findDir(path string, create bool) (pathId string, err error)
} }
// Find the leaf in pathId // Find the leaf in pathId
items, err := f.listAll(pathId, leaf, true, false) found, err := f.listAll(pathId, leaf, true, false, func(item *drive.File) bool {
if item.Title == leaf {
pathId = item.Id
return true
}
return false
})
if err != nil { if err != nil {
return pathId, err return pathId, err
} }
found := false
for _, file := range items {
if file.Title == leaf {
pathId = file.Id
found = true
break
}
}
// If not found create the directory if required or return an error // If not found create the directory if required or return an error
if !found { if !found {
@ -604,19 +619,22 @@ func (fs *FsObjectDrive) readMetaData() (err error) {
return fmt.Errorf("Couldn't find directory: %s", err) return fmt.Errorf("Couldn't find directory: %s", err)
} }
items, err := fs.drive.listAll(directoryId, leaf, false, true) found, err := fs.drive.listAll(directoryId, leaf, false, true, func(item *drive.File) bool {
if item.Title == leaf {
fs.setMetaData(item)
return true
}
return false
})
if err != nil { if err != nil {
return err return err
} }
for _, file := range items { if !found {
if file.Title == leaf {
fs.setMetaData(file)
return nil
}
}
FsDebug(fs, "Couldn't find object") FsDebug(fs, "Couldn't find object")
return fmt.Errorf("Couldn't find object") return fmt.Errorf("Couldn't find object")
} }
return nil
}
// ModTime returns the modification time of the object // ModTime returns the modification time of the object
// //