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
|
// * 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
|
||||||
//
|
//
|
||||||
|
|
Loading…
Reference in a new issue