Feat: GET interface for remote policy

pull/247/head
HFO4 5 years ago
parent 0f93864c8e
commit 1393659668

@ -54,6 +54,7 @@ func (fs *FileSystem) Compress(ctx context.Context, folderIDs, fileIDs []uint) (
defer zipWriter.Close() defer zipWriter.Close()
ctx, _ = context.WithCancel(context.Background()) ctx, _ = context.WithCancel(context.Background())
// ctx = context.WithValue(ctx, fsctx.UserCtx, *fs.User)
// 压缩各个目录及文件 // 压缩各个目录及文件
for i := 0; i < len(folders); i++ { for i := 0; i < len(folders); i++ {
fs.doCompress(ctx, nil, &folders[i], zipWriter, true) fs.doCompress(ctx, nil, &folders[i], zipWriter, true)

@ -17,4 +17,6 @@ const (
HTTPCtx HTTPCtx
// UploadPolicyCtx 上传策略一般为slave模式下使用 // UploadPolicyCtx 上传策略一般为slave模式下使用
UploadPolicyCtx UploadPolicyCtx
// UserCtx 用户
UserCtx
) )

@ -45,9 +45,32 @@ func (handler Handler) getAPI(scope string) string {
} }
// Get 获取文件内容 // Get 获取文件内容
// TODO 测试
func (handler Handler) Get(ctx context.Context, path string) (response.RSCloser, error) { func (handler Handler) Get(ctx context.Context, path string) (response.RSCloser, error) {
// 尝试获取速度限制 TODO 是否需要在这里限制?
speedLimit := 0
if user, ok := ctx.Value(fsctx.UserCtx).(model.User); ok {
speedLimit = user.Group.SpeedLimit
}
return nil, nil // 获取文件源地址
downloadURL, err := handler.Source(ctx, path, url.URL{}, 0, true, speedLimit)
if err != nil {
return nil, err
}
// 获取文件数据流
resp, err := handler.Client.Request(
"GET",
downloadURL,
nil,
).GetRSCloser()
if err != nil {
return nil, err
}
return resp, nil
} }
// Put 将文件流保存到指定目录 // Put 将文件流保存到指定目录
@ -125,9 +148,10 @@ func (handler Handler) Source(
isDownload bool, isDownload bool,
speed int, speed int,
) (string, error) { ) (string, error) {
file, ok := ctx.Value(fsctx.FileModelCtx).(model.File) // 尝试从上下文获取文件名
if !ok { fileName := "file"
return "", errors.New("无法获取文件记录上下文") if file, ok := ctx.Value(fsctx.FileModelCtx).(model.File); ok {
fileName = file.Name
} }
serverURL, err := url.Parse(handler.Policy.Server) serverURL, err := url.Parse(handler.Policy.Server)
@ -144,10 +168,10 @@ func (handler Handler) Source(
} }
// 签名下载地址 // 签名下载地址
sourcePath := base64.RawURLEncoding.EncodeToString([]byte(file.SourceName)) sourcePath := base64.RawURLEncoding.EncodeToString([]byte(path))
signedURI, err = auth.SignURI( signedURI, err = auth.SignURI(
handler.AuthInstance, handler.AuthInstance,
fmt.Sprintf("%s/%d/%s/%s", controller, speed, sourcePath, file.Name), fmt.Sprintf("%s/%d/%s/%s", controller, speed, sourcePath, fileName),
ttl, ttl,
) )

@ -58,12 +58,13 @@ func TestHandler_Source(t *testing.T) {
// 无法获取上下文 // 无法获取上下文
{ {
handler := Handler{ handler := Handler{
Policy: &model.Policy{Server: "/"},
AuthInstance: auth.HMACAuth{}, AuthInstance: auth.HMACAuth{},
} }
ctx := context.Background() ctx := context.Background()
res, err := handler.Source(ctx, "", url.URL{}, 0, true, 0) res, err := handler.Source(ctx, "", url.URL{}, 0, true, 0)
asserts.Error(err) asserts.NoError(err)
asserts.Empty(res) asserts.NotEmpty(res)
} }
// 成功 // 成功

@ -1,8 +1,10 @@
package request package request
import ( import (
"errors"
"fmt" "fmt"
"github.com/HFO4/cloudreve/pkg/auth" "github.com/HFO4/cloudreve/pkg/auth"
"github.com/HFO4/cloudreve/pkg/filesystem/response"
"io" "io"
"io/ioutil" "io/ioutil"
"net/http" "net/http"
@ -120,3 +122,29 @@ func (resp Response) GetResponse(expectStatus int) (string, error) {
} }
return string(respBody), err return string(respBody), err
} }
type nopRSCloser struct {
body io.ReadCloser
}
// GetRSCloser 返回带有空seeker的body reader
func (resp Response) GetRSCloser() (response.RSCloser, error) {
return nopRSCloser{
body: resp.Response.Body,
}, resp.Err
}
// Read 实现 nopRSCloser reader
func (instance nopRSCloser) Read(p []byte) (n int, err error) {
return instance.body.Read(p)
}
// 实现 nopRSCloser closer
func (instance nopRSCloser) Close() error {
return instance.body.Close()
}
// 实现 nopRSCloser seeker
func (instance nopRSCloser) Seek(offset int64, whence int) (int64, error) {
return 0, errors.New("未实现")
}

@ -229,14 +229,14 @@ func (h *Handler) handleGetHeadPost(w http.ResponseWriter, r *http.Request, fs *
} }
ctx := r.Context() ctx := r.Context()
rs, err := fs.GetContent(ctx, reqPath)
rs, err := fs.Preview(ctx, reqPath)
if err != nil { if err != nil {
if err == filesystem.ErrObjectNotExist { if err == filesystem.ErrObjectNotExist {
return http.StatusNotFound, err return http.StatusNotFound, err
} }
return http.StatusInternalServerError, err return http.StatusInternalServerError, err
} }
defer rs.Close()
etag, err := findETag(ctx, fs, h.LockSystem[fs.User.ID], reqPath, &fs.FileTarget[0]) etag, err := findETag(ctx, fs, h.LockSystem[fs.User.ID], reqPath, &fs.FileTarget[0])
if err != nil { if err != nil {
@ -244,8 +244,15 @@ func (h *Handler) handleGetHeadPost(w http.ResponseWriter, r *http.Request, fs *
} }
w.Header().Set("ETag", etag) w.Header().Set("ETag", etag)
// 获取文件内容 if !rs.Redirect {
http.ServeContent(w, r, reqPath, fs.FileTarget[0].UpdatedAt, rs) defer rs.Content.Close()
// 获取文件内容
http.ServeContent(w, r, reqPath, fs.FileTarget[0].UpdatedAt, rs.Content)
return 0, nil
}
http.Redirect(w, r, rs.URL, 301)
return 0, nil return 0, nil
} }

Loading…
Cancel
Save