pcloud: add support for recursive list

This commit is contained in:
Niels van de Weem 2021-11-26 20:55:50 +01:00 committed by Nick Craig-Wood
parent c41814fd2d
commit df09c3f555
2 changed files with 81 additions and 47 deletions

View file

@ -2,8 +2,6 @@
// object storage system. // object storage system.
package pcloud package pcloud
// FIXME implement ListR? /listfolder can do recursive lists
// FIXME cleanup returns login required? // FIXME cleanup returns login required?
// FIXME mime type? Fix overview if implement. // FIXME mime type? Fix overview if implement.
@ -27,6 +25,7 @@ import (
"github.com/rclone/rclone/fs/config/obscure" "github.com/rclone/rclone/fs/config/obscure"
"github.com/rclone/rclone/fs/fserrors" "github.com/rclone/rclone/fs/fserrors"
"github.com/rclone/rclone/fs/hash" "github.com/rclone/rclone/fs/hash"
"github.com/rclone/rclone/fs/walk"
"github.com/rclone/rclone/lib/dircache" "github.com/rclone/rclone/lib/dircache"
"github.com/rclone/rclone/lib/encoder" "github.com/rclone/rclone/lib/encoder"
"github.com/rclone/rclone/lib/oauthutil" "github.com/rclone/rclone/lib/oauthutil"
@ -246,7 +245,7 @@ func (f *Fs) readMetaDataForPath(ctx context.Context, path string) (info *api.It
return nil, err return nil, err
} }
found, err := f.listAll(ctx, directoryID, false, true, func(item *api.Item) bool { found, err := f.listAll(ctx, directoryID, false, true, false, func(item *api.Item) bool {
if item.Name == leaf { if item.Name == leaf {
info = item info = item
return true return true
@ -380,7 +379,7 @@ func (f *Fs) NewObject(ctx context.Context, remote string) (fs.Object, error) {
// FindLeaf finds a directory of name leaf in the folder with ID pathID // FindLeaf finds a directory of name leaf in the folder with ID pathID
func (f *Fs) FindLeaf(ctx context.Context, pathID, leaf string) (pathIDOut string, found bool, err error) { func (f *Fs) FindLeaf(ctx context.Context, pathID, leaf string) (pathIDOut string, found bool, err error) {
// Find the leaf in pathID // Find the leaf in pathID
found, err = f.listAll(ctx, pathID, true, false, func(item *api.Item) bool { found, err = f.listAll(ctx, pathID, true, false, false, func(item *api.Item) bool {
if item.Name == leaf { if item.Name == leaf {
pathIDOut = item.ID pathIDOut = item.ID
return true return true
@ -446,14 +445,16 @@ type listAllFn func(*api.Item) bool
// Lists the directory required calling the user function on each item found // 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 // If the user fn ever returns true then it early exits with found = true
func (f *Fs) listAll(ctx context.Context, dirID string, directoriesOnly bool, filesOnly bool, fn listAllFn) (found bool, err error) { func (f *Fs) listAll(ctx context.Context, dirID string, directoriesOnly bool, filesOnly bool, recursive bool, fn listAllFn) (found bool, err error) {
opts := rest.Opts{ opts := rest.Opts{
Method: "GET", Method: "GET",
Path: "/listfolder", Path: "/listfolder",
Parameters: url.Values{}, Parameters: url.Values{},
} }
if recursive {
opts.Parameters.Set("recursive", "1")
}
opts.Parameters.Set("folderid", dirIDtoNumber(dirID)) opts.Parameters.Set("folderid", dirIDtoNumber(dirID))
// FIXME can do recursive
var result api.ItemResult var result api.ItemResult
var resp *http.Response var resp *http.Response
@ -465,8 +466,10 @@ func (f *Fs) listAll(ctx context.Context, dirID string, directoriesOnly bool, fi
if err != nil { if err != nil {
return found, fmt.Errorf("couldn't list files: %w", err) return found, fmt.Errorf("couldn't list files: %w", err)
} }
for i := range result.Metadata.Contents { var recursiveContents func(is []api.Item, path string)
item := &result.Metadata.Contents[i] recursiveContents = func(is []api.Item, path string) {
for i := range is {
item := &is[i]
if item.IsFolder { if item.IsFolder {
if filesOnly { if filesOnly {
continue continue
@ -476,15 +479,58 @@ func (f *Fs) listAll(ctx context.Context, dirID string, directoriesOnly bool, fi
continue continue
} }
} }
item.Name = f.opt.Enc.ToStandardName(item.Name) item.Name = path + f.opt.Enc.ToStandardName(item.Name)
if fn(item) { if fn(item) {
found = true found = true
break break
} }
if recursive {
recursiveContents(item.Contents, item.Name+"/")
} }
}
}
recursiveContents(result.Metadata.Contents, "")
return return
} }
// listHelper iterates over all items from the directory
// and calls the callback for each element.
func (f *Fs) listHelper(ctx context.Context, dir string, recursive bool, callback func(entries fs.DirEntry) error) (err error) {
directoryID, err := f.dirCache.FindDir(ctx, dir, false)
if err != nil {
return err
}
var iErr error
_, err = f.listAll(ctx, directoryID, false, false, recursive, func(info *api.Item) bool {
remote := path.Join(dir, info.Name)
if info.IsFolder {
// cache the directory ID for later lookups
f.dirCache.Put(remote, info.ID)
d := fs.NewDir(remote, info.ModTime()).SetID(info.ID)
// FIXME more info from dir?
iErr = callback(d)
} else {
o, err := f.newObjectWithInfo(ctx, remote, info)
if err != nil {
iErr = err
return true
}
iErr = callback(o)
}
if iErr != nil {
return true
}
return false
})
if err != nil {
return err
}
if iErr != nil {
return iErr
}
return nil
}
// List the objects and directories in dir into entries. The // List the objects and directories in dir into entries. The
// entries can be returned in any order but should be for a // entries can be returned in any order but should be for a
// complete directory. // complete directory.
@ -495,36 +541,24 @@ func (f *Fs) listAll(ctx context.Context, dirID string, directoriesOnly bool, fi
// This should return ErrDirNotFound if the directory isn't // This should return ErrDirNotFound if the directory isn't
// found. // found.
func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err error) { func (f *Fs) List(ctx context.Context, dir string) (entries fs.DirEntries, err error) {
directoryID, err := f.dirCache.FindDir(ctx, dir, false) err = f.listHelper(ctx, dir, false, func(o fs.DirEntry) error {
if err != nil {
return nil, err
}
var iErr error
_, err = f.listAll(ctx, directoryID, false, false, func(info *api.Item) bool {
remote := path.Join(dir, info.Name)
if info.IsFolder {
// cache the directory ID for later lookups
f.dirCache.Put(remote, info.ID)
d := fs.NewDir(remote, info.ModTime()).SetID(info.ID)
// FIXME more info from dir?
entries = append(entries, d)
} else {
o, err := f.newObjectWithInfo(ctx, remote, info)
if err != nil {
iErr = err
return true
}
entries = append(entries, o) entries = append(entries, o)
} return nil
return false })
return entries, err
}
// ListR lists the objects and directories of the Fs starting
// from dir recursively into out.
func (f *Fs) ListR(ctx context.Context, dir string, callback fs.ListRCallback) (err error) {
list := walk.NewListRHelper(callback)
err = f.listHelper(ctx, dir, true, func(o fs.DirEntry) error {
return list.Add(o)
}) })
if err != nil { if err != nil {
return nil, err return err
} }
if iErr != nil { return list.Flush()
return nil, iErr
}
return entries, nil
} }
// Creates from the parameters passed in a half finished Object which // Creates from the parameters passed in a half finished Object which

View file

@ -264,7 +264,7 @@ backends:
fastlist: true fastlist: true
- backend: "pcloud" - backend: "pcloud"
remote: "TestPcloud:" remote: "TestPcloud:"
fastlist: false fastlist: true
- backend: "webdav" - backend: "webdav"
remote: "TestWebdavNextcloud:" remote: "TestWebdavNextcloud:"
ignore: ignore: