mirror of
				https://github.com/cloudreve/cloudreve.git
				synced 2025-10-31 00:27:31 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			105 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			105 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package filesystem
 | ||
| 
 | ||
| import (
 | ||
| 	"context"
 | ||
| 	model "github.com/HFO4/cloudreve/models"
 | ||
| 	"github.com/HFO4/cloudreve/pkg/util"
 | ||
| 	"github.com/juju/ratelimit"
 | ||
| 	"io"
 | ||
| )
 | ||
| 
 | ||
| /* ============
 | ||
| 	 文件相关
 | ||
|    ============
 | ||
| */
 | ||
| 
 | ||
| // 限速后的ReaderSeeker
 | ||
| type lrs struct {
 | ||
| 	io.ReadSeeker
 | ||
| 	r io.Reader
 | ||
| }
 | ||
| 
 | ||
| func (r lrs) Read(p []byte) (int, error) {
 | ||
| 	return r.r.Read(p)
 | ||
| }
 | ||
| 
 | ||
| // withSpeedLimit 给原有的ReadSeeker加上限速
 | ||
| func withSpeedLimit(rs io.ReadSeeker, speed int) io.ReadSeeker {
 | ||
| 	bucket := ratelimit.NewBucketWithRate(float64(speed), int64(speed))
 | ||
| 	lrs := lrs{rs, ratelimit.Reader(rs, bucket)}
 | ||
| 	return lrs
 | ||
| }
 | ||
| 
 | ||
| // AddFile 新增文件记录
 | ||
| func (fs *FileSystem) AddFile(ctx context.Context, parent *model.Folder) (*model.File, error) {
 | ||
| 	file := ctx.Value(FileHeaderCtx).(FileHeader)
 | ||
| 	filePath := ctx.Value(SavePathCtx).(string)
 | ||
| 
 | ||
| 	newFile := model.File{
 | ||
| 		Name:       file.GetFileName(),
 | ||
| 		SourceName: filePath,
 | ||
| 		UserID:     fs.User.ID,
 | ||
| 		Size:       file.GetSize(),
 | ||
| 		FolderID:   parent.ID,
 | ||
| 		PolicyID:   fs.User.Policy.ID,
 | ||
| 		Dir:        parent.PositionAbsolute,
 | ||
| 	}
 | ||
| 	_, err := newFile.Create()
 | ||
| 
 | ||
| 	if err != nil {
 | ||
| 		return nil, err
 | ||
| 	}
 | ||
| 
 | ||
| 	return &newFile, nil
 | ||
| }
 | ||
| 
 | ||
| // GetDownloadContent 获取用于下载的文件流
 | ||
| func (fs *FileSystem) GetDownloadContent(ctx context.Context, path string) (io.ReadSeeker, error) {
 | ||
| 	// 获取原始文件流
 | ||
| 	rs, err := fs.GetContent(ctx, path)
 | ||
| 	if err != nil {
 | ||
| 		return nil, err
 | ||
| 	}
 | ||
| 
 | ||
| 	// 如果用户组有速度限制,就返回限制流速的ReaderSeeker
 | ||
| 	if fs.User.Group.SpeedLimit != 0 {
 | ||
| 		return withSpeedLimit(rs, fs.User.Group.SpeedLimit), nil
 | ||
| 	}
 | ||
| 	// 否则返回原始流
 | ||
| 	return rs, nil
 | ||
| 
 | ||
| }
 | ||
| 
 | ||
| // GetContent 获取文件内容,path为虚拟路径
 | ||
| // TODO:测试
 | ||
| func (fs *FileSystem) GetContent(ctx context.Context, path string) (io.ReadSeeker, error) {
 | ||
| 	// 触发`下载前`钩子
 | ||
| 	err := fs.Trigger(ctx, fs.BeforeFileDownload)
 | ||
| 	if err != nil {
 | ||
| 		util.Log().Debug("BeforeFileDownload 钩子执行失败,%s", err)
 | ||
| 		return nil, err
 | ||
| 	}
 | ||
| 
 | ||
| 	// 找到文件
 | ||
| 	exist, file := fs.IsFileExist(path)
 | ||
| 	if !exist {
 | ||
| 		return nil, ErrObjectNotExist
 | ||
| 	}
 | ||
| 	fs.Target = &file
 | ||
| 
 | ||
| 	// 将当前存储策略重设为文件使用的
 | ||
| 	fs.Policy = file.GetPolicy()
 | ||
| 	err = fs.dispatchHandler()
 | ||
| 	if err != nil {
 | ||
| 		return nil, err
 | ||
| 	}
 | ||
| 
 | ||
| 	// 获取文件流
 | ||
| 	rs, err := fs.Handler.Get(ctx, file.SourceName)
 | ||
| 	if err != nil {
 | ||
| 		return nil, ErrIO.WithError(err)
 | ||
| 	}
 | ||
| 
 | ||
| 	return rs, nil
 | ||
| }
 | 
