mirror of
https://github.com/1f349/go-webdav.git
synced 2024-12-23 00:34:23 +00:00
d7891ce50c
We were sometimes using TitleCase, sometimes Lowercase. Let's align on the idiomatic Go naming and pick TitleCase everywhere.
305 lines
7.3 KiB
Go
305 lines
7.3 KiB
Go
package internal
|
|
|
|
import (
|
|
"encoding/xml"
|
|
"errors"
|
|
"fmt"
|
|
"mime"
|
|
"net/http"
|
|
"net/url"
|
|
"strings"
|
|
)
|
|
|
|
func ServeError(w http.ResponseWriter, err error) {
|
|
code := http.StatusInternalServerError
|
|
var httpErr *HTTPError
|
|
if errors.As(err, &httpErr) {
|
|
code = httpErr.Code
|
|
}
|
|
|
|
var errElt *Error
|
|
if errors.As(err, &errElt) {
|
|
w.WriteHeader(code)
|
|
ServeXML(w).Encode(errElt)
|
|
return
|
|
}
|
|
|
|
http.Error(w, err.Error(), code)
|
|
}
|
|
|
|
func DecodeXMLRequest(r *http.Request, v interface{}) error {
|
|
t, _, _ := mime.ParseMediaType(r.Header.Get("Content-Type"))
|
|
if t != "application/xml" && t != "text/xml" {
|
|
return HTTPErrorf(http.StatusBadRequest, "webdav: expected application/xml request")
|
|
}
|
|
|
|
if err := xml.NewDecoder(r.Body).Decode(v); err != nil {
|
|
return &HTTPError{http.StatusBadRequest, err}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func ServeXML(w http.ResponseWriter) *xml.Encoder {
|
|
w.Header().Add("Content-Type", "text/xml; charset=\"utf-8\"")
|
|
w.Write([]byte(xml.Header))
|
|
return xml.NewEncoder(w)
|
|
}
|
|
|
|
func ServeMultiStatus(w http.ResponseWriter, ms *MultiStatus) error {
|
|
// TODO: streaming
|
|
w.WriteHeader(http.StatusMultiStatus)
|
|
return ServeXML(w).Encode(ms)
|
|
}
|
|
|
|
type Backend interface {
|
|
Options(r *http.Request) (caps []string, allow []string, err error)
|
|
HeadGet(w http.ResponseWriter, r *http.Request) error
|
|
PropFind(r *http.Request, pf *PropFind, depth Depth) (*MultiStatus, error)
|
|
PropPatch(r *http.Request, pu *PropertyUpdate) (*Response, error)
|
|
Put(r *http.Request) (*Href, error)
|
|
Delete(r *http.Request) error
|
|
Mkcol(r *http.Request) error
|
|
Copy(r *http.Request, dest *Href, recursive, overwrite bool) (created bool, err error)
|
|
Move(r *http.Request, dest *Href, overwrite bool) (created bool, err error)
|
|
}
|
|
|
|
type Handler struct {
|
|
Backend Backend
|
|
}
|
|
|
|
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
var err error
|
|
if h.Backend == nil {
|
|
err = fmt.Errorf("webdav: no backend available")
|
|
} else {
|
|
switch r.Method {
|
|
case http.MethodOptions:
|
|
err = h.handleOptions(w, r)
|
|
case http.MethodGet, http.MethodHead:
|
|
err = h.Backend.HeadGet(w, r)
|
|
case http.MethodPut:
|
|
var href *Href
|
|
href, err = h.Backend.Put(r)
|
|
if err == nil {
|
|
// TODO: Last-Modified, ETag, Content-Type if the request has
|
|
// been copied verbatim
|
|
if href != nil {
|
|
w.Header().Set("Location", (*url.URL)(href).String())
|
|
}
|
|
// TODO: http.StatusNoContent if the resource already existed
|
|
w.WriteHeader(http.StatusCreated)
|
|
}
|
|
case http.MethodDelete:
|
|
// TODO: send a multistatus in case of partial failure
|
|
err = h.Backend.Delete(r)
|
|
if err == nil {
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
case "PROPFIND":
|
|
err = h.handlePropfind(w, r)
|
|
case "PROPPATCH":
|
|
err = h.handleProppatch(w, r)
|
|
case "MKCOL":
|
|
err = h.Backend.Mkcol(r)
|
|
if err == nil {
|
|
w.WriteHeader(http.StatusCreated)
|
|
}
|
|
case "COPY", "MOVE":
|
|
err = h.handleCopyMove(w, r)
|
|
default:
|
|
err = HTTPErrorf(http.StatusMethodNotAllowed, "webdav: unsupported method")
|
|
}
|
|
}
|
|
|
|
if err != nil {
|
|
ServeError(w, err)
|
|
}
|
|
}
|
|
|
|
func (h *Handler) handleOptions(w http.ResponseWriter, r *http.Request) error {
|
|
caps, allow, err := h.Backend.Options(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
caps = append([]string{"1", "3"}, caps...)
|
|
|
|
w.Header().Add("DAV", strings.Join(caps, ", "))
|
|
w.Header().Add("Allow", strings.Join(allow, ", "))
|
|
w.WriteHeader(http.StatusNoContent)
|
|
return nil
|
|
}
|
|
|
|
func (h *Handler) handlePropfind(w http.ResponseWriter, r *http.Request) error {
|
|
var propfind PropFind
|
|
if err := DecodeXMLRequest(r, &propfind); err != nil {
|
|
return err
|
|
}
|
|
|
|
depth := DepthInfinity
|
|
if s := r.Header.Get("Depth"); s != "" {
|
|
var err error
|
|
depth, err = ParseDepth(s)
|
|
if err != nil {
|
|
return &HTTPError{http.StatusBadRequest, err}
|
|
}
|
|
}
|
|
|
|
ms, err := h.Backend.PropFind(r, &propfind, depth)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return ServeMultiStatus(w, ms)
|
|
}
|
|
|
|
type PropFindFunc func(raw *RawXMLValue) (interface{}, error)
|
|
|
|
func NewPropFindResponse(path string, propfind *PropFind, props map[xml.Name]PropFindFunc) (*Response, error) {
|
|
resp := NewOKResponse(path)
|
|
|
|
if _, ok := props[ResourceTypeName]; !ok {
|
|
props[ResourceTypeName] = func(*RawXMLValue) (interface{}, error) {
|
|
return NewResourceType(), nil
|
|
}
|
|
}
|
|
|
|
if propfind.PropName != nil {
|
|
for xmlName, _ := range props {
|
|
emptyVal := NewRawXMLElement(xmlName, nil, nil)
|
|
if err := resp.EncodeProp(http.StatusOK, emptyVal); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
} else if propfind.AllProp != nil {
|
|
// TODO: add support for propfind.Include
|
|
for xmlName, f := range props {
|
|
emptyVal := NewRawXMLElement(xmlName, nil, nil)
|
|
|
|
val, err := f(emptyVal)
|
|
|
|
code := http.StatusOK
|
|
if err != nil {
|
|
// TODO: don't throw away error message here
|
|
code = HTTPErrorFromError(err).Code
|
|
val = emptyVal
|
|
}
|
|
|
|
if err := resp.EncodeProp(code, val); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
} else if prop := propfind.Prop; prop != nil {
|
|
for _, raw := range prop.Raw {
|
|
xmlName, ok := raw.XMLName()
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
emptyVal := NewRawXMLElement(xmlName, nil, nil)
|
|
|
|
var code int
|
|
var val interface{} = emptyVal
|
|
f, ok := props[xmlName]
|
|
if ok {
|
|
if v, err := f(&raw); err != nil {
|
|
// TODO: don't throw away error message here
|
|
code = HTTPErrorFromError(err).Code
|
|
} else {
|
|
code = http.StatusOK
|
|
val = v
|
|
}
|
|
} else {
|
|
code = http.StatusNotFound
|
|
}
|
|
|
|
if err := resp.EncodeProp(code, val); err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
} else {
|
|
return nil, HTTPErrorf(http.StatusBadRequest, "webdav: request missing propname, allprop or prop element")
|
|
}
|
|
|
|
return resp, nil
|
|
}
|
|
|
|
func (h *Handler) handleProppatch(w http.ResponseWriter, r *http.Request) error {
|
|
var update PropertyUpdate
|
|
if err := DecodeXMLRequest(r, &update); err != nil {
|
|
return err
|
|
}
|
|
|
|
resp, err := h.Backend.PropPatch(r, &update)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
ms := NewMultiStatus(*resp)
|
|
return ServeMultiStatus(w, ms)
|
|
}
|
|
|
|
func parseDestination(h http.Header) (*Href, error) {
|
|
destHref := h.Get("Destination")
|
|
if destHref == "" {
|
|
return nil, HTTPErrorf(http.StatusBadRequest, "webdav: missing Destination header in MOVE request")
|
|
}
|
|
dest, err := url.Parse(destHref)
|
|
if err != nil {
|
|
return nil, HTTPErrorf(http.StatusBadRequest, "webdav: marlformed Destination header in MOVE request: %v", err)
|
|
}
|
|
return (*Href)(dest), nil
|
|
}
|
|
|
|
func (h *Handler) handleCopyMove(w http.ResponseWriter, r *http.Request) error {
|
|
dest, err := parseDestination(r.Header)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
overwrite := true
|
|
if s := r.Header.Get("Overwrite"); s != "" {
|
|
overwrite, err = ParseOverwrite(s)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
depth := DepthInfinity
|
|
if s := r.Header.Get("Depth"); s != "" {
|
|
depth, err = ParseDepth(s)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
var created bool
|
|
if r.Method == "COPY" {
|
|
var recursive bool
|
|
switch depth {
|
|
case DepthZero:
|
|
recursive = false
|
|
case DepthOne:
|
|
return HTTPErrorf(http.StatusBadRequest, `webdav: "Depth: 1" is not supported in COPY request`)
|
|
case DepthInfinity:
|
|
recursive = true
|
|
}
|
|
|
|
created, err = h.Backend.Copy(r, dest, recursive, overwrite)
|
|
} else {
|
|
if depth != DepthInfinity {
|
|
return HTTPErrorf(http.StatusBadRequest, `webdav: only "Depth: infinity" is accepted in MOVE request`)
|
|
}
|
|
created, err = h.Backend.Move(r, dest, overwrite)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if created {
|
|
w.WriteHeader(http.StatusCreated)
|
|
} else {
|
|
w.WriteHeader(http.StatusNoContent)
|
|
}
|
|
return nil
|
|
}
|