Cloudreve/pkg/filesystem/hooks.go

138 lines
3.4 KiB
Go
Raw Normal View History

2019-11-16 16:05:10 +08:00
package filesystem
2019-11-16 20:31:34 +08:00
import (
"context"
"errors"
"github.com/HFO4/cloudreve/pkg/util"
2019-11-16 20:31:34 +08:00
)
2019-11-16 16:05:10 +08:00
2019-11-26 11:42:26 +08:00
// Hook 钩子函数
type Hook func(ctx context.Context, fs *FileSystem) error
// Use 注入钩子
func (fs *FileSystem) Use(name string, hook Hook) {
switch name {
case "BeforeUpload":
fs.BeforeUpload = append(fs.BeforeUpload, hook)
case "AfterUpload":
fs.AfterUpload = append(fs.AfterUpload, hook)
case "AfterValidateFailed":
fs.AfterValidateFailed = append(fs.AfterValidateFailed, hook)
case "AfterUploadCanceled":
fs.AfterUploadCanceled = append(fs.AfterUploadCanceled, hook)
2019-11-26 20:59:57 +08:00
case "BeforeFileDownload":
fs.BeforeFileDownload = append(fs.BeforeFileDownload, hook)
2019-11-26 11:42:26 +08:00
}
}
// Trigger 触发钩子,遇到第一个错误时
// 返回错误,后续钩子不会继续执行
func (fs *FileSystem) Trigger(ctx context.Context, hooks []Hook) error {
for _, hook := range hooks {
err := hook(ctx, fs)
if err != nil {
util.Log().Warning("钩子执行失败:%s", err)
return err
}
}
return nil
}
2019-11-26 20:59:57 +08:00
// HookIsFileExist 检查虚拟路径文件是否存在
func HookIsFileExist(ctx context.Context, fs *FileSystem) error {
filePath := ctx.Value(PathCtx).(string)
if ok, _ := fs.IsFileExist(filePath); ok {
return nil
}
return ErrObjectNotExist
}
2019-11-26 11:42:26 +08:00
// HookValidateFile 一系列对文件检验的集合
func HookValidateFile(ctx context.Context, fs *FileSystem) error {
file := ctx.Value(FileHeaderCtx).(FileHeader)
2019-11-18 19:09:56 +08:00
2019-11-16 16:05:10 +08:00
// 验证单文件尺寸
2019-11-16 20:31:34 +08:00
if !fs.ValidateFileSize(ctx, file.GetSize()) {
return ErrFileSizeTooBig
2019-11-16 16:05:10 +08:00
}
// 验证文件名
if !fs.ValidateLegalName(ctx, file.GetFileName()) {
return ErrIllegalObjectName
}
2019-11-16 16:05:10 +08:00
// 验证扩展名
2019-11-16 20:31:34 +08:00
if !fs.ValidateExtension(ctx, file.GetFileName()) {
return ErrFileExtensionNotAllowed
2019-11-17 13:50:14 +08:00
}
2019-11-26 11:42:26 +08:00
return nil
}
// HookValidateCapacity 验证并扣除用户容量,包含数据库操作
func HookValidateCapacity(ctx context.Context, fs *FileSystem) error {
file := ctx.Value(FileHeaderCtx).(FileHeader)
2019-11-17 13:50:14 +08:00
// 验证并扣除容量
if !fs.ValidateCapacity(ctx, file.GetSize()) {
return ErrInsufficientCapacity
2019-11-16 16:05:10 +08:00
}
return nil
}
2019-11-26 11:42:26 +08:00
// HookDeleteTempFile 删除已保存的临时文件
func HookDeleteTempFile(ctx context.Context, fs *FileSystem) error {
2019-11-18 19:32:06 +08:00
filePath := ctx.Value(SavePathCtx).(string)
// 删除临时文件
if util.Exists(filePath) {
_, err := fs.Handler.Delete(ctx, []string{filePath})
if err != nil {
return err
}
}
2019-11-26 11:42:26 +08:00
return nil
}
// HookGiveBackCapacity 归还用户容量
func HookGiveBackCapacity(ctx context.Context, fs *FileSystem) error {
file := ctx.Value(FileHeaderCtx).(FileHeader)
// 归还用户容量
if !fs.User.DeductionStorage(file.GetSize()) {
return errors.New("无法继续降低用户已用存储")
}
return nil
}
2019-11-18 19:09:56 +08:00
// GenericAfterUpload 文件上传完成后,包含数据库操作
func GenericAfterUpload(ctx context.Context, fs *FileSystem) error {
// 文件存放的虚拟路径
virtualPath := ctx.Value(FileHeaderCtx).(FileHeader).GetVirtualPath()
2019-11-18 19:09:56 +08:00
// 检查路径是否存在
isExist, folder := fs.IsPathExist(virtualPath)
if !isExist {
return ErrPathNotExist
2019-11-18 19:09:56 +08:00
}
// 检查文件是否存在
if ok, _ := fs.IsChildFileExist(
folder,
ctx.Value(FileHeaderCtx).(FileHeader).GetFileName(),
); ok {
return ErrFileExisted
}
// 向数据库中插入记录
file, err := fs.AddFile(ctx, folder)
if err != nil {
return ErrInsertFileRecord
}
2019-11-20 15:24:26 +08:00
// 异步尝试生成缩略图
go fs.GenerateThumbnail(ctx, file)
2019-11-20 15:24:26 +08:00
2019-11-18 19:09:56 +08:00
return nil
}