ftp: Fix no error on listing non-existent directory

This commit is contained in:
Nick Craig-Wood 2018-04-20 23:08:08 +01:00
parent 9fbc40c5b9
commit da7a77ef2e

View File

@ -266,9 +266,8 @@ func translateErrorDir(err error) error {
return err return err
} }
// NewObject finds the Object at remote. If it can't be found // findItem finds a directory entry for the name in its parent directory
// it returns the error fs.ErrorObjectNotFound. func (f *Fs) findItem(remote string) (entry *ftp.Entry, err error) {
func (f *Fs) NewObject(remote string) (o fs.Object, err error) {
// defer fs.Trace(remote, "")("o=%v, err=%v", &o, &err) // defer fs.Trace(remote, "")("o=%v, err=%v", &o, &err)
fullPath := path.Join(f.root, remote) fullPath := path.Join(f.root, remote)
dir := path.Dir(fullPath) dir := path.Dir(fullPath)
@ -276,32 +275,58 @@ func (f *Fs) NewObject(remote string) (o fs.Object, err error) {
c, err := f.getFtpConnection() c, err := f.getFtpConnection()
if err != nil { if err != nil {
return nil, errors.Wrap(err, "NewObject") return nil, errors.Wrap(err, "findItem")
} }
files, err := c.List(dir) files, err := c.List(dir)
f.putFtpConnection(&c, err) f.putFtpConnection(&c, err)
if err != nil { if err != nil {
return nil, translateErrorFile(err) return nil, translateErrorFile(err)
} }
for i, file := range files { for _, file := range files {
if file.Type != ftp.EntryTypeFolder && file.Name == base { if file.Name == base {
o := &Object{ return file, nil
fs: f,
remote: remote,
}
info := &FileInfo{
Name: remote,
Size: files[i].Size,
ModTime: files[i].Time,
}
o.info = info
return o, nil
} }
} }
return nil, nil
}
// NewObject finds the Object at remote. If it can't be found
// it returns the error fs.ErrorObjectNotFound.
func (f *Fs) NewObject(remote string) (o fs.Object, err error) {
// defer fs.Trace(remote, "")("o=%v, err=%v", &o, &err)
entry, err := f.findItem(remote)
if err != nil {
return nil, errors.Wrap(err, "NewObject")
}
if entry != nil && entry.Type != ftp.EntryTypeFolder {
o := &Object{
fs: f,
remote: remote,
}
info := &FileInfo{
Name: remote,
Size: entry.Size,
ModTime: entry.Time,
}
o.info = info
return o, nil
}
return nil, fs.ErrorObjectNotFound return nil, fs.ErrorObjectNotFound
} }
// dirExists checks the directory pointed to by remote exists or not
func (f *Fs) dirExists(remote string) (exists bool, err error) {
entry, err := f.findItem(remote)
if err != nil {
return false, errors.Wrap(err, "dirExists")
}
if entry != nil && entry.Type == ftp.EntryTypeFolder {
return true, nil
}
return false, 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.
@ -322,6 +347,18 @@ func (f *Fs) List(dir string) (entries fs.DirEntries, err error) {
if err != nil { if err != nil {
return nil, translateErrorDir(err) return nil, translateErrorDir(err)
} }
// Annoyingly FTP returns success for a directory which
// doesn't exist, so check it really doesn't exist if no
// entries found.
if len(files) == 0 {
exists, err := f.dirExists(dir)
if err != nil {
return nil, errors.Wrap(err, "list")
}
if !exists {
return nil, fs.ErrorDirNotFound
}
}
for i := range files { for i := range files {
object := files[i] object := files[i]
newremote := path.Join(dir, object.Name) newremote := path.Join(dir, object.Name)