2022-06-25 21:34:44 +08:00
|
|
|
package middlewares
|
|
|
|
|
|
|
|
import (
|
2023-03-06 23:41:06 +08:00
|
|
|
"crypto/subtle"
|
|
|
|
|
2022-06-29 16:08:55 +08:00
|
|
|
"github.com/alist-org/alist/v3/internal/conf"
|
2022-06-26 19:09:28 +08:00
|
|
|
"github.com/alist-org/alist/v3/internal/model"
|
2022-12-18 19:51:20 +08:00
|
|
|
"github.com/alist-org/alist/v3/internal/op"
|
2022-06-28 14:18:10 +08:00
|
|
|
"github.com/alist-org/alist/v3/internal/setting"
|
2022-06-28 18:12:53 +08:00
|
|
|
"github.com/alist-org/alist/v3/server/common"
|
2022-06-25 21:34:44 +08:00
|
|
|
"github.com/gin-gonic/gin"
|
2022-07-23 21:33:53 +08:00
|
|
|
log "github.com/sirupsen/logrus"
|
2022-06-25 21:34:44 +08:00
|
|
|
)
|
|
|
|
|
2022-06-26 16:39:02 +08:00
|
|
|
// Auth is a middleware that checks if the user is logged in.
|
|
|
|
// if token is empty, set user to guest
|
|
|
|
func Auth(c *gin.Context) {
|
2022-06-25 21:34:44 +08:00
|
|
|
token := c.GetHeader("Authorization")
|
2023-03-06 23:41:06 +08:00
|
|
|
if subtle.ConstantTimeCompare([]byte(token), []byte(setting.GetStr(conf.Token))) == 1 {
|
2022-12-18 19:51:20 +08:00
|
|
|
admin, err := op.GetAdmin()
|
2022-06-28 14:18:10 +08:00
|
|
|
if err != nil {
|
2022-06-28 18:12:53 +08:00
|
|
|
common.ErrorResp(c, err, 500)
|
2022-06-28 14:18:10 +08:00
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Set("user", admin)
|
2022-07-23 21:33:53 +08:00
|
|
|
log.Debugf("use admin token: %+v", admin)
|
2022-06-28 14:18:10 +08:00
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
2022-06-26 16:39:02 +08:00
|
|
|
if token == "" {
|
2022-12-18 19:51:20 +08:00
|
|
|
guest, err := op.GetGuest()
|
2022-06-26 16:55:37 +08:00
|
|
|
if err != nil {
|
2022-06-28 18:12:53 +08:00
|
|
|
common.ErrorResp(c, err, 500)
|
2022-06-26 16:55:37 +08:00
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2023-02-04 11:44:17 +08:00
|
|
|
if guest.Disabled {
|
|
|
|
common.ErrorStrResp(c, "Guest user is disabled, login please", 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2022-06-26 16:39:02 +08:00
|
|
|
c.Set("user", guest)
|
2022-07-23 21:33:53 +08:00
|
|
|
log.Debugf("use empty token: %+v", guest)
|
2022-06-26 16:39:02 +08:00
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
2022-06-28 18:12:53 +08:00
|
|
|
userClaims, err := common.ParseToken(token)
|
2022-06-25 21:34:44 +08:00
|
|
|
if err != nil {
|
2022-06-28 18:12:53 +08:00
|
|
|
common.ErrorResp(c, err, 401)
|
2022-06-25 21:34:44 +08:00
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2022-12-18 19:51:20 +08:00
|
|
|
user, err := op.GetUserByName(userClaims.Username)
|
2022-06-25 21:34:44 +08:00
|
|
|
if err != nil {
|
2022-06-28 18:12:53 +08:00
|
|
|
common.ErrorResp(c, err, 401)
|
2022-06-25 21:34:44 +08:00
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2023-11-13 15:22:42 +08:00
|
|
|
// validate password timestamp
|
|
|
|
if userClaims.PwdTS != user.PwdTS {
|
|
|
|
common.ErrorStrResp(c, "Password has been changed, login please", 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2023-02-04 11:44:17 +08:00
|
|
|
if user.Disabled {
|
|
|
|
common.ErrorStrResp(c, "Current user is disabled, replace please", 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2022-06-25 21:34:44 +08:00
|
|
|
c.Set("user", user)
|
2022-07-23 21:33:53 +08:00
|
|
|
log.Debugf("use login token: %+v", user)
|
2022-06-25 21:34:44 +08:00
|
|
|
c.Next()
|
|
|
|
}
|
2022-06-26 19:09:28 +08:00
|
|
|
|
2023-08-14 22:54:38 +08:00
|
|
|
func Authn(c *gin.Context) {
|
|
|
|
token := c.GetHeader("Authorization")
|
|
|
|
if subtle.ConstantTimeCompare([]byte(token), []byte(setting.GetStr(conf.Token))) == 1 {
|
|
|
|
admin, err := op.GetAdmin()
|
|
|
|
if err != nil {
|
|
|
|
common.ErrorResp(c, err, 500)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Set("user", admin)
|
|
|
|
log.Debugf("use admin token: %+v", admin)
|
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if token == "" {
|
|
|
|
guest, err := op.GetGuest()
|
|
|
|
if err != nil {
|
|
|
|
common.ErrorResp(c, err, 500)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Set("user", guest)
|
|
|
|
log.Debugf("use empty token: %+v", guest)
|
|
|
|
c.Next()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
userClaims, err := common.ParseToken(token)
|
|
|
|
if err != nil {
|
|
|
|
common.ErrorResp(c, err, 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
user, err := op.GetUserByName(userClaims.Username)
|
|
|
|
if err != nil {
|
|
|
|
common.ErrorResp(c, err, 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2023-11-13 15:22:42 +08:00
|
|
|
// validate password timestamp
|
|
|
|
if userClaims.PwdTS != user.PwdTS {
|
|
|
|
common.ErrorStrResp(c, "Password has been changed, login please", 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
2023-08-14 22:54:38 +08:00
|
|
|
if user.Disabled {
|
|
|
|
common.ErrorStrResp(c, "Current user is disabled, replace please", 401)
|
|
|
|
c.Abort()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
c.Set("user", user)
|
|
|
|
log.Debugf("use login token: %+v", user)
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
|
2024-11-01 23:32:26 +08:00
|
|
|
func AuthNotGuest(c *gin.Context) {
|
|
|
|
user := c.MustGet("user").(*model.User)
|
|
|
|
if user.IsGuest() {
|
|
|
|
common.ErrorStrResp(c, "You are a guest", 403)
|
|
|
|
c.Abort()
|
|
|
|
} else {
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-26 19:09:28 +08:00
|
|
|
func AuthAdmin(c *gin.Context) {
|
|
|
|
user := c.MustGet("user").(*model.User)
|
|
|
|
if !user.IsAdmin() {
|
2022-06-28 18:12:53 +08:00
|
|
|
common.ErrorStrResp(c, "You are not an admin", 403)
|
2022-06-26 19:09:28 +08:00
|
|
|
c.Abort()
|
|
|
|
} else {
|
|
|
|
c.Next()
|
|
|
|
}
|
|
|
|
}
|