|
|
package filesystem
|
|
|
|
|
|
import (
|
|
|
"context"
|
|
|
model "github.com/cloudreve/Cloudreve/v3/models"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/cache"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/cluster"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/conf"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/filesystem/driver/local"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/filesystem/fsctx"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/serializer"
|
|
|
"github.com/cloudreve/Cloudreve/v3/pkg/util"
|
|
|
"io/ioutil"
|
|
|
"strings"
|
|
|
)
|
|
|
|
|
|
// Hook 钩子函数
|
|
|
type Hook func(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error
|
|
|
|
|
|
// Use 注入钩子
|
|
|
func (fs *FileSystem) Use(name string, hook Hook) {
|
|
|
if fs.Hooks == nil {
|
|
|
fs.Hooks = make(map[string][]Hook)
|
|
|
}
|
|
|
if _, ok := fs.Hooks[name]; ok {
|
|
|
fs.Hooks[name] = append(fs.Hooks[name], hook)
|
|
|
return
|
|
|
}
|
|
|
fs.Hooks[name] = []Hook{hook}
|
|
|
}
|
|
|
|
|
|
// CleanHooks 清空钩子,name为空表示全部清空
|
|
|
func (fs *FileSystem) CleanHooks(name string) {
|
|
|
if name == "" {
|
|
|
fs.Hooks = nil
|
|
|
} else {
|
|
|
delete(fs.Hooks, name)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Trigger 触发钩子,遇到第一个错误时
|
|
|
// 返回错误,后续钩子不会继续执行
|
|
|
func (fs *FileSystem) Trigger(ctx context.Context, name string, file fsctx.FileHeader) error {
|
|
|
if hooks, ok := fs.Hooks[name]; ok {
|
|
|
for _, hook := range hooks {
|
|
|
err := hook(ctx, fs, file)
|
|
|
if err != nil {
|
|
|
util.Log().Warning("钩子执行失败:%s", err)
|
|
|
return err
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookValidateFile 一系列对文件检验的集合
|
|
|
func HookValidateFile(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
fileInfo := file.Info()
|
|
|
|
|
|
// 验证单文件尺寸
|
|
|
if !fs.ValidateFileSize(ctx, fileInfo.Size) {
|
|
|
return ErrFileSizeTooBig
|
|
|
}
|
|
|
|
|
|
// 验证文件名
|
|
|
if !fs.ValidateLegalName(ctx, fileInfo.FileName) {
|
|
|
return ErrIllegalObjectName
|
|
|
}
|
|
|
|
|
|
// 验证扩展名
|
|
|
if !fs.ValidateExtension(ctx, fileInfo.FileName) {
|
|
|
return ErrFileExtensionNotAllowed
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
|
|
|
}
|
|
|
|
|
|
// HookResetPolicy 重设存储策略为上下文已有文件
|
|
|
func HookResetPolicy(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
originFile, ok := ctx.Value(fsctx.FileModelCtx).(model.File)
|
|
|
if !ok {
|
|
|
return ErrObjectNotExist
|
|
|
}
|
|
|
|
|
|
fs.Policy = originFile.GetPolicy()
|
|
|
return fs.DispatchHandler()
|
|
|
}
|
|
|
|
|
|
// HookValidateCapacity 验证用户容量
|
|
|
func HookValidateCapacity(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
// 验证并扣除容量
|
|
|
if fs.User.GetRemainingCapacity() < file.Info().Size {
|
|
|
return ErrInsufficientCapacity
|
|
|
}
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookValidateCapacityDiff 根据原有文件和新文件的大小验证用户容量
|
|
|
func HookValidateCapacityDiff(ctx context.Context, fs *FileSystem, newFile fsctx.FileHeader) error {
|
|
|
originFile := ctx.Value(fsctx.FileModelCtx).(model.File)
|
|
|
newFileSize := newFile.Info().Size
|
|
|
|
|
|
if newFileSize > originFile.Size {
|
|
|
return HookValidateCapacity(ctx, fs, newFile)
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookDeleteTempFile 删除已保存的临时文件
|
|
|
func HookDeleteTempFile(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
// 删除临时文件
|
|
|
_, err := fs.Handler.Delete(ctx, []string{file.Info().SavePath})
|
|
|
if err != nil {
|
|
|
util.Log().Warning("无法清理上传临时文件,%s", err)
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookCleanFileContent 清空文件内容
|
|
|
func HookCleanFileContent(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
// 清空内容
|
|
|
return fs.Handler.Put(ctx, &fsctx.FileStream{
|
|
|
File: ioutil.NopCloser(strings.NewReader("")),
|
|
|
SavePath: file.Info().SavePath,
|
|
|
Size: 0,
|
|
|
Mode: fsctx.Overwrite,
|
|
|
})
|
|
|
}
|
|
|
|
|
|
// HookClearFileSize 将原始文件的尺寸设为0
|
|
|
func HookClearFileSize(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
originFile, ok := ctx.Value(fsctx.FileModelCtx).(model.File)
|
|
|
if !ok {
|
|
|
return ErrObjectNotExist
|
|
|
}
|
|
|
return originFile.UpdateSize(0)
|
|
|
}
|
|
|
|
|
|
// HookCancelContext 取消上下文
|
|
|
func HookCancelContext(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
cancelFunc, ok := ctx.Value(fsctx.CancelFuncCtx).(context.CancelFunc)
|
|
|
if ok {
|
|
|
cancelFunc()
|
|
|
}
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookUpdateSourceName 更新文件SourceName
|
|
|
// TODO:测试
|
|
|
func HookUpdateSourceName(ctx context.Context, fs *FileSystem, file fsctx.FileHeader) error {
|
|
|
originFile, ok := ctx.Value(fsctx.FileModelCtx).(model.File)
|
|
|
if !ok {
|
|
|
return ErrObjectNotExist
|
|
|
}
|
|
|
return originFile.UpdateSourceName(originFile.SourceName)
|
|
|
}
|
|
|
|
|
|
// GenericAfterUpdate 文件内容更新后
|
|
|
func GenericAfterUpdate(ctx context.Context, fs *FileSystem, newFile fsctx.FileHeader) error {
|
|
|
// 更新文件尺寸
|
|
|
originFile, ok := ctx.Value(fsctx.FileModelCtx).(model.File)
|
|
|
if !ok {
|
|
|
return ErrObjectNotExist
|
|
|
}
|
|
|
|
|
|
newFile.SetModel(&originFile)
|
|
|
|
|
|
err := originFile.UpdateSize(newFile.Info().Size)
|
|
|
if err != nil {
|
|
|
return err
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// SlaveAfterUpload Slave模式下上传完成钩子
|
|
|
func SlaveAfterUpload(session *serializer.UploadSession) Hook {
|
|
|
return func(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileInfo := fileHeader.Info()
|
|
|
|
|
|
// 构造一个model.File,用于生成缩略图
|
|
|
file := model.File{
|
|
|
Name: fileInfo.FileName,
|
|
|
SourceName: fileInfo.SavePath,
|
|
|
}
|
|
|
fs.GenerateThumbnail(ctx, &file)
|
|
|
|
|
|
if session.Callback == "" {
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// 发送回调请求
|
|
|
callbackBody := serializer.UploadCallback{
|
|
|
PicInfo: file.PicInfo,
|
|
|
}
|
|
|
|
|
|
return cluster.RemoteCallback(session.Callback, callbackBody)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// GenericAfterUpload 文件上传完成后,包含数据库操作
|
|
|
func GenericAfterUpload(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileInfo := fileHeader.Info()
|
|
|
|
|
|
// 检查路径是否存在,不存在就创建
|
|
|
isExist, folder := fs.IsPathExist(fileInfo.VirtualPath)
|
|
|
if !isExist {
|
|
|
newFolder, err := fs.CreateDirectory(ctx, fileInfo.VirtualPath)
|
|
|
if err != nil {
|
|
|
return err
|
|
|
}
|
|
|
folder = newFolder
|
|
|
}
|
|
|
|
|
|
// 检查文件是否存在
|
|
|
if ok, file := fs.IsChildFileExist(
|
|
|
folder,
|
|
|
fileInfo.FileName,
|
|
|
); ok {
|
|
|
if file.UploadSessionID != nil {
|
|
|
return ErrFileUploadSessionExisted
|
|
|
}
|
|
|
|
|
|
return ErrFileExisted
|
|
|
}
|
|
|
|
|
|
// 向数据库中插入记录
|
|
|
file, err := fs.AddFile(ctx, folder, fileHeader)
|
|
|
if err != nil {
|
|
|
return ErrInsertFileRecord
|
|
|
}
|
|
|
fileHeader.SetModel(file)
|
|
|
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookGenerateThumb 生成缩略图
|
|
|
func HookGenerateThumb(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
// 异步尝试生成缩略图
|
|
|
fileMode := fileHeader.Info().Model.(*model.File)
|
|
|
if fs.Policy.IsThumbGenerateNeeded() {
|
|
|
fs.recycleLock.Lock()
|
|
|
go func() {
|
|
|
defer fs.recycleLock.Unlock()
|
|
|
_, _ = fs.Handler.Delete(ctx, []string{fileMode.SourceName + conf.ThumbConfig.FileSuffix})
|
|
|
fs.GenerateThumbnail(ctx, fileMode)
|
|
|
}()
|
|
|
}
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookClearFileHeaderSize 将FileHeader大小设定为0
|
|
|
func HookClearFileHeaderSize(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileHeader.SetSize(0)
|
|
|
return nil
|
|
|
}
|
|
|
|
|
|
// HookTruncateFileTo 将物理文件截断至 size
|
|
|
func HookTruncateFileTo(size uint64) Hook {
|
|
|
return func(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
if handler, ok := fs.Handler.(local.Driver); ok {
|
|
|
return handler.Truncate(ctx, fileHeader.Info().SavePath, size)
|
|
|
}
|
|
|
|
|
|
return nil
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// HookChunkUploadFinished 单个分片上传结束后
|
|
|
func HookChunkUploaded(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileInfo := fileHeader.Info()
|
|
|
|
|
|
// 更新文件大小
|
|
|
return fileInfo.Model.(*model.File).UpdateSize(fileInfo.AppendStart + fileInfo.Size)
|
|
|
}
|
|
|
|
|
|
// HookChunkUploadFailed 单个分片上传失败后
|
|
|
func HookChunkUploadFailed(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileInfo := fileHeader.Info()
|
|
|
|
|
|
// 更新文件大小
|
|
|
return fileInfo.Model.(*model.File).UpdateSize(fileInfo.AppendStart)
|
|
|
}
|
|
|
|
|
|
// HookPopPlaceholderToFile 将占位文件提升为正式文件
|
|
|
func HookPopPlaceholderToFile(picInfo string) Hook {
|
|
|
return func(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
fileInfo := fileHeader.Info()
|
|
|
fileModel := fileInfo.Model.(*model.File)
|
|
|
return fileModel.PopChunkToFile(fileInfo.LastModified, picInfo)
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// HookChunkUploadFinished 分片上传结束后处理文件
|
|
|
func HookDeleteUploadSession(id string) Hook {
|
|
|
return func(ctx context.Context, fs *FileSystem, fileHeader fsctx.FileHeader) error {
|
|
|
cache.Deletes([]string{id}, UploadSessionCachePrefix)
|
|
|
return nil
|
|
|
}
|
|
|
}
|