mirror of https://github.com/Xhofe/alist
				
				
				
			
		
			
				
	
	
		
			133 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
			
		
		
	
	
			133 lines
		
	
	
		
			3.6 KiB
		
	
	
	
		
			Go
		
	
	
| package server
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"crypto/subtle"
 | |
| 	"github.com/alist-org/alist/v3/internal/stream"
 | |
| 	"github.com/alist-org/alist/v3/server/middlewares"
 | |
| 	"net/http"
 | |
| 	"path"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/alist-org/alist/v3/internal/conf"
 | |
| 	"github.com/alist-org/alist/v3/internal/model"
 | |
| 	"github.com/alist-org/alist/v3/internal/op"
 | |
| 	"github.com/alist-org/alist/v3/internal/setting"
 | |
| 	"github.com/alist-org/alist/v3/server/webdav"
 | |
| 	"github.com/gin-gonic/gin"
 | |
| 	log "github.com/sirupsen/logrus"
 | |
| )
 | |
| 
 | |
| var handler *webdav.Handler
 | |
| 
 | |
| func WebDav(dav *gin.RouterGroup) {
 | |
| 	handler = &webdav.Handler{
 | |
| 		Prefix:     path.Join(conf.URL.Path, "/dav"),
 | |
| 		LockSystem: webdav.NewMemLS(),
 | |
| 		Logger: func(request *http.Request, err error) {
 | |
| 			log.Errorf("%s %s %+v", request.Method, request.URL.Path, err)
 | |
| 		},
 | |
| 	}
 | |
| 	dav.Use(WebDAVAuth)
 | |
| 	uploadLimiter := middlewares.UploadRateLimiter(stream.ClientUploadLimit)
 | |
| 	downloadLimiter := middlewares.DownloadRateLimiter(stream.ClientDownloadLimit)
 | |
| 	dav.Any("/*path", uploadLimiter, downloadLimiter, ServeWebDAV)
 | |
| 	dav.Any("", uploadLimiter, downloadLimiter, ServeWebDAV)
 | |
| 	dav.Handle("PROPFIND", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("PROPFIND", "", ServeWebDAV)
 | |
| 	dav.Handle("MKCOL", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("LOCK", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("UNLOCK", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("PROPPATCH", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("COPY", "/*path", ServeWebDAV)
 | |
| 	dav.Handle("MOVE", "/*path", ServeWebDAV)
 | |
| }
 | |
| 
 | |
| func ServeWebDAV(c *gin.Context) {
 | |
| 	user := c.MustGet("user").(*model.User)
 | |
| 	ctx := context.WithValue(c.Request.Context(), "user", user)
 | |
| 	handler.ServeHTTP(c.Writer, c.Request.WithContext(ctx))
 | |
| }
 | |
| 
 | |
| func WebDAVAuth(c *gin.Context) {
 | |
| 	guest, _ := op.GetGuest()
 | |
| 	username, password, ok := c.Request.BasicAuth()
 | |
| 	if !ok {
 | |
| 		bt := c.GetHeader("Authorization")
 | |
| 		log.Debugf("[webdav auth] token: %s", bt)
 | |
| 		if strings.HasPrefix(bt, "Bearer") {
 | |
| 			bt = strings.TrimPrefix(bt, "Bearer ")
 | |
| 			token := setting.GetStr(conf.Token)
 | |
| 			if token != "" && subtle.ConstantTimeCompare([]byte(bt), []byte(token)) == 1 {
 | |
| 				admin, err := op.GetAdmin()
 | |
| 				if err != nil {
 | |
| 					log.Errorf("[webdav auth] failed get admin user: %+v", err)
 | |
| 					c.Status(http.StatusInternalServerError)
 | |
| 					c.Abort()
 | |
| 					return
 | |
| 				}
 | |
| 				c.Set("user", admin)
 | |
| 				c.Next()
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 		if c.Request.Method == "OPTIONS" {
 | |
| 			c.Set("user", guest)
 | |
| 			c.Next()
 | |
| 			return
 | |
| 		}
 | |
| 		c.Writer.Header()["WWW-Authenticate"] = []string{`Basic realm="alist"`}
 | |
| 		c.Status(http.StatusUnauthorized)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	user, err := op.GetUserByName(username)
 | |
| 	if err != nil || user.ValidateRawPassword(password) != nil {
 | |
| 		if c.Request.Method == "OPTIONS" {
 | |
| 			c.Set("user", guest)
 | |
| 			c.Next()
 | |
| 			return
 | |
| 		}
 | |
| 		c.Status(http.StatusUnauthorized)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if user.Disabled || !user.CanWebdavRead() {
 | |
| 		if c.Request.Method == "OPTIONS" {
 | |
| 			c.Set("user", guest)
 | |
| 			c.Next()
 | |
| 			return
 | |
| 		}
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if (c.Request.Method == "PUT" || c.Request.Method == "MKCOL") && (!user.CanWebdavManage() || !user.CanWrite()) {
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if c.Request.Method == "MOVE" && (!user.CanWebdavManage() || (!user.CanMove() && !user.CanRename())) {
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if c.Request.Method == "COPY" && (!user.CanWebdavManage() || !user.CanCopy()) {
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if c.Request.Method == "DELETE" && (!user.CanWebdavManage() || !user.CanRemove()) {
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	if c.Request.Method == "PROPPATCH" && !user.CanWebdavManage() {
 | |
| 		c.Status(http.StatusForbidden)
 | |
| 		c.Abort()
 | |
| 		return
 | |
| 	}
 | |
| 	c.Set("user", user)
 | |
| 	c.Next()
 | |
| }
 |