webdav: add --webdav-base-path flag for unusual servers

See: https://forum.rclone.org/t/how-to-specify-a-webdav-backend-root-path-using-connection-string-syntax/31092
This commit is contained in:
Nick Craig-Wood 2022-06-08 14:31:25 +01:00
parent 26db80c270
commit dfc5b0460b

View File

@ -124,6 +124,22 @@ You can set multiple headers, e.g. '"Cookie","name=value","Authorization","xxx"'
`, `,
Default: fs.CommaSepList{}, Default: fs.CommaSepList{},
Advanced: true, Advanced: true,
}, {
Name: "base_path",
Help: `Base path of expected replies
Normally WebDAV servers return the files they are listing under the
url path as specified above. However some WebDAV servers return files
with URLs that are not under the endpoint URL. This causes rclone to
get confused and return errors like
Item with unknown path received: "/remote.php/webdav/folder1/", "/elsewhere/remote.php/webdav/folder1/"
errors. If that is the case, then set "base_path" to the path
specified in the error message up to the first item, in the above
example "/elsewhere/remote.php/webdav/".
`,
Advanced: true,
}}, }},
}) })
} }
@ -138,6 +154,7 @@ type Options struct {
BearerTokenCommand string `config:"bearer_token_command"` BearerTokenCommand string `config:"bearer_token_command"`
Enc encoder.MultiEncoder `config:"encoding"` Enc encoder.MultiEncoder `config:"encoding"`
Headers fs.CommaSepList `config:"headers"` Headers fs.CommaSepList `config:"headers"`
BasePath string `config:"base_path"`
} }
// Fs represents a remote webdav // Fs represents a remote webdav
@ -693,6 +710,10 @@ func (f *Fs) listAll(ctx context.Context, dir string, directoriesOnly bool, file
if err != nil { if err != nil {
return false, fmt.Errorf("couldn't join URL: %w", err) return false, fmt.Errorf("couldn't join URL: %w", err)
} }
basePath := baseURL.Path
if f.opt.BasePath != "" {
basePath = f.opt.BasePath
}
for i := range result.Responses { for i := range result.Responses {
item := &result.Responses[i] item := &result.Responses[i]
isDir := itemIsDir(item) isDir := itemIsDir(item)
@ -707,11 +728,11 @@ func (f *Fs) listAll(ctx context.Context, dir string, directoriesOnly bool, file
if isDir { if isDir {
u.Path = addSlash(u.Path) u.Path = addSlash(u.Path)
} }
if !strings.HasPrefix(u.Path, baseURL.Path) { if !strings.HasPrefix(u.Path, basePath) {
fs.Debugf(nil, "Item with unknown path received: %q, %q", u.Path, baseURL.Path) fs.Debugf(nil, "Item with unknown path received: %q, %q", u.Path, basePath)
continue continue
} }
subPath := u.Path[len(baseURL.Path):] subPath := u.Path[len(basePath):]
if f.opt.Enc != encoder.EncodeZero { if f.opt.Enc != encoder.EncodeZero {
subPath = f.opt.Enc.ToStandardPath(subPath) subPath = f.opt.Enc.ToStandardPath(subPath)
} }