From 4ba58884b1eaa739a5b51c98bd080345a8ff0fae Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood <nick@craig-wood.com> Date: Wed, 20 Dec 2017 11:49:42 +0000 Subject: [PATCH] webdav: decode multiple <s:propstat> more carefully - fixes nextcloud 12.0.4 For some reason nextcloud sends multiple propstat responses now, one with a 404 status. rclone was interpreting the last status and assuming the file was missing. --- webdav/api/types.go | 40 ++++++++++++++++++++++++++++++++++++++-- 1 file changed, 38 insertions(+), 2 deletions(-) diff --git a/webdav/api/types.go b/webdav/api/types.go index c856a3007..d7a69b273 100644 --- a/webdav/api/types.go +++ b/webdav/api/types.go @@ -25,8 +25,40 @@ type Response struct { } // Prop is the properties of a response +// +// This is a lazy way of decoding the multiple <s:propstat> in the +// response. +// +// The response might look like this +// +// <d:response> +// <d:href>/remote.php/webdav/Nextcloud%20Manual.pdf</d:href> +// <d:propstat> +// <d:prop> +// <d:getlastmodified>Tue, 19 Dec 2017 22:02:36 GMT</d:getlastmodified> +// <d:getcontentlength>4143665</d:getcontentlength> +// <d:resourcetype/> +// <d:getetag>"048d7be4437ff7deeae94db50ff3e209"</d:getetag> +// <d:getcontenttype>application/pdf</d:getcontenttype> +// </d:prop> +// <d:status>HTTP/1.1 200 OK</d:status> +// </d:propstat> +// <d:propstat> +// <d:prop> +// <d:quota-used-bytes/> +// <d:quota-available-bytes/> +// </d:prop> +// <d:status>HTTP/1.1 404 Not Found</d:status> +// </d:propstat> +// </d:response> +// +// So we elide the array of <d:propstat> and within that the array of +// <d:prop> into one struct. +// +// Note that status collects all the status values for which we just +// check the first is OK. type Prop struct { - Status string `xml:"DAV: status"` + Status []string `xml:"DAV: status"` Name string `xml:"DAV: prop>displayname,omitempty"` Type *xml.Name `xml:"DAV: prop>resourcetype>collection,omitempty"` Size int64 `xml:"DAV: prop>getcontentlength,omitempty"` @@ -38,7 +70,11 @@ var parseStatus = regexp.MustCompile(`^HTTP/[0-9.]+\s+(\d+)\s+(.*)$`) // StatusOK examines the Status and returns an OK flag func (p *Prop) StatusOK() bool { - match := parseStatus.FindStringSubmatch(p.Status) + // Assume OK if no statuses received + if len(p.Status) == 0 { + return true + } + match := parseStatus.FindStringSubmatch(p.Status[0]) if len(match) < 3 { return false }