mirror of https://github.com/cloudreve/Cloudreve
Feat: client-upload file in oss
parent
0e62665d7f
commit
dd198becce
1
go.mod
1
go.mod
|
@ -4,6 +4,7 @@ go 1.12
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/DATA-DOG/go-sqlmock v1.3.3
|
github.com/DATA-DOG/go-sqlmock v1.3.3
|
||||||
|
github.com/aliyun/aliyun-oss-go-sdk v2.0.5+incompatible
|
||||||
github.com/cloudflare/cfssl v0.0.0-20190726000631-633726f6bcb7
|
github.com/cloudflare/cfssl v0.0.0-20190726000631-633726f6bcb7
|
||||||
github.com/duo-labs/webauthn v0.0.0-20191119193225-4bf9a0f776d4
|
github.com/duo-labs/webauthn v0.0.0-20191119193225-4bf9a0f776d4
|
||||||
github.com/fatih/color v1.7.0
|
github.com/fatih/color v1.7.0
|
||||||
|
|
3
go.sum
3
go.sum
|
@ -10,6 +10,8 @@ github.com/agl/ed25519 v0.0.0-20170116200512-5312a6153412 h1:w1UutsfOrms1J05zt7I
|
||||||
github.com/agl/ed25519 v0.0.0-20170116200512-5312a6153412/go.mod h1:WPjqKcmVOxf0XSf3YxCJs6N6AOSrOx3obionmG7T0y0=
|
github.com/agl/ed25519 v0.0.0-20170116200512-5312a6153412/go.mod h1:WPjqKcmVOxf0XSf3YxCJs6N6AOSrOx3obionmG7T0y0=
|
||||||
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc=
|
||||||
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0=
|
||||||
|
github.com/aliyun/aliyun-oss-go-sdk v2.0.5+incompatible h1:A3oZlWPD/Poa19FvNbw+Zu4yKAurDBTjlRDilYGBiS4=
|
||||||
|
github.com/aliyun/aliyun-oss-go-sdk v2.0.5+incompatible/go.mod h1:T/Aws4fEfogEE9v+HPhhw+CntffsBHJ8nXQCwKr0/g8=
|
||||||
github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ=
|
github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ=
|
||||||
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
|
github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q=
|
||||||
github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff h1:RmdPFa+slIr4SCBg4st/l/vZWVe9QJKMXGO60Bxbe04=
|
github.com/boj/redistore v0.0.0-20180917114910-cd5dcc76aeff h1:RmdPFa+slIr4SCBg4st/l/vZWVe9QJKMXGO60Bxbe04=
|
||||||
|
@ -213,6 +215,7 @@ golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7w
|
||||||
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
|
||||||
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
|
||||||
|
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c h1:fqgJT0MGcGpPgpWU7VRdRjuArfcOvC4AoJmILihzhDg=
|
||||||
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
|
||||||
golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
|
||||||
|
|
|
@ -165,7 +165,6 @@ func RemoteCallbackAuth() gin.HandlerFunc {
|
||||||
}
|
}
|
||||||
|
|
||||||
// QiniuCallbackAuth 七牛回调签名验证
|
// QiniuCallbackAuth 七牛回调签名验证
|
||||||
// TODO 测试
|
|
||||||
func QiniuCallbackAuth() gin.HandlerFunc {
|
func QiniuCallbackAuth() gin.HandlerFunc {
|
||||||
return func(c *gin.Context) {
|
return func(c *gin.Context) {
|
||||||
// 验证key并查找用户
|
// 验证key并查找用户
|
||||||
|
@ -194,3 +193,20 @@ func QiniuCallbackAuth() gin.HandlerFunc {
|
||||||
c.Next()
|
c.Next()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// OSSCallbackAuth 阿里云OSS回调签名验证
|
||||||
|
func OSSCallbackAuth() gin.HandlerFunc {
|
||||||
|
return func(c *gin.Context) {
|
||||||
|
// 验证key并查找用户
|
||||||
|
resp, _ := uploadCallbackCheck(c)
|
||||||
|
if resp.Code != 0 {
|
||||||
|
c.JSON(401, serializer.QiniuCallbackFailed{Error: resp.Msg})
|
||||||
|
c.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO 验证OSS给出的签名
|
||||||
|
|
||||||
|
c.Next()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -424,3 +424,53 @@ func TestQiniuCallbackAuth(t *testing.T) {
|
||||||
asserts.True(c.IsAborted())
|
asserts.True(c.IsAborted())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestOSSCallbackAuth(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
rec := httptest.NewRecorder()
|
||||||
|
AuthFunc := OSSCallbackAuth()
|
||||||
|
|
||||||
|
// Callback Key 相关验证失败
|
||||||
|
{
|
||||||
|
c, _ := gin.CreateTestContext(rec)
|
||||||
|
c.Params = []gin.Param{
|
||||||
|
{"key", "testOSSBackRemote"},
|
||||||
|
}
|
||||||
|
c.Request, _ = http.NewRequest("POST", "/api/v3/callback/oss/testQiniuBackRemote", nil)
|
||||||
|
AuthFunc(c)
|
||||||
|
asserts.True(c.IsAborted())
|
||||||
|
}
|
||||||
|
|
||||||
|
// 成功
|
||||||
|
{
|
||||||
|
cache.Set(
|
||||||
|
"callback_testCallBackOSS",
|
||||||
|
serializer.UploadSession{
|
||||||
|
UID: 1,
|
||||||
|
PolicyID: 2,
|
||||||
|
VirtualPath: "/",
|
||||||
|
},
|
||||||
|
0,
|
||||||
|
)
|
||||||
|
cache.Deletes([]string{"1"}, "policy_")
|
||||||
|
mock.ExpectQuery("SELECT(.+)users(.+)").
|
||||||
|
WillReturnRows(sqlmock.NewRows([]string{"id", "group_id"}).AddRow(1, 1))
|
||||||
|
mock.ExpectQuery("SELECT(.+)groups(.+)").
|
||||||
|
WillReturnRows(sqlmock.NewRows([]string{"id", "policies"}).AddRow(1, "[2]"))
|
||||||
|
mock.ExpectQuery("SELECT(.+)policies(.+)").
|
||||||
|
WillReturnRows(sqlmock.NewRows([]string{"id", "access_key", "secret_key"}).AddRow(2, "123", "123"))
|
||||||
|
c, _ := gin.CreateTestContext(rec)
|
||||||
|
c.Params = []gin.Param{
|
||||||
|
{"key", "testCallBackOSS"},
|
||||||
|
}
|
||||||
|
c.Request, _ = http.NewRequest("POST", "/api/v3/callback/qiniu/testCallBackOSS", nil)
|
||||||
|
mac := qbox.NewMac("123", "123")
|
||||||
|
token, err := mac.SignRequest(c.Request)
|
||||||
|
asserts.NoError(err)
|
||||||
|
c.Request.Header["Authorization"] = []string{"QBox " + token}
|
||||||
|
AuthFunc(c)
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.False(c.IsAborted())
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
|
@ -111,7 +111,7 @@ func (policy *Policy) GeneratePath(uid uint, origin string) string {
|
||||||
func (policy *Policy) GenerateFileName(uid uint, origin string) string {
|
func (policy *Policy) GenerateFileName(uid uint, origin string) string {
|
||||||
// 未开启自动重命名时,直接返回原始文件名
|
// 未开启自动重命名时,直接返回原始文件名
|
||||||
if !policy.AutoRename {
|
if !policy.AutoRename {
|
||||||
return origin
|
return policy.getOriginNameRule(origin)
|
||||||
}
|
}
|
||||||
|
|
||||||
fileRule := policy.FileNameRule
|
fileRule := policy.FileNameRule
|
||||||
|
@ -125,28 +125,31 @@ func (policy *Policy) GenerateFileName(uid uint, origin string) string {
|
||||||
"{date}": time.Now().Format("20060102"),
|
"{date}": time.Now().Format("20060102"),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
replaceTable["{originname}"] = policy.getOriginNameRule(origin)
|
||||||
|
|
||||||
|
fileRule = util.Replace(replaceTable, fileRule)
|
||||||
|
return fileRule
|
||||||
|
}
|
||||||
|
|
||||||
|
func (policy Policy) getOriginNameRule(origin string) string {
|
||||||
// 部分存储策略可以使用{origin}代表原始文件名
|
// 部分存储策略可以使用{origin}代表原始文件名
|
||||||
if origin == "" {
|
if origin == "" {
|
||||||
// 如果上游未传回原始文件名,则使用占位符,让云存储端替换
|
// 如果上游未传回原始文件名,则使用占位符,让云存储端替换
|
||||||
switch policy.Type {
|
switch policy.Type {
|
||||||
case "qiniu":
|
case "qiniu":
|
||||||
// 七牛会将$(fname)自动替换为原始文件名
|
// 七牛会将$(fname)自动替换为原始文件名
|
||||||
replaceTable["{originname}"] = "$(fname)"
|
return "$(fname)"
|
||||||
case "local", "remote":
|
case "local", "remote":
|
||||||
replaceTable["{originname}"] = origin
|
return origin
|
||||||
case "oss":
|
case "oss":
|
||||||
// OSS会将${filename}自动替换为原始文件名
|
// OSS会将${filename}自动替换为原始文件名
|
||||||
replaceTable["{originname}"] = "${filename}"
|
return "${filename}"
|
||||||
case "upyun":
|
case "upyun":
|
||||||
// Upyun会将{filename}{.suffix}自动替换为原始文件名
|
// Upyun会将{filename}{.suffix}自动替换为原始文件名
|
||||||
replaceTable["{originname}"] = "{filename}{.suffix}"
|
return "{filename}{.suffix}"
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
replaceTable["{originname}"] = origin
|
|
||||||
}
|
}
|
||||||
|
return origin
|
||||||
fileRule = util.Replace(replaceTable, fileRule)
|
|
||||||
return fileRule
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// IsDirectlyPreview 返回此策略下文件是否可以直接预览(不需要重定向)
|
// IsDirectlyPreview 返回此策略下文件是否可以直接预览(不需要重定向)
|
||||||
|
@ -172,6 +175,8 @@ func (policy *Policy) GetUploadURL() string {
|
||||||
controller, _ = url.Parse("/api/v3/file/upload")
|
controller, _ = url.Parse("/api/v3/file/upload")
|
||||||
case "remote":
|
case "remote":
|
||||||
controller, _ = url.Parse("/api/v3/slave/upload")
|
controller, _ = url.Parse("/api/v3/slave/upload")
|
||||||
|
case "oss":
|
||||||
|
return policy.BaseURL
|
||||||
default:
|
default:
|
||||||
controller, _ = url.Parse("")
|
controller, _ = url.Parse("")
|
||||||
}
|
}
|
||||||
|
|
|
@ -91,47 +91,62 @@ func TestPolicy_GeneratePath(t *testing.T) {
|
||||||
|
|
||||||
func TestPolicy_GenerateFileName(t *testing.T) {
|
func TestPolicy_GenerateFileName(t *testing.T) {
|
||||||
asserts := assert.New(t)
|
asserts := assert.New(t)
|
||||||
testPolicy := Policy{
|
// 重命名关闭
|
||||||
AutoRename: true,
|
{
|
||||||
|
testPolicy := Policy{
|
||||||
|
AutoRename: false,
|
||||||
|
}
|
||||||
|
testPolicy.FileNameRule = "{randomkey16}"
|
||||||
|
asserts.Equal("123.txt", testPolicy.GenerateFileName(1, "123.txt"))
|
||||||
|
|
||||||
|
testPolicy.Type = "oss"
|
||||||
|
asserts.Equal("${filename}", testPolicy.GenerateFileName(1, ""))
|
||||||
}
|
}
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{randomkey16}"
|
// 重命名开启
|
||||||
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 16)
|
{
|
||||||
|
testPolicy := Policy{
|
||||||
|
AutoRename: true,
|
||||||
|
}
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{randomkey8}"
|
testPolicy.FileNameRule = "{randomkey16}"
|
||||||
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 8)
|
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 16)
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{timestamp}"
|
testPolicy.FileNameRule = "{randomkey8}"
|
||||||
asserts.Equal(testPolicy.GenerateFileName(1, "123.txt"), strconv.FormatInt(time.Now().Unix(), 10))
|
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 8)
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{uid}"
|
testPolicy.FileNameRule = "{timestamp}"
|
||||||
asserts.Equal(testPolicy.GenerateFileName(1, "123.txt"), strconv.Itoa(int(1)))
|
asserts.Equal(testPolicy.GenerateFileName(1, "123.txt"), strconv.FormatInt(time.Now().Unix(), 10))
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{datetime}"
|
testPolicy.FileNameRule = "{uid}"
|
||||||
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 14)
|
asserts.Equal(testPolicy.GenerateFileName(1, "123.txt"), strconv.Itoa(int(1)))
|
||||||
|
|
||||||
testPolicy.FileNameRule = "{date}"
|
testPolicy.FileNameRule = "{datetime}"
|
||||||
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 8)
|
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 14)
|
||||||
|
|
||||||
testPolicy.FileNameRule = "123{date}ss{datetime}"
|
testPolicy.FileNameRule = "{date}"
|
||||||
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 27)
|
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 8)
|
||||||
|
|
||||||
// 支持{originname}的策略
|
testPolicy.FileNameRule = "123{date}ss{datetime}"
|
||||||
testPolicy.Type = "local"
|
asserts.Len(testPolicy.GenerateFileName(1, "123.txt"), 27)
|
||||||
testPolicy.FileNameRule = "123{originname}"
|
|
||||||
asserts.Equal("123123.txt", testPolicy.GenerateFileName(1, "123.txt"))
|
|
||||||
|
|
||||||
testPolicy.Type = "qiniu"
|
// 支持{originname}的策略
|
||||||
testPolicy.FileNameRule = "{uid}123{originname}"
|
testPolicy.Type = "local"
|
||||||
asserts.Equal("1123123.txt", testPolicy.GenerateFileName(1, "123.txt"))
|
testPolicy.FileNameRule = "123{originname}"
|
||||||
|
asserts.Equal("123123.txt", testPolicy.GenerateFileName(1, "123.txt"))
|
||||||
|
|
||||||
testPolicy.Type = "oss"
|
testPolicy.Type = "qiniu"
|
||||||
testPolicy.FileNameRule = "{uid}123{originname}"
|
testPolicy.FileNameRule = "{uid}123{originname}"
|
||||||
asserts.Equal("1123${filename}", testPolicy.GenerateFileName(1, ""))
|
asserts.Equal("1123123.txt", testPolicy.GenerateFileName(1, "123.txt"))
|
||||||
|
|
||||||
testPolicy.Type = "upyun"
|
testPolicy.Type = "oss"
|
||||||
testPolicy.FileNameRule = "{uid}123{originname}"
|
testPolicy.FileNameRule = "{uid}123{originname}"
|
||||||
asserts.Equal("1123{filename}{.suffix}", testPolicy.GenerateFileName(1, ""))
|
asserts.Equal("1123${filename}", testPolicy.GenerateFileName(1, ""))
|
||||||
|
|
||||||
|
testPolicy.Type = "upyun"
|
||||||
|
testPolicy.FileNameRule = "{uid}123{originname}"
|
||||||
|
asserts.Equal("1123{filename}{.suffix}", testPolicy.GenerateFileName(1, ""))
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -414,6 +414,35 @@ func TestHookClearFileSize(t *testing.T) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestHookUpdateSourceName(t *testing.T) {
|
||||||
|
asserts := assert.New(t)
|
||||||
|
fs := &FileSystem{User: &model.User{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
}}
|
||||||
|
|
||||||
|
// 成功
|
||||||
|
{
|
||||||
|
originFile := model.File{
|
||||||
|
Model: gorm.Model{ID: 1},
|
||||||
|
SourceName: "new.txt",
|
||||||
|
}
|
||||||
|
ctx := context.WithValue(context.Background(), fsctx.FileModelCtx, originFile)
|
||||||
|
mock.ExpectBegin()
|
||||||
|
mock.ExpectExec("UPDATE(.+)").WithArgs("new.txt", sqlmock.AnyArg(), 1).WillReturnResult(sqlmock.NewResult(1, 1))
|
||||||
|
mock.ExpectCommit()
|
||||||
|
err := HookUpdateSourceName(ctx, fs)
|
||||||
|
asserts.NoError(mock.ExpectationsWereMet())
|
||||||
|
asserts.NoError(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 上下文错误
|
||||||
|
{
|
||||||
|
ctx := context.Background()
|
||||||
|
err := HookUpdateSourceName(ctx, fs)
|
||||||
|
asserts.Error(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func TestGenericAfterUpdate(t *testing.T) {
|
func TestGenericAfterUpdate(t *testing.T) {
|
||||||
asserts := assert.New(t)
|
asserts := assert.New(t)
|
||||||
fs := &FileSystem{User: &model.User{
|
fs := &FileSystem{User: &model.User{
|
||||||
|
|
|
@ -2,17 +2,64 @@ package oss
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"crypto/hmac"
|
||||||
|
"crypto/sha1"
|
||||||
|
"encoding/base64"
|
||||||
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
model "github.com/HFO4/cloudreve/models"
|
model "github.com/HFO4/cloudreve/models"
|
||||||
|
"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"
|
||||||
|
"github.com/aliyun/aliyun-oss-go-sdk/oss"
|
||||||
"io"
|
"io"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"path"
|
||||||
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// UploadPolicy 阿里云OSS上传策略
|
||||||
|
type UploadPolicy struct {
|
||||||
|
Expiration string `json:"expiration"`
|
||||||
|
Conditions []interface{} `json:"conditions"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// CallbackPolicy 回调策略
|
||||||
|
type CallbackPolicy struct {
|
||||||
|
CallbackURL string `json:"callbackUrl"`
|
||||||
|
CallbackBody string `json:"callbackBody"`
|
||||||
|
CallbackBodyType string `json:"callbackBodyType"`
|
||||||
|
}
|
||||||
|
|
||||||
// Handler 阿里云OSS策略适配器
|
// Handler 阿里云OSS策略适配器
|
||||||
type Handler struct {
|
type Handler struct {
|
||||||
Policy *model.Policy
|
Policy *model.Policy
|
||||||
|
client *oss.Client
|
||||||
|
bucket *oss.Bucket
|
||||||
|
}
|
||||||
|
|
||||||
|
// InitOSSClient 初始化OSS鉴权客户端
|
||||||
|
func (handler *Handler) InitOSSClient() error {
|
||||||
|
if handler.Policy == nil {
|
||||||
|
return errors.New("存储策略为空")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化客户端
|
||||||
|
client, err := oss.New(handler.Policy.Server, handler.Policy.AccessKey, handler.Policy.SecretKey)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
handler.client = client
|
||||||
|
|
||||||
|
// 初始化存储桶
|
||||||
|
bucket, err := client.Bucket(handler.Policy.BucketName)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
handler.bucket = bucket
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get 获取文件
|
// Get 获取文件
|
||||||
|
@ -50,5 +97,74 @@ func (handler Handler) Source(
|
||||||
|
|
||||||
// Token 获取上传策略和认证Token
|
// Token 获取上传策略和认证Token
|
||||||
func (handler Handler) Token(ctx context.Context, TTL int64, key string) (serializer.UploadCredential, error) {
|
func (handler Handler) Token(ctx context.Context, TTL int64, key string) (serializer.UploadCredential, error) {
|
||||||
return serializer.UploadCredential{}, errors.New("未实现")
|
// 读取上下文中生成的存储路径
|
||||||
|
savePath, ok := ctx.Value(fsctx.SavePathCtx).(string)
|
||||||
|
if !ok {
|
||||||
|
return serializer.UploadCredential{}, errors.New("无法获取存储路径")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 生成回调地址
|
||||||
|
siteURL := model.GetSiteURL()
|
||||||
|
apiBaseURI, _ := url.Parse("/api/v3/callback/oss/" + key)
|
||||||
|
apiURL := siteURL.ResolveReference(apiBaseURI)
|
||||||
|
|
||||||
|
// 回调策略
|
||||||
|
callbackPolicy := CallbackPolicy{
|
||||||
|
CallbackURL: apiURL.String(),
|
||||||
|
CallbackBody: `{"name":${x:fname},"source_name":${object},"size":${size},"pic_info":"${imageInfo.width},${imageInfo.height}"}`,
|
||||||
|
CallbackBodyType: "application/json",
|
||||||
|
}
|
||||||
|
|
||||||
|
// 上传策略
|
||||||
|
postPolicy := UploadPolicy{
|
||||||
|
Expiration: time.Now().UTC().Add(time.Duration(TTL) * time.Second).Format(time.RFC3339),
|
||||||
|
Conditions: []interface{}{
|
||||||
|
map[string]string{"bucket": handler.Policy.BucketName},
|
||||||
|
[]string{"starts-with", "$key", path.Dir(savePath)},
|
||||||
|
[]interface{}{"content-length-range", 0, handler.Policy.MaxSize},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
return handler.getUploadCredential(ctx, postPolicy, callbackPolicy, TTL)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (handler Handler) getUploadCredential(ctx context.Context, policy UploadPolicy, callback CallbackPolicy, TTL int64) (serializer.UploadCredential, error) {
|
||||||
|
// 读取上下文中生成的存储路径
|
||||||
|
savePath, ok := ctx.Value(fsctx.SavePathCtx).(string)
|
||||||
|
if !ok {
|
||||||
|
return serializer.UploadCredential{}, errors.New("无法获取存储路径")
|
||||||
|
}
|
||||||
|
|
||||||
|
// 处理回调策略
|
||||||
|
callbackPolicyEncoded := ""
|
||||||
|
if callback.CallbackURL != "" {
|
||||||
|
callbackPolicyJSON, err := json.Marshal(callback)
|
||||||
|
if err != nil {
|
||||||
|
return serializer.UploadCredential{}, err
|
||||||
|
}
|
||||||
|
callbackPolicyEncoded = base64.StdEncoding.EncodeToString(callbackPolicyJSON)
|
||||||
|
policy.Conditions = append(policy.Conditions, map[string]string{"callback": callbackPolicyEncoded})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编码上传策略
|
||||||
|
policyJSON, err := json.Marshal(policy)
|
||||||
|
if err != nil {
|
||||||
|
return serializer.UploadCredential{}, err
|
||||||
|
}
|
||||||
|
policyEncoded := base64.StdEncoding.EncodeToString(policyJSON)
|
||||||
|
|
||||||
|
// 签名上传策略
|
||||||
|
hmacSign := hmac.New(sha1.New, []byte(handler.Policy.SecretKey))
|
||||||
|
_, err = io.WriteString(hmacSign, policyEncoded)
|
||||||
|
if err != nil {
|
||||||
|
return serializer.UploadCredential{}, err
|
||||||
|
}
|
||||||
|
signature := base64.StdEncoding.EncodeToString(hmacSign.Sum(nil))
|
||||||
|
|
||||||
|
return serializer.UploadCredential{
|
||||||
|
Policy: fmt.Sprintf("%s:%s", callbackPolicyEncoded, policyEncoded),
|
||||||
|
Path: savePath,
|
||||||
|
AccessKey: handler.Policy.AccessKey,
|
||||||
|
Token: signature,
|
||||||
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -18,8 +18,10 @@ type UploadPolicy struct {
|
||||||
|
|
||||||
// UploadCredential 返回给客户端的上传凭证
|
// UploadCredential 返回给客户端的上传凭证
|
||||||
type UploadCredential struct {
|
type UploadCredential struct {
|
||||||
Token string `json:"token"`
|
Token string `json:"token"`
|
||||||
Policy string `json:"policy"`
|
Policy string `json:"policy"`
|
||||||
|
Path string `json:"path"`
|
||||||
|
AccessKey string `json:"ak"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// UploadSession 上传会话
|
// UploadSession 上传会话
|
||||||
|
|
|
@ -19,7 +19,7 @@ func RemoteCallback(c *gin.Context) {
|
||||||
|
|
||||||
// QiniuCallback 七牛上传回调
|
// QiniuCallback 七牛上传回调
|
||||||
func QiniuCallback(c *gin.Context) {
|
func QiniuCallback(c *gin.Context) {
|
||||||
var callbackBody callback.QiniuUploadCallbackService
|
var callbackBody callback.UploadCallbackService
|
||||||
if err := c.ShouldBindJSON(&callbackBody); err == nil {
|
if err := c.ShouldBindJSON(&callbackBody); err == nil {
|
||||||
res := callback.ProcessCallback(callbackBody, c)
|
res := callback.ProcessCallback(callbackBody, c)
|
||||||
if res.Code != 0 {
|
if res.Code != 0 {
|
||||||
|
@ -31,3 +31,14 @@ func QiniuCallback(c *gin.Context) {
|
||||||
c.JSON(401, ErrorResponse(err))
|
c.JSON(401, ErrorResponse(err))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// OSSCallback 阿里云OSS上传回调
|
||||||
|
func OSSCallback(c *gin.Context) {
|
||||||
|
var callbackBody callback.UploadCallbackService
|
||||||
|
if err := c.ShouldBindJSON(&callbackBody); err == nil {
|
||||||
|
res := callback.ProcessCallback(callbackBody, c)
|
||||||
|
c.JSON(200, res)
|
||||||
|
} else {
|
||||||
|
c.JSON(200, ErrorResponse(err))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -138,6 +138,12 @@ func InitMasterRouter() *gin.Engine {
|
||||||
middleware.QiniuCallbackAuth(),
|
middleware.QiniuCallbackAuth(),
|
||||||
controllers.QiniuCallback,
|
controllers.QiniuCallback,
|
||||||
)
|
)
|
||||||
|
// 阿里云OSS策略上传回调
|
||||||
|
callback.POST(
|
||||||
|
"oss/:key",
|
||||||
|
middleware.OSSCallbackAuth(),
|
||||||
|
controllers.OSSCallback,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
// 需要登录保护的
|
// 需要登录保护的
|
||||||
|
|
|
@ -25,8 +25,8 @@ func (service RemoteUploadCallbackService) GetBody() serializer.UploadCallback {
|
||||||
return service.Data
|
return service.Data
|
||||||
}
|
}
|
||||||
|
|
||||||
// QiniuUploadCallbackService 七牛存储上传回调请求服务
|
// UploadCallbackService 云存储上传回调请求服务
|
||||||
type QiniuUploadCallbackService struct {
|
type UploadCallbackService struct {
|
||||||
Name string `json:"name"`
|
Name string `json:"name"`
|
||||||
SourceName string `json:"source_name"`
|
SourceName string `json:"source_name"`
|
||||||
PicInfo string `json:"pic_info"`
|
PicInfo string `json:"pic_info"`
|
||||||
|
@ -34,7 +34,7 @@ type QiniuUploadCallbackService struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetBody 返回回调正文
|
// GetBody 返回回调正文
|
||||||
func (service QiniuUploadCallbackService) GetBody() serializer.UploadCallback {
|
func (service UploadCallbackService) GetBody() serializer.UploadCallback {
|
||||||
return serializer.UploadCallback{
|
return serializer.UploadCallback{
|
||||||
Name: service.Name,
|
Name: service.Name,
|
||||||
SourceName: service.SourceName,
|
SourceName: service.SourceName,
|
||||||
|
|
|
@ -11,7 +11,7 @@ import (
|
||||||
// UploadCredentialService 获取上传凭证服务
|
// UploadCredentialService 获取上传凭证服务
|
||||||
type UploadCredentialService struct {
|
type UploadCredentialService struct {
|
||||||
Path string `form:"path" binding:"required"`
|
Path string `form:"path" binding:"required"`
|
||||||
Size uint64 `form:"size" binding:"required,min=0"`
|
Size uint64 `form:"size" binding:"min=0"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get 获取新的上传凭证
|
// Get 获取新的上传凭证
|
||||||
|
|
Loading…
Reference in New Issue