go-webdav/server.go

311 lines
8.6 KiB
Go
Raw Normal View History

package webdav
import (
"encoding/xml"
"io"
"net/http"
"os"
"strconv"
2022-05-03 16:53:06 +01:00
"strings"
"github.com/emersion/go-webdav/internal"
)
2020-01-21 20:01:18 +00:00
// FileSystem is a WebDAV server backend.
type FileSystem interface {
Open(name string) (io.ReadCloser, error)
Stat(name string) (*FileInfo, error)
Readdir(name string, recursive bool) ([]FileInfo, error)
2020-01-21 20:19:44 +00:00
Create(name string) (io.WriteCloser, error)
2020-01-21 20:46:01 +00:00
RemoveAll(name string) error
2020-01-21 21:05:59 +00:00
Mkdir(name string) error
2020-01-22 12:00:42 +00:00
Copy(name, dest string, recursive, overwrite bool) (created bool, err error)
2020-01-22 10:43:36 +00:00
MoveAll(name, dest string, overwrite bool) (created bool, err error)
}
2020-01-21 20:01:18 +00:00
// Handler handles WebDAV HTTP requests. It can be used to create a WebDAV
// server.
type Handler struct {
FileSystem FileSystem
}
2020-01-21 20:01:18 +00:00
// ServeHTTP implements http.Handler.
func (h *Handler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
if h.FileSystem == nil {
2020-01-17 10:30:42 +00:00
http.Error(w, "webdav: no filesystem available", http.StatusInternalServerError)
return
}
2020-01-17 10:30:42 +00:00
b := backend{h.FileSystem}
hh := internal.Handler{&b}
hh.ServeHTTP(w, r)
}
2020-01-17 10:30:42 +00:00
type backend struct {
FileSystem FileSystem
}
func (b *backend) Options(r *http.Request) (caps []string, allow []string, err error) {
fi, err := b.FileSystem.Stat(r.URL.Path)
2020-01-17 10:41:44 +00:00
if os.IsNotExist(err) {
return nil, []string{http.MethodOptions, http.MethodPut, "MKCOL"}, nil
2020-01-17 10:41:44 +00:00
} else if err != nil {
return nil, nil, err
2020-01-17 10:41:44 +00:00
}
allow = []string{
http.MethodOptions,
http.MethodDelete,
"PROPFIND",
"COPY",
"MOVE",
}
if !fi.IsDir {
allow = append(allow, http.MethodHead, http.MethodGet, http.MethodPut)
2020-01-17 10:41:44 +00:00
}
return nil, allow, nil
2020-01-17 10:41:44 +00:00
}
2020-01-17 10:30:42 +00:00
func (b *backend) HeadGet(w http.ResponseWriter, r *http.Request) error {
fi, err := b.FileSystem.Stat(r.URL.Path)
if os.IsNotExist(err) {
return &internal.HTTPError{Code: http.StatusNotFound, Err: err}
} else if err != nil {
return err
}
if fi.IsDir {
return &internal.HTTPError{Code: http.StatusMethodNotAllowed}
}
f, err := b.FileSystem.Open(r.URL.Path)
if err != nil {
return err
}
defer f.Close()
2020-01-17 10:32:13 +00:00
w.Header().Set("Content-Length", strconv.FormatInt(fi.Size, 10))
2020-01-21 21:43:13 +00:00
if fi.MIMEType != "" {
w.Header().Set("Content-Type", fi.MIMEType)
}
if !fi.ModTime.IsZero() {
w.Header().Set("Last-Modified", fi.ModTime.UTC().Format(http.TimeFormat))
}
if fi.ETag != "" {
2020-02-03 20:48:31 +00:00
w.Header().Set("ETag", internal.ETag(fi.ETag).String())
}
2020-01-21 21:43:13 +00:00
if rs, ok := f.(io.ReadSeeker); ok {
// If it's an io.Seeker, use http.ServeContent which supports ranges
http.ServeContent(w, r, r.URL.Path, fi.ModTime, rs)
} else {
if r.Method != http.MethodHead {
io.Copy(w, f)
}
}
return nil
}
2020-01-17 10:30:42 +00:00
func (b *backend) Propfind(r *http.Request, propfind *internal.Propfind, depth internal.Depth) (*internal.Multistatus, error) {
// TODO: use partial error Response on error
fi, err := b.FileSystem.Stat(r.URL.Path)
if os.IsNotExist(err) {
return nil, &internal.HTTPError{Code: http.StatusNotFound, Err: err}
} else if err != nil {
2020-01-17 10:30:42 +00:00
return nil, err
}
2020-01-15 17:39:25 +00:00
var resps []internal.Response
if depth != internal.DepthZero && fi.IsDir {
children, err := b.FileSystem.Readdir(r.URL.Path, depth == internal.DepthInfinity)
2020-01-15 17:39:25 +00:00
if err != nil {
return nil, err
2020-01-15 17:39:25 +00:00
}
resps = make([]internal.Response, len(children))
for i, child := range children {
resp, err := b.propfindFile(propfind, &child)
if err != nil {
return nil, err
2020-01-15 17:39:25 +00:00
}
resps[i] = *resp
2020-01-15 17:39:25 +00:00
}
} else {
resp, err := b.propfindFile(propfind, fi)
if err != nil {
return nil, err
}
resps = []internal.Response{*resp}
2020-01-15 17:39:25 +00:00
}
return internal.NewMultistatus(resps...), nil
2020-01-15 17:39:25 +00:00
}
func (b *backend) propfindFile(propfind *internal.Propfind, fi *FileInfo) (*internal.Response, error) {
props := make(map[xml.Name]internal.PropfindFunc)
2020-01-17 16:09:23 +00:00
props[internal.ResourceTypeName] = func(*internal.RawXMLValue) (interface{}, error) {
var types []xml.Name
if fi.IsDir {
types = append(types, internal.CollectionName)
}
return internal.NewResourceType(types...), nil
}
if !fi.IsDir {
2020-01-17 16:09:23 +00:00
props[internal.GetContentLengthName] = func(*internal.RawXMLValue) (interface{}, error) {
return &internal.GetContentLength{Length: fi.Size}, nil
2020-01-15 18:08:38 +00:00
}
2020-01-21 21:44:10 +00:00
if !fi.ModTime.IsZero() {
props[internal.GetLastModifiedName] = func(*internal.RawXMLValue) (interface{}, error) {
return &internal.GetLastModified{LastModified: internal.Time(fi.ModTime)}, nil
}
2020-01-15 18:08:38 +00:00
}
2020-01-21 21:43:13 +00:00
if fi.MIMEType != "" {
props[internal.GetContentTypeName] = func(*internal.RawXMLValue) (interface{}, error) {
return &internal.GetContentType{Type: fi.MIMEType}, nil
}
}
if fi.ETag != "" {
props[internal.GetETagName] = func(*internal.RawXMLValue) (interface{}, error) {
2020-02-03 20:48:31 +00:00
return &internal.GetETag{ETag: internal.ETag(fi.ETag)}, nil
}
}
}
return internal.NewPropfindResponse(fi.Path, propfind, props)
}
2020-01-21 20:46:01 +00:00
func (b *backend) Proppatch(r *http.Request, update *internal.Propertyupdate) (*internal.Response, error) {
// TODO: return a failed Response instead
return nil, internal.HTTPErrorf(http.StatusForbidden, "webdav: PROPPATCH is unsupported")
}
func (b *backend) Put(r *http.Request) (*internal.Href, error) {
2020-01-21 20:46:01 +00:00
wc, err := b.FileSystem.Create(r.URL.Path)
if err != nil {
return nil, err
2020-01-21 20:46:01 +00:00
}
defer wc.Close()
if _, err := io.Copy(wc, r.Body); err != nil {
return nil, err
2020-01-21 20:46:01 +00:00
}
return nil, wc.Close()
2020-01-21 20:46:01 +00:00
}
func (b *backend) Delete(r *http.Request) error {
err := b.FileSystem.RemoveAll(r.URL.Path)
2020-01-21 20:46:01 +00:00
if os.IsNotExist(err) {
return &internal.HTTPError{Code: http.StatusNotFound, Err: err}
}
return err
2020-01-21 20:46:01 +00:00
}
2020-01-21 21:05:59 +00:00
func (b *backend) Mkcol(r *http.Request) error {
if r.Header.Get("Content-Type") != "" {
return internal.HTTPErrorf(http.StatusUnsupportedMediaType, "webdav: request body not supported in MKCOL request")
}
err := b.FileSystem.Mkdir(r.URL.Path)
if os.IsNotExist(err) {
return &internal.HTTPError{Code: http.StatusConflict, Err: err}
}
return err
}
2020-01-22 10:43:36 +00:00
2020-01-22 12:00:42 +00:00
func (b *backend) Copy(r *http.Request, dest *internal.Href, recursive, overwrite bool) (created bool, err error) {
created, err = b.FileSystem.Copy(r.URL.Path, dest.Path, recursive, overwrite)
if os.IsExist(err) {
return false, &internal.HTTPError{http.StatusPreconditionFailed, err}
}
return created, err
}
2020-01-22 10:43:36 +00:00
func (b *backend) Move(r *http.Request, dest *internal.Href, overwrite bool) (created bool, err error) {
created, err = b.FileSystem.MoveAll(r.URL.Path, dest.Path, overwrite)
if os.IsExist(err) {
return false, &internal.HTTPError{http.StatusPreconditionFailed, err}
}
return created, err
}
2022-05-03 16:53:06 +01:00
type ServePrincipalOptions struct {
Principal *Principal
Props map[xml.Name]xml.Marshaler
}
2022-05-03 16:53:06 +01:00
// ServePrincipal replies to a request on a principal URI.
func ServePrincipal(w http.ResponseWriter, r *http.Request, options *ServePrincipalOptions) {
2022-05-03 16:53:06 +01:00
switch r.Method {
case http.MethodOptions:
caps := append([]string{"1", "3"})
allow := []string{http.MethodOptions, "PROPFIND"}
w.Header().Add("DAV", strings.Join(caps, ", "))
w.Header().Add("Allow", strings.Join(allow, ", "))
w.WriteHeader(http.StatusNoContent)
case "PROPFIND":
if err := servePrincipalPropfind(w, r, options); err != nil {
2022-05-03 16:53:06 +01:00
internal.ServeError(w, err)
}
default:
http.Error(w, "unsupported method", http.StatusMethodNotAllowed)
}
}
func servePrincipalPropfind(w http.ResponseWriter, r *http.Request, options *ServePrincipalOptions) error {
principal := options.Principal
2022-05-03 16:53:06 +01:00
var propfind internal.Propfind
if err := internal.DecodeXMLRequest(r, &propfind); err != nil {
return err
}
// TODO: handle Depth
props := map[xml.Name]internal.PropfindFunc{
internal.ResourceTypeName: func(*internal.RawXMLValue) (interface{}, error) {
return internal.NewResourceType(principalName), nil
},
internal.DisplayNameName: func(*internal.RawXMLValue) (interface{}, error) {
return &internal.DisplayName{Name: principal.Name}, nil
},
internal.CurrentUserPrincipalName: func(*internal.RawXMLValue) (interface{}, error) {
// TODO: allow serving a principal different from the current user's
return &internal.CurrentUserPrincipal{Href: internal.Href{Path: principal.Path}}, nil
},
principalAlternateURISetName: func(*internal.RawXMLValue) (interface{}, error) {
return &principalAlternateURISet{}, nil // TODO: allow customizing
},
principalURLName: func(*internal.RawXMLValue) (interface{}, error) {
return &principalURL{
Href: internal.Href{Path: principal.Path},
}, nil
},
groupMembershipName: func(*internal.RawXMLValue) (interface{}, error) {
return &groupMembership{}, nil // TODO: allow customizing
},
}
for name, v := range options.Props {
v := v // capture variable
props[name] = func(*internal.RawXMLValue) (interface{}, error) {
return v, nil
}
}
2022-05-03 16:53:06 +01:00
resp, err := internal.NewPropfindResponse(r.URL.Path, &propfind, props)
if err != nil {
return err
}
ms := internal.NewMultistatus(*resp)
return internal.ServeMultistatus(w, ms)
}