Merge pull request #183 from jawr/walker_path_fix

Rework walker.Next
This commit is contained in:
Julien Laffaye 2020-07-28 14:05:16 -04:00 committed by GitHub
commit a3536fc55e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 38 additions and 26 deletions

View File

@ -1,10 +1,7 @@
package ftp package ftp
import ( import (
"fmt"
"os"
"path" "path"
"strings"
) )
//Walker traverses the directory tree of a remote FTP server //Walker traverses the directory tree of a remote FTP server
@ -26,38 +23,27 @@ type item struct {
// which will then be available through the Path, Stat, and Err methods. // which will then be available through the Path, Stat, and Err methods.
// It returns false when the walk stops at the end of the tree. // It returns false when the walk stops at the end of the tree.
func (w *Walker) Next() bool { func (w *Walker) Next() bool {
var isRoot bool // check if we need to init cur, maybe this should be inside Walk
if w.cur == nil { if w.cur == nil {
w.cur = &item{ w.cur = &item{
path: strings.Trim(w.root, string(os.PathSeparator)), path: w.root,
entry: &Entry{
Type: EntryTypeFolder,
},
} }
isRoot = true
} }
entries, err := w.serverConn.List(w.cur.path) if w.descend && w.cur.entry.Type == EntryTypeFolder {
w.cur.err = err entries, err := w.serverConn.List(w.cur.path)
if err == nil {
if len(entries) == 0 {
w.cur.err = fmt.Errorf("no such file or directory: %s", w.cur.path)
// an error occured, drop out and stop walking
if err != nil {
w.cur.err = err
return false return false
} }
if isRoot && len(entries) == 1 && entries[0].Type == EntryTypeFile { for _, entry := range entries {
w.cur.err = fmt.Errorf("root is not a directory: %s", w.cur.path) if entry.Name == "." || entry.Name == ".." {
return false
}
for i, entry := range entries {
if entry.Name == "." || (i == 0 && entry.Type == EntryTypeFile) {
entry.Name = path.Base(w.cur.path)
w.cur.entry = entry
continue
}
if entry.Name == ".." || !w.descend {
continue continue
} }
@ -74,9 +60,12 @@ func (w *Walker) Next() bool {
return false return false
} }
// update cur
i := len(w.stack) - 1 i := len(w.stack) - 1
w.cur = w.stack[i] w.cur = w.stack[i]
w.stack = w.stack[:i] w.stack = w.stack[:i]
// reset SkipDir
w.descend = true w.descend = true
return true return true

View File

@ -184,3 +184,26 @@ func TestCurAndStackSetCorrectly(t *testing.T) {
assert.Equal(0, len(w.stack)) assert.Equal(0, len(w.stack))
assert.Equal("file", w.cur.entry.Name) assert.Equal("file", w.cur.entry.Name)
} }
func TestCurInit(t *testing.T) {
mock, err := newFtpMock(t, "127.0.0.1")
if err != nil {
t.Fatal(err)
}
defer mock.Close()
c, cErr := Connect(mock.Addr())
if cErr != nil {
t.Fatal(err)
}
w := c.Walk("/root")
result := w.Next()
// mock fs has one file 'lo'
assert.Equal(t, true, result, "Result should return false")
assert.Equal(t, 0, len(w.stack))
assert.Equal(t, "/root/lo", w.Path())
}