mirror of
https://github.com/rclone/rclone.git
synced 2025-01-22 14:19:56 +01:00
serve webdav: implement owncloud checksum and modtime extensions
* implement owncloud checksum and modtime extensions for webdav server * test rclone webdav server as owncloud webdav
This commit is contained in:
parent
1f887f7ba0
commit
ceb9406c2f
@ -3,10 +3,12 @@ package webdav
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"encoding/xml"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -255,6 +257,52 @@ func (w *WebDAV) auth(user, pass string) (value interface{}, err error) {
|
|||||||
return VFS, err
|
return VFS, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type webdavRW struct {
|
||||||
|
http.ResponseWriter
|
||||||
|
status int
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rw *webdavRW) WriteHeader(statusCode int) {
|
||||||
|
rw.status = statusCode
|
||||||
|
rw.ResponseWriter.WriteHeader(statusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rw *webdavRW) isSuccessfull() bool {
|
||||||
|
return rw.status == 0 || (rw.status >= 200 && rw.status <= 299)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (w *WebDAV) postprocess(r *http.Request, remote string) {
|
||||||
|
// set modtime from requests, don't write to client because status is already written
|
||||||
|
switch r.Method {
|
||||||
|
case "COPY", "MOVE", "PUT":
|
||||||
|
VFS, err := w.getVFS(r.Context())
|
||||||
|
if err != nil {
|
||||||
|
fs.Errorf(nil, "Failed to get VFS: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the node
|
||||||
|
node, err := VFS.Stat(remote)
|
||||||
|
if err != nil {
|
||||||
|
fs.Errorf(nil, "Failed to stat node: %v", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
mh := r.Header.Get("X-OC-Mtime")
|
||||||
|
if mh != "" {
|
||||||
|
modtimeUnix, err := strconv.ParseInt(mh, 10, 64)
|
||||||
|
if err == nil {
|
||||||
|
err = node.SetModTime(time.Unix(modtimeUnix, 0))
|
||||||
|
if err != nil {
|
||||||
|
fs.Errorf(nil, "Failed to set modtime: %v", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fs.Errorf(nil, "Failed to parse modtime: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (w *WebDAV) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
|
func (w *WebDAV) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
|
||||||
urlPath := r.URL.Path
|
urlPath := r.URL.Path
|
||||||
isDir := strings.HasSuffix(urlPath, "/")
|
isDir := strings.HasSuffix(urlPath, "/")
|
||||||
@ -266,7 +314,12 @@ func (w *WebDAV) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
|
|||||||
// Add URL Prefix back to path since webdavhandler needs to
|
// Add URL Prefix back to path since webdavhandler needs to
|
||||||
// return absolute references.
|
// return absolute references.
|
||||||
r.URL.Path = w.opt.HTTP.BaseURL + r.URL.Path
|
r.URL.Path = w.opt.HTTP.BaseURL + r.URL.Path
|
||||||
w.webdavhandler.ServeHTTP(rw, r)
|
wrw := &webdavRW{ResponseWriter: rw}
|
||||||
|
w.webdavhandler.ServeHTTP(wrw, r)
|
||||||
|
|
||||||
|
if wrw.isSuccessfull() {
|
||||||
|
w.postprocess(r, remote)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// serveDir serves a directory index at dirRemote
|
// serveDir serves a directory index at dirRemote
|
||||||
@ -356,7 +409,7 @@ func (w *WebDAV) OpenFile(ctx context.Context, name string, flags int, perm os.F
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return Handle{Handle: f, w: w}, nil
|
return Handle{Handle: f, w: w, ctx: ctx}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// RemoveAll removes a file or a directory and its contents
|
// RemoveAll removes a file or a directory and its contents
|
||||||
@ -404,7 +457,8 @@ func (w *WebDAV) Stat(ctx context.Context, name string) (fi os.FileInfo, err err
|
|||||||
// Handle represents an open file
|
// Handle represents an open file
|
||||||
type Handle struct {
|
type Handle struct {
|
||||||
vfs.Handle
|
vfs.Handle
|
||||||
w *WebDAV
|
w *WebDAV
|
||||||
|
ctx context.Context
|
||||||
}
|
}
|
||||||
|
|
||||||
// Readdir reads directory entries from the handle
|
// Readdir reads directory entries from the handle
|
||||||
@ -429,6 +483,65 @@ func (h Handle) Stat() (fi os.FileInfo, err error) {
|
|||||||
return FileInfo{FileInfo: fi, w: h.w}, nil
|
return FileInfo{FileInfo: fi, w: h.w}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DeadProps returns extra properties about the handle
|
||||||
|
func (h Handle) DeadProps() (map[xml.Name]webdav.Property, error) {
|
||||||
|
var (
|
||||||
|
xmlName xml.Name
|
||||||
|
property webdav.Property
|
||||||
|
properties = make(map[xml.Name]webdav.Property)
|
||||||
|
)
|
||||||
|
if h.w.opt.HashType != hash.None {
|
||||||
|
entry := h.Handle.Node().DirEntry()
|
||||||
|
if o, ok := entry.(fs.Object); ok {
|
||||||
|
hash, err := o.Hash(h.ctx, h.w.opt.HashType)
|
||||||
|
if err == nil {
|
||||||
|
xmlName.Space = "http://owncloud.org/ns"
|
||||||
|
xmlName.Local = "checksums"
|
||||||
|
property.XMLName = xmlName
|
||||||
|
property.InnerXML = append(property.InnerXML, "<checksum xmlns=\"http://owncloud.org/ns\">"...)
|
||||||
|
property.InnerXML = append(property.InnerXML, strings.ToUpper(h.w.opt.HashType.String())...)
|
||||||
|
property.InnerXML = append(property.InnerXML, ':')
|
||||||
|
property.InnerXML = append(property.InnerXML, hash...)
|
||||||
|
property.InnerXML = append(property.InnerXML, "</checksum>"...)
|
||||||
|
properties[xmlName] = property
|
||||||
|
} else {
|
||||||
|
fs.Errorf(nil, "failed to calculate hash: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
xmlName.Space = "DAV:"
|
||||||
|
xmlName.Local = "lastmodified"
|
||||||
|
property.XMLName = xmlName
|
||||||
|
property.InnerXML = strconv.AppendInt(nil, h.Handle.Node().ModTime().Unix(), 10)
|
||||||
|
properties[xmlName] = property
|
||||||
|
|
||||||
|
return properties, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Patch changes modtime of the underlying resources, it returns ok for all properties, the error is from setModtime if any
|
||||||
|
// FIXME does not check for invalid property and SetModTime error
|
||||||
|
func (h Handle) Patch(proppatches []webdav.Proppatch) ([]webdav.Propstat, error) {
|
||||||
|
var (
|
||||||
|
stat webdav.Propstat
|
||||||
|
err error
|
||||||
|
)
|
||||||
|
stat.Status = http.StatusOK
|
||||||
|
for _, patch := range proppatches {
|
||||||
|
for _, prop := range patch.Props {
|
||||||
|
stat.Props = append(stat.Props, webdav.Property{XMLName: prop.XMLName})
|
||||||
|
if prop.XMLName.Space == "DAV:" && prop.XMLName.Local == "lastmodified" {
|
||||||
|
var modtimeUnix int64
|
||||||
|
modtimeUnix, err = strconv.ParseInt(string(prop.InnerXML), 10, 64)
|
||||||
|
if err == nil {
|
||||||
|
err = h.Handle.Node().SetModTime(time.Unix(modtimeUnix, 0))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return []webdav.Propstat{stat}, err
|
||||||
|
}
|
||||||
|
|
||||||
// FileInfo represents info about a file satisfying os.FileInfo and
|
// FileInfo represents info about a file satisfying os.FileInfo and
|
||||||
// also some additional interfaces for webdav for ETag and ContentType
|
// also some additional interfaces for webdav for ETag and ContentType
|
||||||
type FileInfo struct {
|
type FileInfo struct {
|
||||||
|
@ -65,7 +65,7 @@ func TestWebDav(t *testing.T) {
|
|||||||
// Config for the backend we'll use to connect to the server
|
// Config for the backend we'll use to connect to the server
|
||||||
config := configmap.Simple{
|
config := configmap.Simple{
|
||||||
"type": "webdav",
|
"type": "webdav",
|
||||||
"vendor": "other",
|
"vendor": "owncloud",
|
||||||
"url": w.Server.URLs()[0],
|
"url": w.Server.URLs()[0],
|
||||||
"user": testUser,
|
"user": testUser,
|
||||||
"pass": obscure.MustObscure(testPass),
|
"pass": obscure.MustObscure(testPass),
|
||||||
|
Loading…
Reference in New Issue
Block a user