Cloudreve/models/share.go

205 lines
5.1 KiB
Go
Raw Normal View History

2020-01-26 13:07:05 +08:00
package model
import (
"errors"
"fmt"
"github.com/HFO4/cloudreve/pkg/cache"
2020-01-26 14:57:07 +08:00
"github.com/HFO4/cloudreve/pkg/hashid"
2020-01-26 13:07:05 +08:00
"github.com/HFO4/cloudreve/pkg/util"
"github.com/gin-gonic/gin"
2020-01-26 13:07:05 +08:00
"github.com/jinzhu/gorm"
"math"
2020-01-26 13:07:05 +08:00
"time"
)
// Share 分享模型
type Share struct {
gorm.Model
Password string // 分享密码,空值为非加密分享
IsDir bool // 原始资源是否为目录
UserID uint // 创建用户ID
SourceID uint // 原始资源ID
Views int // 浏览数
Downloads int // 下载数
RemainDownloads int // 剩余下载配额,负值标识无限制
Expires *time.Time // 过期时间,空值表示无过期时间
Score int // 每人次下载扣除积分
PreviewEnabled bool // 是否允许直接预览
2020-01-27 13:34:39 +08:00
// 数据库忽略字段
User User `gorm:"PRELOAD:false,association_autoupdate:false"`
File File `gorm:"PRELOAD:false,association_autoupdate:false"`
Folder Folder `gorm:"PRELOAD:false,association_autoupdate:false"`
2020-01-26 13:07:05 +08:00
}
// Create 创建分享
func (share *Share) Create() (uint, error) {
if err := DB.Create(share).Error; err != nil {
util.Log().Warning("无法插入数据库记录, %s", err)
return 0, err
}
return share.ID, nil
}
2020-01-26 14:57:07 +08:00
// GetShareByHashID 根据HashID查找分享
func GetShareByHashID(hashID string) *Share {
id, err := hashid.DecodeHashID(hashID, hashid.ShareID)
if err != nil {
return nil
}
var share Share
result := DB.First(&share, id)
if result.Error != nil {
return nil
}
return &share
}
2020-01-27 13:34:39 +08:00
// IsAvailable 返回此分享是否可用(是否过期)
func (share *Share) IsAvailable() bool {
if share.RemainDownloads == 0 {
return false
}
if share.Expires != nil && time.Now().After(*share.Expires) {
return false
}
// 检查源对象是否存在
var sourceID uint
if share.IsDir {
2020-02-01 13:14:50 +08:00
folder := share.SourceFolder()
2020-01-27 13:34:39 +08:00
sourceID = folder.ID
} else {
2020-02-01 13:14:50 +08:00
file := share.SourceFile()
2020-01-27 13:34:39 +08:00
sourceID = file.ID
}
if sourceID == 0 {
2020-02-01 13:14:50 +08:00
// TODO 是否要在这里删除这个无效分享?
2020-01-27 13:34:39 +08:00
return false
}
return true
}
2020-02-01 13:14:50 +08:00
// Creator 获取分享的创建者
func (share *Share) Creator() *User {
2020-01-27 13:34:39 +08:00
if share.User.ID == 0 {
share.User, _ = GetUserByID(share.UserID)
}
return &share.User
}
2020-02-01 13:14:50 +08:00
// Source 返回源对象
func (share *Share) Source() interface{} {
if share.IsDir {
2020-02-01 13:14:50 +08:00
return share.SourceFolder()
}
2020-02-01 13:14:50 +08:00
return share.SourceFile()
}
2020-02-01 13:14:50 +08:00
// SourceFolder 获取源目录
func (share *Share) SourceFolder() *Folder {
2020-01-27 13:34:39 +08:00
if share.Folder.ID == 0 {
folders, _ := GetFoldersByIDs([]uint{share.SourceID}, share.UserID)
if len(folders) > 0 {
share.Folder = folders[0]
}
}
return &share.Folder
}
2020-02-01 13:14:50 +08:00
// SourceFile 获取源文件
func (share *Share) SourceFile() *File {
2020-01-27 13:34:39 +08:00
if share.File.ID == 0 {
files, _ := GetFilesByIDs([]uint{share.SourceID}, share.UserID)
if len(files) > 0 {
share.File = files[0]
}
}
return &share.File
}
// CanBeDownloadBy 返回此分享是否可以被给定用户下载
func (share *Share) CanBeDownloadBy(user *User) error {
// 用户组权限
2020-02-02 14:40:07 +08:00
if !user.Group.OptionsSerialized.ShareDownload {
if user.IsAnonymous() {
return errors.New("未登录用户无法下载")
}
return errors.New("您当前的用户组无权下载")
}
// 需要积分但未登录
if share.Score > 0 && user.IsAnonymous() {
return errors.New("未登录用户无法下载")
}
return nil
}
// WasDownloadedBy 返回分享是否已被用户下载过
func (share *Share) WasDownloadedBy(user *User, c *gin.Context) (exist bool) {
if user.IsAnonymous() {
exist = util.GetSession(c, fmt.Sprintf("share_%d_%d", share.ID, user.ID)) != nil
} else {
_, exist = cache.Get(fmt.Sprintf("share_%d_%d", share.ID, user.ID))
}
return exist
}
// DownloadBy 增加下载次数、检查积分等,匿名用户不会缓存
func (share *Share) DownloadBy(user *User, c *gin.Context) error {
if !share.WasDownloadedBy(user, c) {
if err := share.Purchase(user); err != nil {
return err
}
share.Downloaded()
if !user.IsAnonymous() {
cache.Set(fmt.Sprintf("share_%d_%d", share.ID, user.ID), true,
GetIntSetting("share_download_session_timeout", 2073600))
} else {
util.SetSession(c, map[string]interface{}{fmt.Sprintf("share_%d_%d", share.ID, user.ID): true})
}
}
return nil
}
// Purchase 使用积分购买分享
func (share *Share) Purchase(user *User) error {
// 不需要付积分
2020-02-02 14:40:07 +08:00
if share.Score == 0 || user.Group.OptionsSerialized.ShareFree || user.ID == share.UserID {
return nil
}
ok := user.PayScore(share.Score)
if !ok {
return errors.New("积分不足")
}
scoreRate := GetIntSetting("share_score_rate", 100)
gainedScore := int(math.Ceil(float64(share.Score*scoreRate) / 100))
2020-02-01 13:14:50 +08:00
share.Creator().AddScore(gainedScore)
return nil
}
// Viewed 增加访问次数
func (share *Share) Viewed() {
share.Views++
DB.Model(share).UpdateColumn("views", gorm.Expr("views + ?", 1))
}
// Downloaded 增加下载次数
func (share *Share) Downloaded() {
share.Downloads++
if share.RemainDownloads > 0 {
share.RemainDownloads--
}
DB.Model(share).Updates(map[string]interface{}{
"downloads": share.Downloads,
"remain_downloads": share.RemainDownloads,
})
}