You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
cloudreve/pkg/filesystem/filesystem.go

161 lines
3.6 KiB

5 years ago
package filesystem
import (
5 years ago
"context"
"github.com/HFO4/cloudreve/models"
"github.com/HFO4/cloudreve/pkg/filesystem/local"
"github.com/HFO4/cloudreve/pkg/util"
"github.com/gin-gonic/gin"
5 years ago
"io"
"path/filepath"
5 years ago
)
// FileData 上传来的文件数据处理器
type FileData interface {
io.Reader
io.Closer
GetSize() uint64
GetMIMEType() string
GetFileName() string
5 years ago
}
// Handler 存储策略适配器
type Handler interface {
// 上传文件
Put(ctx context.Context, file io.ReadCloser, dst string) error
// 删除一个或多个文件
Delete(ctx context.Context, files []string) ([]string, error)
}
5 years ago
// FileSystem 管理文件的文件系统
type FileSystem struct {
/*
*/
5 years ago
User *model.User
/*
*/
// 上传文件前
BeforeUpload func(ctx context.Context, fs *FileSystem) error
// 上传文件后
5 years ago
AfterUpload func(ctx context.Context, fs *FileSystem) error
// 文件保存成功,插入数据库验证失败后
AfterValidateFailed func(ctx context.Context, fs *FileSystem) error
// 用户取消上传后
AfterUploadCanceled func(ctx context.Context, fs *FileSystem) error
/*
*/
Handler Handler
5 years ago
}
// NewFileSystem 初始化一个文件系统
func NewFileSystem(user *model.User) (*FileSystem, error) {
var handler Handler
// 根据存储策略类型分配适配器
switch user.Policy.Type {
case "local":
handler = local.Handler{}
default:
return nil, UnknownPolicyTypeError
}
// TODO 分配默认钩子
return &FileSystem{
User: user,
Handler: handler,
}, nil
}
/* ================
================
*/
5 years ago
// Upload 上传文件
5 years ago
func (fs *FileSystem) Upload(ctx context.Context, file FileData) (err error) {
ctx = context.WithValue(ctx, FileCtx, file)
// 上传前的钩子
if fs.BeforeUpload != nil {
err = fs.BeforeUpload(ctx, fs)
if err != nil {
return err
}
}
// 生成文件名和路径
savePath := fs.GenerateSavePath(file)
// 处理客户端未完成上传时,关闭连接
go fs.CancelUpload(ctx, savePath, file)
// 保存文件
err = fs.Handler.Put(ctx, file, savePath)
if err != nil {
return err
}
// 上传完成后的钩子
if fs.AfterUpload != nil {
ctx = context.WithValue(ctx, SavePathCtx, savePath)
err = fs.AfterUpload(ctx, fs)
if err != nil {
// 上传完成后续处理失败
if fs.AfterValidateFailed != nil {
followUpErr := fs.AfterValidateFailed(ctx, fs)
// 失败后再失败...
if followUpErr != nil {
util.Log().Warning("AfterValidateFailed 钩子执行失败,%s", followUpErr)
}
}
return err
}
}
5 years ago
return nil
}
// GenerateSavePath 生成要存放文件的路径
func (fs *FileSystem) GenerateSavePath(file FileData) string {
return filepath.Join(
fs.User.Policy.GeneratePath(fs.User.Model.ID),
fs.User.Policy.GenerateFileName(fs.User.Model.ID, file.GetFileName()),
)
}
// CancelUpload 监测客户端取消上传
func (fs *FileSystem) CancelUpload(ctx context.Context, path string, file FileData) {
ginCtx := ctx.Value(GinCtx).(*gin.Context)
select {
case <-ctx.Done():
// 客户端正常关闭,不执行操作
case <-ginCtx.Request.Context().Done():
// 客户端取消了上传
if fs.AfterUploadCanceled == nil {
return
}
ctx = context.WithValue(ctx, SavePathCtx, path)
err := fs.AfterUploadCanceled(ctx, fs)
if err != nil {
util.Log().Warning("执行 AfterUploadCanceled 钩子出错,%s", err)
}
}
}
/* =================
/
=================
*/
// IsPathExist 返回给定目录是否存在
func (fs *FileSystem) IsPathExist(path string) bool {
_, err := model.GetFolderByPath(path, fs.User.ID)
return err == nil
}