update download retry && bucket domain
This commit is contained in:
@@ -6,12 +6,16 @@ import (
|
|||||||
"net/http"
|
"net/http"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type BucketDomainRule struct {
|
||||||
|
Status string `xml:"Status,omitempty"`
|
||||||
|
Name string `xml:"Name,omitempty"`
|
||||||
|
Type string `xml:"Type,omitempty"`
|
||||||
|
ForcedReplacement string `xml:"ForcedReplacement,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
type BucketPutDomainOptions struct {
|
type BucketPutDomainOptions struct {
|
||||||
XMLName xml.Name `xml:"DomainConfiguration"`
|
XMLName xml.Name `xml:"DomainConfiguration"`
|
||||||
Status string `xml:"DomainRule>Status"`
|
Rules []BucketDomainRule `xml:"DomainRule,omitempty"`
|
||||||
Name string `xml:"DomainRule>Name"`
|
|
||||||
Type string `xml:"DomainRule>Type"`
|
|
||||||
ForcedReplacement string `xml:"DomainRule>ForcedReplacement,omitempty"`
|
|
||||||
}
|
}
|
||||||
type BucketGetDomainResult BucketPutDomainOptions
|
type BucketGetDomainResult BucketPutDomainOptions
|
||||||
|
|
||||||
@@ -37,3 +41,13 @@ func (s *BucketService) GetDomain(ctx context.Context) (*BucketGetDomainResult,
|
|||||||
resp, err := s.client.doRetry(ctx, sendOpt)
|
resp, err := s.client.doRetry(ctx, sendOpt)
|
||||||
return &res, resp, err
|
return &res, resp, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *BucketService) DeleteDomain(ctx context.Context) (*Response, error) {
|
||||||
|
sendOpt := &sendOptions{
|
||||||
|
baseURL: s.client.BaseURL.BucketURL,
|
||||||
|
uri: "/?domain",
|
||||||
|
method: http.MethodDelete,
|
||||||
|
}
|
||||||
|
resp, err := s.client.doRetry(ctx, sendOpt)
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
|
|||||||
@@ -41,11 +41,15 @@ func TestBucketService_GetDomain(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
want := &BucketGetDomainResult{
|
want := &BucketGetDomainResult{
|
||||||
XMLName: xml.Name{Local: "DomainConfiguration"},
|
XMLName: xml.Name{Local: "DomainConfiguration"},
|
||||||
Status: "ENABLED",
|
Rules: []BucketDomainRule{
|
||||||
Name: "www.abc.com",
|
{
|
||||||
Type: "REST",
|
Status: "ENABLED",
|
||||||
ForcedReplacement: "CNAME",
|
Name: "www.abc.com",
|
||||||
|
Type: "REST",
|
||||||
|
ForcedReplacement: "CNAME",
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
if !reflect.DeepEqual(res, want) {
|
if !reflect.DeepEqual(res, want) {
|
||||||
@@ -58,11 +62,15 @@ func TestBucketService_PutDomain(t *testing.T) {
|
|||||||
defer teardown()
|
defer teardown()
|
||||||
|
|
||||||
opt := &BucketPutDomainOptions{
|
opt := &BucketPutDomainOptions{
|
||||||
XMLName: xml.Name{Local: "DomainConfiguration"},
|
XMLName: xml.Name{Local: "DomainConfiguration"},
|
||||||
Status: "ENABLED",
|
Rules: []BucketDomainRule{
|
||||||
Name: "www.abc.com",
|
{
|
||||||
Type: "REST",
|
Status: "ENABLED",
|
||||||
ForcedReplacement: "CNAME",
|
Name: "www.abc.com",
|
||||||
|
Type: "REST",
|
||||||
|
ForcedReplacement: "CNAME",
|
||||||
|
},
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
rt := 0
|
rt := 0
|
||||||
@@ -95,35 +103,18 @@ func TestBucketService_DeleteDomain(t *testing.T) {
|
|||||||
setup()
|
setup()
|
||||||
defer teardown()
|
defer teardown()
|
||||||
|
|
||||||
opt := &BucketPutDomainOptions{}
|
|
||||||
|
|
||||||
rt := 0
|
|
||||||
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
testMethod(t, r, http.MethodPut)
|
testMethod(t, r, http.MethodDelete)
|
||||||
vs := values{
|
vs := values{
|
||||||
"domain": "",
|
"domain": "",
|
||||||
}
|
}
|
||||||
testFormValues(t, r, vs)
|
testFormValues(t, r, vs)
|
||||||
|
|
||||||
rt++
|
|
||||||
if rt < 3 {
|
|
||||||
w.WriteHeader(http.StatusGatewayTimeout)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
body := new(BucketPutDomainOptions)
|
|
||||||
xml.NewDecoder(r.Body).Decode(body)
|
|
||||||
want := opt
|
|
||||||
want.XMLName = xml.Name{Local: "DomainConfiguration"}
|
|
||||||
if !reflect.DeepEqual(body, want) {
|
|
||||||
t.Errorf("Bucket.PutDomain request\n body: %+v\n, want %+v\n", body, want)
|
|
||||||
}
|
|
||||||
|
|
||||||
w.WriteHeader(http.StatusNoContent)
|
w.WriteHeader(http.StatusNoContent)
|
||||||
})
|
})
|
||||||
|
|
||||||
_, err := client.Bucket.PutDomain(context.Background(), opt)
|
_, err := client.Bucket.DeleteDomain(context.Background())
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Bucket.PutDomain returned error: %v", err)
|
t.Fatalf("Bucket.DeleteDomain returned error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
71
example/bucket/domain.go
Normal file
71
example/bucket/domain.go
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
"github.com/tencentyun/cos-go-sdk-v5"
|
||||||
|
"github.com/tencentyun/cos-go-sdk-v5/debug"
|
||||||
|
)
|
||||||
|
|
||||||
|
func log_status(err error) {
|
||||||
|
if err == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if cos.IsNotFoundError(err) {
|
||||||
|
// WARN
|
||||||
|
fmt.Println("WARN: Resource is not existed")
|
||||||
|
} else if e, ok := cos.IsCOSError(err); ok {
|
||||||
|
fmt.Printf("ERROR: Code: %v\n", e.Code)
|
||||||
|
fmt.Printf("ERROR: Message: %v\n", e.Message)
|
||||||
|
fmt.Printf("ERROR: Resource: %v\n", e.Resource)
|
||||||
|
fmt.Printf("ERROR: RequestId: %v\n", e.RequestID)
|
||||||
|
// ERROR
|
||||||
|
} else {
|
||||||
|
fmt.Printf("ERROR: %v\n", err)
|
||||||
|
// ERROR
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
u, _ := url.Parse("https://test-1259654469.cos.ap-guangzhou.myqcloud.com")
|
||||||
|
b := &cos.BaseURL{
|
||||||
|
BucketURL: u,
|
||||||
|
}
|
||||||
|
c := cos.NewClient(b, &http.Client{
|
||||||
|
Transport: &cos.AuthorizationTransport{
|
||||||
|
SecretID: os.Getenv("COS_SECRETID"),
|
||||||
|
SecretKey: os.Getenv("COS_SECRETKEY"),
|
||||||
|
Transport: &debug.DebugRequestTransport{
|
||||||
|
RequestHeader: true,
|
||||||
|
RequestBody: true,
|
||||||
|
ResponseHeader: true,
|
||||||
|
ResponseBody: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
opt := &cos.BucketPutDomainOptions{
|
||||||
|
Rules: []cos.BucketDomainRule{
|
||||||
|
{
|
||||||
|
Status: "ENABLED",
|
||||||
|
Name: "www.qq.com",
|
||||||
|
Type: "REST",
|
||||||
|
ForcedReplacement: "CNAME",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
_, err := c.Bucket.PutDomain(context.Background(), opt)
|
||||||
|
log_status(err)
|
||||||
|
|
||||||
|
res, _, err := c.Bucket.GetDomain(context.Background())
|
||||||
|
log_status(err)
|
||||||
|
fmt.Printf("%+v\n", res)
|
||||||
|
|
||||||
|
_, err = c.Bucket.DeleteDomain(context.Background())
|
||||||
|
log_status(err)
|
||||||
|
}
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/tencentyun/cos-go-sdk-v5"
|
|
||||||
"github.com/tencentyun/cos-go-sdk-v5/debug"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
u, _ := url.Parse("https://test-1259654469.cos.ap-guangzhou.myqcloud.com")
|
|
||||||
b := &cos.BaseURL{
|
|
||||||
BucketURL: u,
|
|
||||||
}
|
|
||||||
c := cos.NewClient(b, &http.Client{
|
|
||||||
Transport: &cos.AuthorizationTransport{
|
|
||||||
SecretID: os.Getenv("COS_SECRETID"),
|
|
||||||
SecretKey: os.Getenv("COS_SECRETKEY"),
|
|
||||||
Transport: &debug.DebugRequestTransport{
|
|
||||||
RequestHeader: true,
|
|
||||||
RequestBody: true,
|
|
||||||
ResponseHeader: true,
|
|
||||||
ResponseBody: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
_, _, err := c.Bucket.GetDomain(context.Background())
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,42 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"net/http"
|
|
||||||
"net/url"
|
|
||||||
"os"
|
|
||||||
|
|
||||||
"github.com/tencentyun/cos-go-sdk-v5"
|
|
||||||
"github.com/tencentyun/cos-go-sdk-v5/debug"
|
|
||||||
)
|
|
||||||
|
|
||||||
func main() {
|
|
||||||
u, _ := url.Parse("https://test-1259654469.cos.ap-guangzhou.myqcloud.com")
|
|
||||||
b := &cos.BaseURL{
|
|
||||||
BucketURL: u,
|
|
||||||
}
|
|
||||||
c := cos.NewClient(b, &http.Client{
|
|
||||||
Transport: &cos.AuthorizationTransport{
|
|
||||||
SecretID: os.Getenv("COS_SECRETID"),
|
|
||||||
SecretKey: os.Getenv("COS_SECRETKEY"),
|
|
||||||
Transport: &debug.DebugRequestTransport{
|
|
||||||
RequestHeader: true,
|
|
||||||
RequestBody: true,
|
|
||||||
ResponseHeader: true,
|
|
||||||
ResponseBody: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
})
|
|
||||||
|
|
||||||
opt := &cos.BucketPutDomainOptions{
|
|
||||||
Status: "ENABLED",
|
|
||||||
Name: "www.abc.com",
|
|
||||||
Type: "REST",
|
|
||||||
ForcedReplacement: "CNAME",
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err := c.Bucket.PutDomain(context.Background(), opt)
|
|
||||||
if err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
27
object.go
27
object.go
@@ -757,16 +757,12 @@ func downloadWorker(ctx context.Context, s *ObjectService, jobs <-chan *Jobs, re
|
|||||||
res.err = err
|
res.err = err
|
||||||
res.Resp = resp
|
res.Resp = resp
|
||||||
if err != nil {
|
if err != nil {
|
||||||
rt--
|
results <- &res
|
||||||
if rt == 0 {
|
break
|
||||||
results <- &res
|
|
||||||
break
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
defer resp.Body.Close()
|
|
||||||
fd, err := os.OpenFile(j.FilePath, os.O_WRONLY, 0660)
|
fd, err := os.OpenFile(j.FilePath, os.O_WRONLY, 0660)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
resp.Body.Close()
|
||||||
res.err = err
|
res.err = err
|
||||||
results <- &res
|
results <- &res
|
||||||
break
|
break
|
||||||
@@ -774,9 +770,18 @@ func downloadWorker(ctx context.Context, s *ObjectService, jobs <-chan *Jobs, re
|
|||||||
fd.Seek(j.Chunk.OffSet, os.SEEK_SET)
|
fd.Seek(j.Chunk.OffSet, os.SEEK_SET)
|
||||||
n, err := io.Copy(fd, LimitReadCloser(resp.Body, j.Chunk.Size))
|
n, err := io.Copy(fd, LimitReadCloser(resp.Body, j.Chunk.Size))
|
||||||
if n != j.Chunk.Size || err != nil {
|
if n != j.Chunk.Size || err != nil {
|
||||||
res.err = fmt.Errorf("io.Copy Failed, nread:%v, want:%v, err:%v", n, j.Chunk.Size, err)
|
fd.Close()
|
||||||
|
resp.Body.Close()
|
||||||
|
rt--
|
||||||
|
if rt == 0 {
|
||||||
|
res.err = fmt.Errorf("io.Copy Failed, nread:%v, want:%v, err:%v", n, j.Chunk.Size, err)
|
||||||
|
results <- &res
|
||||||
|
break
|
||||||
|
}
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
fd.Close()
|
fd.Close()
|
||||||
|
resp.Body.Close()
|
||||||
results <- &res
|
results <- &res
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@@ -818,7 +823,7 @@ func SplitFileIntoChunks(filePath string, partSize int64) (int64, []Chunk, int,
|
|||||||
return 0, nil, 0, errors.New("Too many parts, out of 10000")
|
return 0, nil, 0, errors.New("Too many parts, out of 10000")
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
partNum, partSize = DividePart(stat.Size(), 64)
|
partNum, partSize = DividePart(stat.Size(), 16)
|
||||||
}
|
}
|
||||||
|
|
||||||
var chunks []Chunk
|
var chunks []Chunk
|
||||||
@@ -1119,7 +1124,7 @@ func SplitSizeIntoChunks(totalBytes int64, partSize int64) ([]Chunk, int, error)
|
|||||||
return nil, 0, errors.New("Too manry parts, out of 10000")
|
return nil, 0, errors.New("Too manry parts, out of 10000")
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
partNum, partSize = DividePart(totalBytes, 64)
|
partNum, partSize = DividePart(totalBytes, 16)
|
||||||
}
|
}
|
||||||
|
|
||||||
var chunks []Chunk
|
var chunks []Chunk
|
||||||
@@ -1293,7 +1298,7 @@ func (s *ObjectService) Download(ctx context.Context, name string, filepath stri
|
|||||||
}
|
}
|
||||||
job := &Jobs{
|
job := &Jobs{
|
||||||
Name: name,
|
Name: name,
|
||||||
RetryTimes: 1,
|
RetryTimes: 3,
|
||||||
FilePath: filepath,
|
FilePath: filepath,
|
||||||
Chunk: chunk,
|
Chunk: chunk,
|
||||||
DownOpt: &downOpt,
|
DownOpt: &downOpt,
|
||||||
|
|||||||
@@ -796,7 +796,7 @@ func TestObjectService_Download(t *testing.T) {
|
|||||||
}
|
}
|
||||||
defer os.Remove(filePath)
|
defer os.Remove(filePath)
|
||||||
// 源文件内容
|
// 源文件内容
|
||||||
totalBytes := int64(1024*1024*9 + 123)
|
totalBytes := int64(1024*1024 + 1230) //1024*1024*9 + 123)
|
||||||
b := make([]byte, totalBytes)
|
b := make([]byte, totalBytes)
|
||||||
_, err = rand.Read(b)
|
_, err = rand.Read(b)
|
||||||
newfile.Write(b)
|
newfile.Write(b)
|
||||||
@@ -817,20 +817,21 @@ func TestObjectService_Download(t *testing.T) {
|
|||||||
start, _ := strconv.ParseInt(slice2[0], 10, 64)
|
start, _ := strconv.ParseInt(slice2[0], 10, 64)
|
||||||
end, _ := strconv.ParseInt(slice2[1], 10, 64)
|
end, _ := strconv.ParseInt(slice2[1], 10, 64)
|
||||||
if retryMap[start] == 0 {
|
if retryMap[start] == 0 {
|
||||||
// 重试校验1
|
// SDK 内部重试
|
||||||
retryMap[start]++
|
retryMap[start]++
|
||||||
w.WriteHeader(http.StatusGatewayTimeout)
|
w.WriteHeader(http.StatusGatewayTimeout)
|
||||||
} else if retryMap[start] == 1 {
|
} else if retryMap[start] == 1 {
|
||||||
// 重试检验2
|
// SDK Download 做重试
|
||||||
retryMap[start]++
|
retryMap[start]++
|
||||||
io.Copy(w, bytes.NewBuffer(b[start:end]))
|
io.Copy(w, bytes.NewBuffer(b[start:end]))
|
||||||
} else if retryMap[start] == 2 {
|
} else if retryMap[start] == 2 {
|
||||||
// 重试检验3
|
// SDK Download 做重试
|
||||||
retryMap[start]++
|
retryMap[start]++
|
||||||
st := math_rand.Int63n(totalBytes - 1024*1024)
|
st := start
|
||||||
et := st + end - start
|
et := st + math_rand.Int63n(1024)
|
||||||
io.Copy(w, bytes.NewBuffer(b[st:et+1]))
|
io.Copy(w, bytes.NewBuffer(b[st:et+1]))
|
||||||
} else {
|
} else {
|
||||||
|
// SDK Download 成功
|
||||||
io.Copy(w, bytes.NewBuffer(b[start:end+1]))
|
io.Copy(w, bytes.NewBuffer(b[start:end+1]))
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@@ -842,18 +843,15 @@ func TestObjectService_Download(t *testing.T) {
|
|||||||
downPath := "down.file" + time.Now().Format(time.RFC3339)
|
downPath := "down.file" + time.Now().Format(time.RFC3339)
|
||||||
defer os.Remove(downPath)
|
defer os.Remove(downPath)
|
||||||
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
||||||
if err == nil {
|
if err != nil {
|
||||||
// 长度不一致 Failed
|
t.Fatalf("Object.Upload returned error: %v", err)
|
||||||
t.Fatalf("Object.Upload returned error: %v", err)
|
}
|
||||||
}
|
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
||||||
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
if err != nil {
|
||||||
if err == nil {
|
|
||||||
// CRC不一致
|
|
||||||
t.Fatalf("Object.Upload returned error: %v", err)
|
t.Fatalf("Object.Upload returned error: %v", err)
|
||||||
}
|
}
|
||||||
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
_, err = client.Object.Download(context.Background(), "test.go.download", downPath, opt)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// 正确
|
|
||||||
t.Fatalf("Object.Upload returned error: %v", err)
|
t.Fatalf("Object.Upload returned error: %v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -895,7 +893,7 @@ func TestObjectService_DownloadWithCheckPoint(t *testing.T) {
|
|||||||
if oddok {
|
if oddok {
|
||||||
io.Copy(w, bytes.NewBuffer(b[start:end+1]))
|
io.Copy(w, bytes.NewBuffer(b[start:end+1]))
|
||||||
} else {
|
} else {
|
||||||
// 数据校验失败, Download不会做重试
|
// 数据校验失败, Download做3次重试
|
||||||
io.Copy(w, bytes.NewBuffer(b[start:end]))
|
io.Copy(w, bytes.NewBuffer(b[start:end]))
|
||||||
}
|
}
|
||||||
oddcount++
|
oddcount++
|
||||||
@@ -934,7 +932,7 @@ func TestObjectService_DownloadWithCheckPoint(t *testing.T) {
|
|||||||
}
|
}
|
||||||
fd.Close()
|
fd.Close()
|
||||||
|
|
||||||
if oddcount != 5 || evencount != 5 {
|
if oddcount != 15 || evencount != 5 {
|
||||||
t.Fatalf("Object.Download failed, odd:%v, even:%v", oddcount, evencount)
|
t.Fatalf("Object.Download failed, odd:%v, even:%v", oddcount, evencount)
|
||||||
}
|
}
|
||||||
// 设置奇数块OK
|
// 设置奇数块OK
|
||||||
@@ -944,7 +942,7 @@ func TestObjectService_DownloadWithCheckPoint(t *testing.T) {
|
|||||||
// 下载成功
|
// 下载成功
|
||||||
t.Fatalf("Object.Download returned error: %v", err)
|
t.Fatalf("Object.Download returned error: %v", err)
|
||||||
}
|
}
|
||||||
if oddcount != 10 || evencount != 5 {
|
if oddcount != 20 || evencount != 5 {
|
||||||
t.Fatalf("Object.Download failed, odd:%v, even:%v", oddcount, evencount)
|
t.Fatalf("Object.Download failed, odd:%v, even:%v", oddcount, evencount)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user