Feat: remote file thumb preview

pull/247/head
HFO4 2020-01-02 13:31:41 +08:00
parent b6102c3ae5
commit 9c48f4b7ad
6 changed files with 37 additions and 15 deletions

View File

@ -3,6 +3,7 @@ package filesystem
import ( import (
"context" "context"
model "github.com/HFO4/cloudreve/models" model "github.com/HFO4/cloudreve/models"
"github.com/HFO4/cloudreve/pkg/conf"
"github.com/HFO4/cloudreve/pkg/filesystem/fsctx" "github.com/HFO4/cloudreve/pkg/filesystem/fsctx"
"github.com/HFO4/cloudreve/pkg/filesystem/response" "github.com/HFO4/cloudreve/pkg/filesystem/response"
"github.com/HFO4/cloudreve/pkg/serializer" "github.com/HFO4/cloudreve/pkg/serializer"
@ -230,13 +231,13 @@ func (fs *FileSystem) GetDownloadURL(ctx context.Context, path string, timeout s
// GetSource 获取可直接访问文件的外链地址 // GetSource 获取可直接访问文件的外链地址
func (fs *FileSystem) GetSource(ctx context.Context, fileID uint) (string, error) { func (fs *FileSystem) GetSource(ctx context.Context, fileID uint) (string, error) {
// 查找文件记录 // 查找文件记录
fileObject, err := model.GetFilesByIDs([]uint{fileID}, fs.User.ID) err := fs.resetFileIDIfNotExist(ctx, fileID)
if err != nil || len(fileObject) == 0 { if err != nil {
return "", ErrObjectNotExist.WithError(err) return "", ErrObjectNotExist.WithError(err)
} }
// 检查存储策略是否可以获得外链 // 检查存储策略是否可以获得外链
if !fileObject[0].GetPolicy().IsOriginLinkEnable { if !fs.Policy.IsOriginLinkEnable {
return "", serializer.NewError( return "", serializer.NewError(
serializer.CodePolicyNotAllowed, serializer.CodePolicyNotAllowed,
"当前存储策略无法获得外链", "当前存储策略无法获得外链",
@ -244,7 +245,7 @@ func (fs *FileSystem) GetSource(ctx context.Context, fileID uint) (string, error
) )
} }
source, err := fs.signURL(ctx, &fileObject[0], 0, false) source, err := fs.signURL(ctx, &fs.FileTarget[0], 0, false)
if err != nil { if err != nil {
return "", serializer.NewError(serializer.CodeNotSet, "无法获取外链", err) return "", serializer.NewError(serializer.CodeNotSet, "无法获取外链", err)
} }
@ -307,6 +308,12 @@ func (fs *FileSystem) resetPolicyToFirstFile(ctx context.Context) error {
if len(fs.FileTarget) == 0 { if len(fs.FileTarget) == 0 {
return ErrObjectNotExist return ErrObjectNotExist
} }
// 从机模式不进行操作
if conf.SystemConfig.Mode == "slave" {
return nil
}
fs.Policy = fs.FileTarget[0].GetPolicy() fs.Policy = fs.FileTarget[0].GetPolicy()
err := fs.dispatchHandler() err := fs.dispatchHandler()
if err != nil { if err != nil {

View File

@ -277,13 +277,13 @@ func TestFileSystem_deleteGroupedFile(t *testing.T) {
func TestFileSystem_GetSource(t *testing.T) { func TestFileSystem_GetSource(t *testing.T) {
asserts := assert.New(t) asserts := assert.New(t)
ctx := context.Background() ctx := context.Background()
fs := FileSystem{
User: &model.User{Model: gorm.Model{ID: 1}},
}
auth.General = auth.HMACAuth{SecretKey: []byte("123")} auth.General = auth.HMACAuth{SecretKey: []byte("123")}
// 正常 // 正常
{ {
fs := FileSystem{
User: &model.User{Model: gorm.Model{ID: 1}},
}
// 清空缓存 // 清空缓存
err := cache.Deletes([]string{"siteURL"}, "setting_") err := cache.Deletes([]string{"siteURL"}, "setting_")
asserts.NoError(err) asserts.NoError(err)
@ -312,6 +312,9 @@ func TestFileSystem_GetSource(t *testing.T) {
// 文件不存在 // 文件不存在
{ {
fs := FileSystem{
User: &model.User{Model: gorm.Model{ID: 1}},
}
// 清空缓存 // 清空缓存
err := cache.Deletes([]string{"siteURL"}, "setting_") err := cache.Deletes([]string{"siteURL"}, "setting_")
asserts.NoError(err) asserts.NoError(err)
@ -332,6 +335,9 @@ func TestFileSystem_GetSource(t *testing.T) {
// 未知上传策略 // 未知上传策略
{ {
fs := FileSystem{
User: &model.User{Model: gorm.Model{ID: 1}},
}
// 清空缓存 // 清空缓存
err := cache.Deletes([]string{"siteURL"}, "setting_") err := cache.Deletes([]string{"siteURL"}, "setting_")
asserts.NoError(err) asserts.NoError(err)
@ -358,6 +364,9 @@ func TestFileSystem_GetSource(t *testing.T) {
// 不允许获取外链 // 不允许获取外链
{ {
fs := FileSystem{
User: &model.User{Model: gorm.Model{ID: 1}},
}
// 清空缓存 // 清空缓存
err := cache.Deletes([]string{"siteURL"}, "setting_") err := cache.Deletes([]string{"siteURL"}, "setting_")
asserts.NoError(err) asserts.NoError(err)

View File

@ -144,7 +144,6 @@ func (fs *FileSystem) dispatchHandler() error {
currentPolicy = fs.Policy currentPolicy = fs.Policy
} }
// 根据存储策略类型分配适配器
switch policyType { switch policyType {
case "mock": case "mock":
return nil return nil

View File

@ -106,7 +106,7 @@ func (handler Handler) Thumb(ctx context.Context, path string) (*response.Conten
sourcePath := base64.RawURLEncoding.EncodeToString([]byte(path)) sourcePath := base64.RawURLEncoding.EncodeToString([]byte(path))
thumbURL := handler.getAPI("thumb") + "/" + sourcePath thumbURL := handler.getAPI("thumb") + "/" + sourcePath
ttl := model.GetIntSetting("slave_api_timeout", 60) ttl := model.GetIntSetting("slave_api_timeout", 60)
signedThumbURL, err := auth.SignURI(handler.AuthInstance, thumbURL, int64(ttl)) signedThumbURL, err := auth.SignURI(handler.AuthInstance, thumbURL, time.Now().Unix()+int64(ttl))
if err != nil { if err != nil {
return nil, err return nil, err
} }

View File

@ -11,7 +11,6 @@ import (
"github.com/HFO4/cloudreve/pkg/util" "github.com/HFO4/cloudreve/pkg/util"
"github.com/HFO4/cloudreve/service/explorer" "github.com/HFO4/cloudreve/service/explorer"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"io"
"net/http" "net/http"
"net/url" "net/url"
"strconv" "strconv"
@ -130,12 +129,9 @@ func Thumb(c *gin.Context) {
c.Redirect(http.StatusMovedPermanently, resp.URL) c.Redirect(http.StatusMovedPermanently, resp.URL)
return return
} }
http.ServeContent(c.Writer, c.Request, "thumb.png", fs.FileTarget[0].UpdatedAt, resp.Content)
// 检查是否需要关闭文件 defer resp.Content.Close()
if fc, ok := resp.Content.(io.Closer); ok { http.ServeContent(c.Writer, c.Request, "thumb.png", fs.FileTarget[0].UpdatedAt, resp.Content)
defer fc.Close()
}
} }

View File

@ -388,5 +388,16 @@ func (service *SlaveFileService) Thumb(ctx context.Context, c *gin.Context) seri
if err != nil { if err != nil {
return serializer.ParamErr("无法解析的文件地址", err) return serializer.ParamErr("无法解析的文件地址", err)
} }
fs.FileTarget = []model.File{{SourceName: string(fileSource), PicInfo: "1,1"}}
// 获取缩略图
resp, err := fs.GetThumb(ctx, 0)
if err != nil {
return serializer.Err(serializer.CodeNotSet, "无法获取缩略图", err)
}
defer resp.Content.Close()
http.ServeContent(c.Writer, c.Request, "thumb.png", time.Now(), resp.Content)
return serializer.Response{Code: 0}
} }