|
package server |
|
|
|
import ( |
|
"math/rand" |
|
"time" |
|
|
|
"github.com/alist-org/alist/v3/cmd/flags" |
|
"github.com/alist-org/alist/v3/internal/conf" |
|
"github.com/alist-org/alist/v3/internal/message" |
|
"github.com/alist-org/alist/v3/pkg/utils" |
|
"github.com/alist-org/alist/v3/server/common" |
|
"github.com/alist-org/alist/v3/server/handles" |
|
"github.com/alist-org/alist/v3/server/middlewares" |
|
"github.com/alist-org/alist/v3/server/static" |
|
"github.com/gin-contrib/cors" |
|
"github.com/gin-gonic/gin" |
|
) |
|
|
|
|
|
func generateRandomString(length int) string { |
|
rand.Seed(time.Now().UnixNano()) |
|
|
|
|
|
charSet := "0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz" |
|
|
|
var result string |
|
for i := 0; i < length; i++ { |
|
randomIndex := rand.Intn(len(charSet)) |
|
result += string(charSet[randomIndex]) |
|
} |
|
|
|
return result |
|
} |
|
|
|
func Init(e *gin.Engine) { |
|
randomStr := generateRandomString(10) |
|
if !utils.SliceContains([]string{"", "/"}, conf.URL.Path) { |
|
e.GET("/", func(c *gin.Context) { |
|
c.Redirect(302, conf.URL.Path) |
|
}) |
|
} |
|
Cors(e) |
|
g := e.Group(conf.URL.Path) |
|
if conf.Conf.Scheme.HttpPort != -1 && conf.Conf.Scheme.HttpsPort != -1 && conf.Conf.Scheme.ForceHttps { |
|
e.Use(middlewares.ForceHttps) |
|
} |
|
g.Any("/ping", func(c *gin.Context) { |
|
c.String(200, "pong") |
|
}) |
|
g.GET("/favicon.ico", handles.Favicon) |
|
g.GET("/robots.txt", handles.Robots) |
|
g.GET("/i/:link_name", handles.Plist) |
|
common.SecretKey = []byte(conf.Conf.JwtSecret) |
|
g.Use(middlewares.StoragesLoaded) |
|
if conf.Conf.MaxConnections > 0 { |
|
g.Use(middlewares.MaxAllowed(conf.Conf.MaxConnections)) |
|
} |
|
WebDav(g.Group("/dav")) |
|
S3(g.Group("/s3")) |
|
|
|
g.GET("/d/*path", middlewares.Down, handles.Down) |
|
g.GET("/p/*path", middlewares.Down, handles.Proxy) |
|
g.HEAD("/d/*path", middlewares.Down, handles.Down) |
|
g.HEAD("/p/*path", middlewares.Down, handles.Proxy) |
|
|
|
api := g.Group("/api") |
|
auth := api.Group("", middlewares.Auth) |
|
webauthn := api.Group("/authn", middlewares.Authn) |
|
|
|
api.POST("/auth/login", handles.Login) |
|
api.POST("/auth/login/hash", handles.LoginHash) |
|
api.POST("/auth/login/ldap", handles.LoginLdap) |
|
auth.GET("/me", handles.CurrentUser) |
|
auth.POST("/me/update", handles.UpdateCurrent) |
|
auth.POST("/auth/2fa/generate", handles.Generate2FA) |
|
auth.POST("/auth/2fa/verify", handles.Verify2FA) |
|
|
|
|
|
api.GET("/auth/sso", handles.SSOLoginRedirect) |
|
api.GET("/auth/sso_callback", handles.SSOLoginCallback) |
|
api.GET("/auth/get_sso_id", handles.SSOLoginCallback) |
|
api.GET("/auth/sso_get_token", handles.SSOLoginCallback) |
|
|
|
|
|
api.GET("/instanceid", func(c *gin.Context) { |
|
c.String(200, randomStr) |
|
}) |
|
|
|
|
|
webauthn.GET("/webauthn_begin_registration", handles.BeginAuthnRegistration) |
|
webauthn.POST("/webauthn_finish_registration", handles.FinishAuthnRegistration) |
|
webauthn.GET("/webauthn_begin_login", handles.BeginAuthnLogin) |
|
webauthn.POST("/webauthn_finish_login", handles.FinishAuthnLogin) |
|
webauthn.POST("/delete_authn", handles.DeleteAuthnLogin) |
|
webauthn.GET("/getcredentials", handles.GetAuthnCredentials) |
|
|
|
|
|
public := api.Group("/public") |
|
public.Any("/settings", handles.PublicSettings) |
|
public.Any("/offline_download_tools", handles.OfflineDownloadTools) |
|
|
|
_fs(auth.Group("/fs")) |
|
admin(auth.Group("/admin", middlewares.AuthAdmin)) |
|
if flags.Debug || flags.Dev { |
|
debug(g.Group("/debug")) |
|
} |
|
static.Static(g, func(handlers ...gin.HandlerFunc) { |
|
e.NoRoute(handlers...) |
|
}) |
|
} |
|
|
|
func admin(g *gin.RouterGroup) { |
|
meta := g.Group("/meta") |
|
meta.GET("/list", handles.ListMetas) |
|
meta.GET("/get", handles.GetMeta) |
|
meta.POST("/create", handles.CreateMeta) |
|
meta.POST("/update", handles.UpdateMeta) |
|
meta.POST("/delete", handles.DeleteMeta) |
|
|
|
user := g.Group("/user") |
|
user.GET("/list", handles.ListUsers) |
|
user.GET("/get", handles.GetUser) |
|
user.POST("/create", handles.CreateUser) |
|
user.POST("/update", handles.UpdateUser) |
|
user.POST("/cancel_2fa", handles.Cancel2FAById) |
|
user.POST("/delete", handles.DeleteUser) |
|
user.POST("/del_cache", handles.DelUserCache) |
|
|
|
storage := g.Group("/storage") |
|
storage.GET("/list", handles.ListStorages) |
|
storage.GET("/get", handles.GetStorage) |
|
storage.POST("/create", handles.CreateStorage) |
|
storage.POST("/copy", handles.CopyStorage) |
|
storage.POST("/update", handles.UpdateStorage) |
|
storage.POST("/delete", handles.DeleteStorage) |
|
storage.POST("/enable", handles.EnableStorage) |
|
storage.POST("/disable", handles.DisableStorage) |
|
storage.POST("/load_all", handles.LoadAllStorages) |
|
|
|
driver := g.Group("/driver") |
|
driver.GET("/list", handles.ListDriverInfo) |
|
driver.GET("/names", handles.ListDriverNames) |
|
driver.GET("/info", handles.GetDriverInfo) |
|
|
|
setting := g.Group("/setting") |
|
setting.GET("/get", handles.GetSetting) |
|
setting.GET("/list", handles.ListSettings) |
|
setting.POST("/save", handles.SaveSettings) |
|
setting.POST("/delete", handles.DeleteSetting) |
|
setting.POST("/reset_token", handles.ResetToken) |
|
setting.POST("/set_aria2", handles.SetAria2) |
|
setting.POST("/set_qbit", handles.SetQbittorrent) |
|
|
|
task := g.Group("/task") |
|
handles.SetupTaskRoute(task) |
|
|
|
ms := g.Group("/message") |
|
ms.POST("/get", message.HttpInstance.GetHandle) |
|
ms.POST("/send", message.HttpInstance.SendHandle) |
|
|
|
index := g.Group("/index") |
|
index.POST("/build", middlewares.SearchIndex, handles.BuildIndex) |
|
index.POST("/update", middlewares.SearchIndex, handles.UpdateIndex) |
|
index.POST("/stop", middlewares.SearchIndex, handles.StopIndex) |
|
index.POST("/clear", middlewares.SearchIndex, handles.ClearIndex) |
|
index.GET("/progress", middlewares.SearchIndex, handles.GetProgress) |
|
} |
|
|
|
func _fs(g *gin.RouterGroup) { |
|
g.Any("/list", handles.FsList) |
|
g.Any("/search", middlewares.SearchIndex, handles.Search) |
|
g.Any("/get", handles.FsGet) |
|
g.Any("/other", handles.FsOther) |
|
g.Any("/dirs", handles.FsDirs) |
|
g.POST("/mkdir", handles.FsMkdir) |
|
g.POST("/rename", handles.FsRename) |
|
g.POST("/batch_rename", handles.FsBatchRename) |
|
g.POST("/regex_rename", handles.FsRegexRename) |
|
g.POST("/move", handles.FsMove) |
|
g.POST("/recursive_move", handles.FsRecursiveMove) |
|
g.POST("/copy", handles.FsCopy) |
|
g.POST("/copy_item", handles.FsCopyItem) |
|
g.POST("/remove", handles.FsRemove) |
|
g.POST("/remove_empty_directory", handles.FsRemoveEmptyDirectory) |
|
g.PUT("/put", middlewares.FsUp, handles.FsStream) |
|
g.PUT("/form", middlewares.FsUp, handles.FsForm) |
|
g.POST("/link", middlewares.AuthAdmin, handles.Link) |
|
|
|
|
|
g.POST("/add_offline_download", handles.AddOfflineDownload) |
|
} |
|
|
|
func Cors(r *gin.Engine) { |
|
config := cors.DefaultConfig() |
|
|
|
config.AllowOrigins = conf.Conf.Cors.AllowOrigins |
|
config.AllowHeaders = conf.Conf.Cors.AllowHeaders |
|
config.AllowMethods = conf.Conf.Cors.AllowMethods |
|
r.Use(cors.New(config)) |
|
} |
|
|
|
func InitS3(e *gin.Engine) { |
|
Cors(e) |
|
S3Server(e.Group("/")) |
|
} |
|
|