mirror of
https://github.com/keven1024/015.git
synced 2026-05-26 07:08:02 +00:00
159 lines
4.6 KiB
Go
159 lines
4.6 KiB
Go
package controllers
|
||
|
||
import (
|
||
"backend/internal/utils"
|
||
"context"
|
||
"encoding/json"
|
||
"fmt"
|
||
"pkg/models"
|
||
u "pkg/utils"
|
||
"time"
|
||
|
||
"github.com/golang-jwt/jwt/v5"
|
||
"github.com/hibiken/asynq"
|
||
"github.com/labstack/echo/v5"
|
||
"github.com/samber/lo"
|
||
"github.com/spf13/cast"
|
||
)
|
||
|
||
type DownloadShareClaims struct {
|
||
ShareId string `json:"share_id"`
|
||
jwt.RegisteredClaims
|
||
}
|
||
|
||
func DownloadShare(c *echo.Context) error {
|
||
token := c.FormValue("token")
|
||
if token == "" {
|
||
return utils.HTTPErrorHandler(c, ErrInvalidRequest)
|
||
}
|
||
claims := DownloadShareClaims{}
|
||
t, err := jwt.ParseWithClaims(token, &claims, func(token *jwt.Token) (interface{}, error) {
|
||
return []byte(u.GetEnv("share.download_secret")), nil
|
||
})
|
||
if err != nil || !t.Valid {
|
||
return utils.HTTPErrorHandler(c, lo.Ternary(err != nil, err, ErrInvalidRequest))
|
||
}
|
||
shareInfo, err := models.GetRedisShareInfo(claims.ShareId)
|
||
if err != nil || shareInfo == nil {
|
||
return utils.HTTPErrorHandler(c, lo.Ternary(err != nil, err, ErrShareNotFound))
|
||
}
|
||
if shareInfo.Type == models.ShareTypeFile {
|
||
fileInfo, _ := models.GetRedisFileInfo(shareInfo.Data)
|
||
uploadPath, err := u.GetUploadDirPath()
|
||
if err != nil {
|
||
return err
|
||
}
|
||
return c.Attachment(fmt.Sprintf("%s/%s", uploadPath, u.GetFileId(fileInfo.FileHash, fileInfo.FileSize)), shareInfo.FileName)
|
||
}
|
||
return utils.HTTPSuccessHandler(c, map[string]any{
|
||
"data": shareInfo.Data,
|
||
})
|
||
}
|
||
|
||
type VaildateShareProps struct {
|
||
ShareId string `json:"share_id"`
|
||
Password string `json:"password"`
|
||
}
|
||
|
||
func VaildateShare(c *echo.Context) error {
|
||
r := new(VaildateShareProps)
|
||
if err := c.Bind(r); err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
|
||
if r.ShareId == "" {
|
||
return utils.HTTPErrorHandler(c, ErrInvalidRequest)
|
||
}
|
||
|
||
shareInfo, err := models.GetRedisShareInfo(r.ShareId)
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
if shareInfo == nil {
|
||
return utils.HTTPErrorHandler(c, ErrShareNotFound)
|
||
}
|
||
if shareInfo.Password != "" {
|
||
if r.Password == "" {
|
||
return utils.HTTPErrorHandler(c, ErrInvalidRequest)
|
||
}
|
||
hash, err := utils.GeneratePasswordHash(r.Password)
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
if hash != shareInfo.Password {
|
||
return utils.HTTPErrorHandler(c, ErrInvalidSharePassword)
|
||
}
|
||
}
|
||
return u.WithLocker(context.Background(), "015:shareInfoMap:"+r.ShareId, 0, func(ctx context.Context) error {
|
||
shareInfo, err := models.GetRedisShareInfo(r.ShareId)
|
||
if err != nil || shareInfo == nil {
|
||
return utils.HTTPErrorHandler(c, lo.Ternary(err != nil, err, ErrShareNotFound))
|
||
}
|
||
if shareInfo.ViewNum < 1 {
|
||
return utils.HTTPErrorHandler(c, ErrInsufficientDownloadQuota)
|
||
}
|
||
downloadWindow := u.GetEnvWithDefault("share.download_window", "12")
|
||
token := jwt.NewWithClaims(jwt.SigningMethodHS256, DownloadShareClaims{
|
||
ShareId: r.ShareId,
|
||
RegisteredClaims: jwt.RegisteredClaims{
|
||
ExpiresAt: jwt.NewNumericDate(time.Now().Add(cast.ToDuration(downloadWindow + "h"))),
|
||
},
|
||
})
|
||
|
||
// Sign and get the complete encoded token as a string using the secret
|
||
downloadToken, err := token.SignedString([]byte(u.GetEnv("share.download_secret")))
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
if shareInfo.Type == models.ShareTypeFile {
|
||
fileInfo, err := models.GetRedisFileInfo(shareInfo.Data)
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
if fileInfo == nil {
|
||
return utils.HTTPErrorHandler(c, ErrShareFileNotFound)
|
||
}
|
||
if fileInfo.FileType != models.FileTypeUpload {
|
||
return utils.HTTPErrorHandler(c, ErrInvalidShareFileState)
|
||
}
|
||
}
|
||
// download_nums 必须放在创建token的时候减掉,不然多线程下载会导致多次减掉
|
||
_, err = models.SetRedisShareInfo(r.ShareId, func(shareInfo *models.RedisShareInfo) *models.RedisShareInfo {
|
||
shareInfo.ViewNum -= 1
|
||
return shareInfo
|
||
})
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
|
||
// 统计分享数
|
||
currentDate := time.Now().Format("2006-01-02")
|
||
_, err = models.SetRedisStat(currentDate, func(stat *models.StatData) *models.StatData {
|
||
stat.DownloadNum += 1
|
||
return stat
|
||
})
|
||
if err != nil {
|
||
return utils.HTTPErrorHandler(c, err)
|
||
}
|
||
|
||
if len(shareInfo.NotifyEmails) > 0 || len(shareInfo.NotifyWebhooks) > 0 {
|
||
payload, err := json.Marshal(map[string]string{
|
||
"share_id": r.ShareId,
|
||
"ip": c.RealIP(),
|
||
})
|
||
if err == nil {
|
||
_, _ = u.GetQueueClient().Enqueue(asynq.NewTask("share:notify", payload))
|
||
}
|
||
}
|
||
|
||
if shareInfo.Type == models.ShareTypeFile {
|
||
return utils.HTTPSuccessHandler(c, map[string]any{
|
||
"token": downloadToken,
|
||
})
|
||
}
|
||
return utils.HTTPSuccessHandler(c, map[string]any{
|
||
"token": downloadToken,
|
||
})
|
||
})
|
||
}
|