forked from TrueCloudLab/rclone
drive: refactor listDir to take a client function
This commit is contained in:
parent
98108caac2
commit
351829e9fe
1 changed files with 52 additions and 34 deletions
80
fs_drive.go
80
fs_drive.go
|
@ -21,6 +21,7 @@ package main
|
|||
// * multiple files with the same name
|
||||
// * files can be in multiple directories
|
||||
// * can have directory loops
|
||||
// * files with / in name
|
||||
|
||||
import (
|
||||
"code.google.com/p/goauth2/oauth"
|
||||
|
@ -133,10 +134,17 @@ func parseDrivePath(path string) (root string, err error) {
|
|||
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
|
||||
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)
|
||||
if title != "" {
|
||||
// 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)
|
||||
}
|
||||
list := f.svc.Files.List().Q(query)
|
||||
OUTER:
|
||||
for {
|
||||
files, err := list.Do()
|
||||
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 == "" {
|
||||
break
|
||||
}
|
||||
|
@ -280,30 +294,33 @@ func (f *FsDrive) NewFsObject(remote string) FsObject {
|
|||
|
||||
// Path should be directory path either "" or "path/"
|
||||
func (f *FsDrive) listDir(dirId string, path string, out FsObjectsChan) error {
|
||||
var subError error
|
||||
// Make the API request
|
||||
items, err := f.listAll(dirId, "", false, false)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
for _, item := range items {
|
||||
_, err := f.listAll(dirId, "", false, false, func(item *drive.File) bool {
|
||||
// Recurse on directories
|
||||
// FIXME should do this in parallel
|
||||
// use a wg to sync then collect error
|
||||
if item.MimeType == driveFolderType {
|
||||
err := f.listDir(item.Id, path+item.Title+"/", out)
|
||||
if err != nil {
|
||||
return err
|
||||
subError = f.listDir(item.Id, path+item.Title+"/", out)
|
||||
if subError != nil {
|
||||
return true
|
||||
}
|
||||
} else {
|
||||
// If item has no MD5 sum it isn't stored on drive, so ignore it
|
||||
if item.Md5Checksum == "" {
|
||||
continue
|
||||
}
|
||||
if item.Md5Checksum != "" {
|
||||
if fs := f.NewFsObjectWithInfo(path+item.Title, item); fs != nil {
|
||||
out <- fs
|
||||
}
|
||||
}
|
||||
}
|
||||
return false
|
||||
})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if subError != nil {
|
||||
return subError
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
|
@ -380,18 +397,16 @@ func (f *FsDrive) _findDir(path string, create bool) (pathId string, err error)
|
|||
}
|
||||
|
||||
// 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 {
|
||||
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 !found {
|
||||
|
@ -604,18 +619,21 @@ func (fs *FsObjectDrive) readMetaData() (err error) {
|
|||
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 {
|
||||
return err
|
||||
}
|
||||
for _, file := range items {
|
||||
if file.Title == leaf {
|
||||
fs.setMetaData(file)
|
||||
return nil
|
||||
}
|
||||
}
|
||||
if !found {
|
||||
FsDebug(fs, "Couldn't find object")
|
||||
return fmt.Errorf("Couldn't find object")
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// ModTime returns the modification time of the object
|
||||
|
|
Loading…
Reference in a new issue