2020-01-17 10:30:42 +00:00
|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
2020-01-17 10:41:44 +00:00
|
|
|
"encoding/xml"
|
2020-01-17 10:30:42 +00:00
|
|
|
"fmt"
|
|
|
|
"mime"
|
2020-01-17 10:41:44 +00:00
|
|
|
"net/http"
|
2020-01-22 10:43:36 +00:00
|
|
|
"net/url"
|
2020-01-17 10:41:44 +00:00
|
|
|
"strings"
|
2020-01-17 10:30:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type HTTPError struct {
|
|
|
|
Code int
|
|
|
|
Err error
|
|
|
|
}
|
|
|
|
|
2020-01-17 13:40:29 +00:00
|
|
|
func HTTPErrorFromError(err error) *HTTPError {
|
|
|
|
if err == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if httpErr, ok := err.(*HTTPError); ok {
|
|
|
|
return httpErr
|
|
|
|
} else {
|
|
|
|
return &HTTPError{http.StatusInternalServerError, err}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-19 14:29:51 +00:00
|
|
|
func IsNotFound(err error) bool {
|
|
|
|
return HTTPErrorFromError(err).Code == http.StatusNotFound
|
|
|
|
}
|
|
|
|
|
2020-01-17 10:30:42 +00:00
|
|
|
func HTTPErrorf(code int, format string, a ...interface{}) *HTTPError {
|
|
|
|
return &HTTPError{code, fmt.Errorf(format, a...)}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (err *HTTPError) Error() string {
|
|
|
|
s := fmt.Sprintf("%v %v", err.Code, http.StatusText(err.Code))
|
|
|
|
if err.Err != nil {
|
|
|
|
return fmt.Sprintf("%v: %v", s, err.Err)
|
|
|
|
} else {
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-19 10:05:56 +00:00
|
|
|
func ServeError(w http.ResponseWriter, err error) {
|
|
|
|
code := http.StatusInternalServerError
|
|
|
|
if httpErr, ok := err.(*HTTPError); ok {
|
|
|
|
code = httpErr.Code
|
|
|
|
}
|
|
|
|
http.Error(w, err.Error(), code)
|
|
|
|
}
|
|
|
|
|
2020-01-19 10:12:45 +00:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
|
2020-01-17 10:30:42 +00:00
|
|
|
type Backend interface {
|
2020-01-29 17:03:47 +00:00
|
|
|
Options(r *http.Request) (caps []string, allow []string, err error)
|
2020-01-17 10:30:42 +00:00
|
|
|
HeadGet(w http.ResponseWriter, r *http.Request) error
|
|
|
|
Propfind(r *http.Request, pf *Propfind, depth Depth) (*Multistatus, error)
|
2020-01-21 22:18:27 +00:00
|
|
|
Proppatch(r *http.Request, pu *Propertyupdate) (*Response, error)
|
2020-01-21 20:46:01 +00:00
|
|
|
Put(r *http.Request) error
|
|
|
|
Delete(r *http.Request) error
|
2020-01-21 21:05:59 +00:00
|
|
|
Mkcol(r *http.Request) error
|
2020-01-22 12:00:42 +00:00
|
|
|
Copy(r *http.Request, dest *Href, recursive, overwrite bool) (created bool, err error)
|
2020-01-22 10:43:36 +00:00
|
|
|
Move(r *http.Request, dest *Href, overwrite bool) (created bool, err error)
|
2020-01-17 10:30:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
2020-01-21 20:19:44 +00:00
|
|
|
case http.MethodPut:
|
|
|
|
err = h.Backend.Put(r)
|
|
|
|
if err == nil {
|
|
|
|
// TODO: Last-Modified, ETag, Content-Type if the request has
|
|
|
|
// been copied verbatim
|
|
|
|
// TODO: Location if the server has mutated the href
|
|
|
|
w.WriteHeader(http.StatusCreated)
|
|
|
|
}
|
2020-01-21 20:46:01 +00:00
|
|
|
case http.MethodDelete:
|
|
|
|
// TODO: send a multistatus in case of partial failure
|
|
|
|
err = h.Backend.Delete(r)
|
|
|
|
if err == nil {
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
2020-01-17 10:30:42 +00:00
|
|
|
case "PROPFIND":
|
|
|
|
err = h.handlePropfind(w, r)
|
2020-01-21 22:18:27 +00:00
|
|
|
case "PROPPATCH":
|
|
|
|
err = h.handleProppatch(w, r)
|
2020-01-21 21:05:59 +00:00
|
|
|
case "MKCOL":
|
|
|
|
err = h.Backend.Mkcol(r)
|
|
|
|
if err == nil {
|
|
|
|
w.WriteHeader(http.StatusCreated)
|
|
|
|
}
|
2020-01-22 12:00:42 +00:00
|
|
|
case "COPY", "MOVE":
|
|
|
|
err = h.handleCopyMove(w, r)
|
2020-01-17 10:30:42 +00:00
|
|
|
default:
|
|
|
|
err = HTTPErrorf(http.StatusMethodNotAllowed, "webdav: unsupported method")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
code := http.StatusInternalServerError
|
|
|
|
if httpErr, ok := err.(*HTTPError); ok {
|
|
|
|
code = httpErr.Code
|
|
|
|
}
|
|
|
|
http.Error(w, err.Error(), code)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *Handler) handleOptions(w http.ResponseWriter, r *http.Request) error {
|
2020-01-29 17:03:47 +00:00
|
|
|
caps, allow, err := h.Backend.Options(r)
|
2020-01-17 10:41:44 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-29 17:03:47 +00:00
|
|
|
caps = append([]string{"1", "3"}, caps...)
|
2020-01-17 10:41:44 +00:00
|
|
|
|
2020-01-29 17:03:47 +00:00
|
|
|
w.Header().Add("DAV", strings.Join(caps, ", "))
|
|
|
|
w.Header().Add("Allow", strings.Join(allow, ", "))
|
2020-01-17 10:30:42 +00:00
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (h *Handler) handlePropfind(w http.ResponseWriter, r *http.Request) error {
|
|
|
|
var propfind Propfind
|
2020-01-19 10:12:45 +00:00
|
|
|
if err := DecodeXMLRequest(r, &propfind); err != nil {
|
|
|
|
return err
|
2020-01-17 10:30:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
depth := DepthInfinity
|
|
|
|
if s := r.Header.Get("Depth"); s != "" {
|
2020-01-19 10:05:56 +00:00
|
|
|
var err error
|
2020-01-17 10:30:42 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-01-19 10:12:45 +00:00
|
|
|
return ServeMultistatus(w, ms)
|
2020-01-17 10:30:42 +00:00
|
|
|
}
|
2020-01-17 13:40:29 +00:00
|
|
|
|
|
|
|
type PropfindFunc func(raw *RawXMLValue) (interface{}, error)
|
|
|
|
|
2020-01-22 10:07:30 +00:00
|
|
|
func NewPropfindResponse(path string, propfind *Propfind, props map[xml.Name]PropfindFunc) (*Response, error) {
|
|
|
|
resp := NewOKResponse(path)
|
2020-01-17 13:40:29 +00:00
|
|
|
|
2020-01-17 16:09:44 +00:00
|
|
|
if _, ok := props[ResourceTypeName]; !ok {
|
|
|
|
props[ResourceTypeName] = func(*RawXMLValue) (interface{}, error) {
|
|
|
|
return NewResourceType(), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-17 13:40:29 +00:00
|
|
|
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 {
|
2020-01-19 10:05:56 +00:00
|
|
|
return nil, HTTPErrorf(http.StatusBadRequest, "webdav: request missing propname, allprop or prop element")
|
2020-01-17 13:40:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return resp, nil
|
|
|
|
}
|
2020-01-21 22:18:27 +00:00
|
|
|
|
|
|
|
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)
|
|
|
|
}
|
2020-01-22 10:43:36 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2020-01-22 12:00:42 +00:00
|
|
|
func (h *Handler) handleCopyMove(w http.ResponseWriter, r *http.Request) error {
|
2020-01-22 10:43:36 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-22 12:00:42 +00:00
|
|
|
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)
|
|
|
|
}
|
2020-01-22 10:43:36 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-01-22 12:00:42 +00:00
|
|
|
|
2020-01-22 10:43:36 +00:00
|
|
|
if created {
|
|
|
|
w.WriteHeader(http.StatusCreated)
|
|
|
|
} else {
|
|
|
|
w.WriteHeader(http.StatusNoContent)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|