1
mirror of https://github.com/rclone/rclone synced 2025-01-11 14:26:24 +01:00

fs: Fix parsing of .. when joining remotes - Fixes #4862

Before this fix setting an alias of `s3:bucket` then using `alias:..`
would use the current working directory!

This fix corrects the path parsing. This parsing is also used in
wrapping backends like crypt, chunker, union etc.

It does not allow looking above the root of the alias, so `alias:..`
now lists `s3:bucket` as you might expect if you did `cd /` then
`ls ..`.
This commit is contained in:
Nick Craig-Wood 2020-12-13 10:26:13 +00:00
parent e45716cac2
commit ea8d13d841
2 changed files with 95 additions and 40 deletions

View File

@ -102,25 +102,47 @@ func Split(remote string) (parent string, leaf string, err error) {
return remoteName + parent, leaf, nil return remoteName + parent, leaf, nil
} }
// JoinRootPath joins any number of path elements into a single path, adding a // Make filePath absolute so it can't read above the root
// separating slash if necessary. The result is Cleaned; in particular, func makeAbsolute(filePath string) string {
// all empty strings are ignored. leadingSlash := strings.HasPrefix(filePath, "/")
filePath = path.Join("/", filePath)
if !leadingSlash && strings.HasPrefix(filePath, "/") {
filePath = filePath[1:]
}
return filePath
}
// JoinRootPath joins filePath onto remote
// //
// If the first non empty element has a leading "//" this is preserved. // If the remote has a leading "//" this is preserved to allow Windows
// network paths to be used as remotes.
//
// If filePath is empty then remote will be returned.
// //
// If the path contains \ these will be converted to / on Windows. // If the path contains \ these will be converted to / on Windows.
func JoinRootPath(elem ...string) string { func JoinRootPath(remote, filePath string) string {
es := make([]string, len(elem)) remote = filepath.ToSlash(remote)
for i := range es { if filePath == "" {
es[i] = filepath.ToSlash(elem[i]) return remote
} }
for i, e := range es { filePath = filepath.ToSlash(filePath)
if e != "" { filePath = makeAbsolute(filePath)
if strings.HasPrefix(e, "//") { if strings.HasPrefix(remote, "//") {
return "/" + path.Clean(strings.Join(es[i:], "/")) return "/" + path.Join(remote, filePath)
} }
return path.Clean(strings.Join(es[i:], "/")) remoteName, remotePath, err := Parse(remote)
if err != nil {
// Couldn't parse so assume it is a path
remoteName = ""
remotePath = remote
}
remotePath = path.Join(remotePath, filePath)
if remoteName != "" {
remoteName += ":"
// if have remote: then normalise the remotePath
if remotePath == "." {
remotePath = ""
} }
} }
return "" return remoteName + remotePath
} }

View File

@ -130,33 +130,66 @@ func TestSplit(t *testing.T) {
} }
} }
} }
func TestJoinRootPath(t *testing.T) {
func TestMakeAbsolute(t *testing.T) {
for _, test := range []struct { for _, test := range []struct {
elements []string in string
want string want string
}{ }{
{nil, ""}, {"", ""},
{[]string{""}, ""}, {".", ""},
{[]string{"/"}, "/"}, {"/.", "/"},
{[]string{"/", "/"}, "/"}, {"../potato", "potato"},
{[]string{"/", "//"}, "/"}, {"/../potato", "/potato"},
{[]string{"/root", ""}, "/root"}, {"./../potato", "potato"},
{[]string{"/root", "/"}, "/root"}, {"//../potato", "/potato"},
{[]string{"/root", "//"}, "/root"}, {"././../potato", "potato"},
{[]string{"/a/b"}, "/a/b"}, {"././potato/../../onion", "onion"},
{[]string{"//", "/"}, "//"},
{[]string{"//server", "path"}, "//server/path"},
{[]string{"//server/sub", "path"}, "//server/sub/path"},
{[]string{"//server", "//path"}, "//server/path"},
{[]string{"//server/sub", "//path"}, "//server/sub/path"},
{[]string{"", "//", "/"}, "//"},
{[]string{"", "//server", "path"}, "//server/path"},
{[]string{"", "//server/sub", "path"}, "//server/sub/path"},
{[]string{"", "//server", "//path"}, "//server/path"},
{[]string{"", "//server/sub", "//path"}, "//server/sub/path"},
{[]string{"", filepath.FromSlash("//server/sub"), filepath.FromSlash("//path")}, "//server/sub/path"},
} { } {
got := JoinRootPath(test.elements...) got := makeAbsolute(test.in)
assert.Equal(t, test.want, got) assert.Equal(t, test.want, got, test)
}
}
func TestJoinRootPath(t *testing.T) {
for _, test := range []struct {
remote string
filePath string
want string
}{
{"", "", ""},
{"", "/", "/"},
{"/", "", "/"},
{"/", "/", "/"},
{"/", "//", "/"},
{"/root", "", "/root"},
{"/root", "/", "/root"},
{"/root", "//", "/root"},
{"/a/b", "", "/a/b"},
{"//", "/", "//"},
{"//server", "path", "//server/path"},
{"//server/sub", "path", "//server/sub/path"},
{"//server", "//path", "//server/path"},
{"//server/sub", "//path", "//server/sub/path"},
{"//", "/", "//"},
{"//server", "path", "//server/path"},
{"//server/sub", "path", "//server/sub/path"},
{"//server", "//path", "//server/path"},
{"//server/sub", "//path", "//server/sub/path"},
{filepath.FromSlash("//server/sub"), filepath.FromSlash("//path"), "//server/sub/path"},
{"s3:", "", "s3:"},
{"s3:", ".", "s3:"},
{"s3:.", ".", "s3:"},
{"s3:", "..", "s3:"},
{"s3:dir", "sub", "s3:dir/sub"},
{"s3:dir", "/sub", "s3:dir/sub"},
{"s3:dir", "./sub", "s3:dir/sub"},
{"s3:/dir", "/sub/", "s3:/dir/sub"},
{"s3:dir", "..", "s3:dir"},
{"s3:dir", "/..", "s3:dir"},
{"s3:dir", "/../", "s3:dir"},
} {
got := JoinRootPath(test.remote, test.filePath)
assert.Equal(t, test.want, got, test)
} }
} }