Files
phylum/server/internal/webdav/handler.go
2024-10-18 00:35:18 +05:30

163 lines
4.1 KiB
Go

package webdav
import (
"context"
"errors"
"io"
"net/http"
"strings"
"github.com/gin-gonic/gin"
"github.com/google/uuid"
"github.com/shroff/phylum/server/internal/api/auth"
"github.com/shroff/phylum/server/internal/core"
webdav "github.com/shroff/phylum/server/internal/webdav/impl"
"github.com/sirupsen/logrus"
)
type handler struct {
app *core.App
prefix string
}
func SetupHandler(r *gin.RouterGroup, app *core.App) {
logrus.Debug("Setting up WebDAV access at " + r.BasePath())
handler := &handler{
app: app,
prefix: r.BasePath(),
}
r.Use(auth.CreateBasicAuthHandler(app))
r.Handle("OPTIONS", "*path", handler.HandleRequest)
r.Handle("GET", "*path", handler.HandleRequest)
r.Handle("PUT", "*path", handler.HandleRequest)
r.Handle("HEAD", "*path", handler.HandleRequest)
r.Handle("POST", "*path", handler.HandleRequest)
r.Handle("DELETE", "*path", handler.HandleRequest)
r.Handle("MOVE", "*path", handler.HandleRequest)
r.Handle("COPY", "*path", handler.HandleRequest)
r.Handle("MKCOL", "*path", handler.HandleRequest)
r.Handle("PROPFIND", "*path", handler.HandleRequest)
r.Handle("PROPPATCH", "*path", handler.HandleRequest)
}
func (h *handler) HandleRequest(c *gin.Context) {
fs := auth.GetFileSystem(c)
if fs == nil {
c.AbortWithStatus(http.StatusInternalServerError)
return
}
webdavHandler := webdav.Handler{
Prefix: h.prefix,
FileSystem: adapter{fs: fs.(core.FileSystem)},
LockSystem: webdav.NewMemLS(),
}
webdavHandler.ServeHTTP(c.Writer, c.Request)
}
type adapter struct {
fs core.FileSystem
}
func (a adapter) Stat(ctx context.Context, name string) (core.Resource, error) {
return a.fs.ResourceByPath(name)
}
func (a adapter) OpenRead(r core.Resource, start, len int64) (io.ReadCloser, error) {
return a.fs.OpenRead(r, start, len)
}
func (a adapter) ReadDir(r core.Resource) ([]core.Resource, error) {
return a.fs.ReadDir(r)
}
func (a adapter) OpenWrite(ctx context.Context, name string) (io.WriteCloser, error) {
resource, err := a.Stat(ctx, name)
if err != nil {
if err == core.ErrResourceNotFound {
// Try to create the resource if the parent collection exists
name = strings.TrimRight(name, "/")
index := strings.LastIndex(name, "/")
parentPath := name[0:index]
parent, err := a.Stat(ctx, parentPath)
if err != nil {
return nil, err
}
resourceName := name[index+1:]
resourceId := uuid.New()
resource, err = a.fs.CreateMemberResource(parent, resourceId, resourceName, false)
if err != nil {
return nil, err
}
return a.fs.OpenWrite(resource)
}
return nil, err
}
if resource.Dir {
return nil, core.ErrResourceCollection
}
return a.fs.OpenWrite(resource)
}
func (a adapter) RemoveAll(ctx context.Context, name string) error {
resource, err := a.Stat(ctx, name)
if err != nil {
return err
}
_, err = a.fs.DeleteRecursive(resource, false)
return err
}
func (a adapter) Mkdir(ctx context.Context, name string) error {
if _, err := a.Stat(ctx, name); !errors.Is(err, core.ErrResourceNotFound) {
if err == nil {
return core.ErrResourceNameConflict
}
return err
}
name = strings.TrimRight(name, "/")
index := strings.LastIndex(name, "/")
parentPath := name[0:index]
parent, err := a.Stat(ctx, parentPath)
if err != nil {
if errors.Is(err, core.ErrResourceNotFound) {
return core.ErrParentNotFound
}
return err
}
dirName := name[index+1:]
_, err = a.fs.CreateMemberResource(parent, uuid.New(), dirName, true)
return err
}
func (a adapter) Rename(ctx context.Context, oldName, newName string) error {
src, err := a.Stat(ctx, oldName)
if err != nil {
return core.ErrResourceNotFound
}
_, err = a.Stat(ctx, newName)
if err == nil {
return core.ErrResourceNameConflict
}
newName = strings.TrimRight(newName, "/")
index := strings.LastIndex(newName, "/")
parentPath := newName[0:index]
newName = newName[index+1:]
parent, err := a.Stat(ctx, parentPath)
if err != nil {
if errors.Is(err, core.ErrResourceNotFound) {
return core.ErrParentNotFound
}
return err
}
parentID := parent.ID
if _, err = a.fs.Move(src, parentID, newName, false); err != nil {
return err
}
return nil
}