Skip to content

Commit 6a2da87

Browse files
authored
Merge pull request #98 from agin719/cos-v4-dev
Cos v4 dev
2 parents 5057561 + c07e497 commit 6a2da87

File tree

10 files changed

+262
-28
lines changed

10 files changed

+262
-28
lines changed

bucket_accelerate.go

Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
package cos
2+
3+
import (
4+
"context"
5+
"encoding/xml"
6+
"net/http"
7+
)
8+
9+
type BucketPutAccelerateOptions struct {
10+
XMLName xml.Name `xml:"AccelerateConfiguration"`
11+
Status string `xml:"Status,omitempty"`
12+
Type string `xml:"Type,omitempty"`
13+
}
14+
type BucketGetAccelerateResult BucketPutAccelerateOptions
15+
16+
func (s *BucketService) PutAccelerate(ctx context.Context, opt *BucketPutAccelerateOptions) (*Response, error) {
17+
sendOpt := &sendOptions{
18+
baseURL: s.client.BaseURL.BucketURL,
19+
uri: "/?accelerate",
20+
method: http.MethodPut,
21+
body: opt,
22+
}
23+
resp, err := s.client.send(ctx, sendOpt)
24+
return resp, err
25+
}
26+
27+
func (s *BucketService) GetAccelerate(ctx context.Context) (*BucketGetAccelerateResult, *Response, error) {
28+
var res BucketGetAccelerateResult
29+
sendOpt := &sendOptions{
30+
baseURL: s.client.BaseURL.BucketURL,
31+
uri: "/?accelerate",
32+
method: http.MethodGet,
33+
result: &res,
34+
}
35+
resp, err := s.client.send(ctx, sendOpt)
36+
return &res, resp, err
37+
}

bucket_accelerate_test.go

Lines changed: 74 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
package cos
2+
3+
import (
4+
"context"
5+
"encoding/xml"
6+
"fmt"
7+
"net/http"
8+
"reflect"
9+
"testing"
10+
)
11+
12+
func TestBucketService_GetAccelerate(t *testing.T) {
13+
setup()
14+
defer teardown()
15+
16+
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
17+
testMethod(t, r, "GET")
18+
vs := values{
19+
"accelerate": "",
20+
}
21+
testFormValues(t, r, vs)
22+
fmt.Fprint(w, `<AccelerateConfiguration>
23+
<Status>Enabled</Status>
24+
<Type>COS</Type>
25+
</AccelerateConfiguration>`)
26+
})
27+
28+
res, _, err := client.Bucket.GetAccelerate(context.Background())
29+
if err != nil {
30+
t.Fatalf("Bucket.GetAccelerate returned error %v", err)
31+
}
32+
33+
want := &BucketGetAccelerateResult{
34+
XMLName: xml.Name{Local: "AccelerateConfiguration"},
35+
Status: "Enabled",
36+
Type: "COS",
37+
}
38+
39+
if !reflect.DeepEqual(res, want) {
40+
t.Errorf("Bucket.GetAccelerate returned %+v, want %+v", res, want)
41+
}
42+
}
43+
44+
func TestBucketService_PutAccelerate(t *testing.T) {
45+
setup()
46+
defer teardown()
47+
48+
opt := &BucketPutAccelerateOptions{
49+
XMLName: xml.Name{Local: "AccelerateConfiguration"},
50+
Status: "Enabled",
51+
Type: "COS",
52+
}
53+
54+
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
55+
testMethod(t, r, "PUT")
56+
vs := values{
57+
"accelerate": "",
58+
}
59+
testFormValues(t, r, vs)
60+
61+
body := new(BucketPutAccelerateOptions)
62+
xml.NewDecoder(r.Body).Decode(body)
63+
want := opt
64+
want.XMLName = xml.Name{Local: "AccelerateConfiguration"}
65+
if !reflect.DeepEqual(body, want) {
66+
t.Errorf("Bucket.PutAccelerate request\n body: %+v\n, want %+v\n", body, want)
67+
}
68+
})
69+
70+
_, err := client.Bucket.PutAccelerate(context.Background(), opt)
71+
if err != nil {
72+
t.Fatalf("Bucket.PutAccelerate returned error: %v", err)
73+
}
74+
}

cos.go

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ import (
2222

2323
const (
2424
// Version current go sdk version
25-
Version = "0.7.15"
25+
Version = "0.7.16"
2626
userAgent = "cos-go-sdk-v5/" + Version
2727
contentTypeXML = "application/xml"
2828
defaultServiceBaseURL = "http://service.cos.myqcloud.com"

costesting/ci_test.go

Lines changed: 25 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -882,6 +882,31 @@ func (s *CosTestSuite) TestReferer() {
882882
assert.Equal(s.T(), opt.EmptyReferConfiguration, res.EmptyReferConfiguration, "GetReferer Failed")
883883
}
884884

885+
func (s *CosTestSuite) TestAccelerate() {
886+
opt := &cos.BucketPutAccelerateOptions{
887+
Status: "Enabled",
888+
Type: "COS",
889+
}
890+
_, err := s.Client.Bucket.PutAccelerate(context.Background(), opt)
891+
assert.Nil(s.T(), err, "PutAccelerate Failed")
892+
893+
time.Sleep(time.Second)
894+
res, _, err := s.Client.Bucket.GetAccelerate(context.Background())
895+
assert.Nil(s.T(), err, "GetAccelerate Failed")
896+
assert.Equal(s.T(), opt.Status, res.Status, "GetAccelerate Failed")
897+
assert.Equal(s.T(), opt.Type, res.Type, "GetAccelerate Failed")
898+
899+
opt.Status = "Suspended"
900+
_, err = s.Client.Bucket.PutAccelerate(context.Background(), opt)
901+
assert.Nil(s.T(), err, "PutAccelerate Failed")
902+
903+
time.Sleep(time.Second)
904+
res, _, err = s.Client.Bucket.GetAccelerate(context.Background())
905+
assert.Nil(s.T(), err, "GetAccelerate Failed")
906+
assert.Equal(s.T(), opt.Status, res.Status, "GetAccelerate Failed")
907+
assert.Equal(s.T(), opt.Type, res.Type, "GetAccelerate Failed")
908+
}
909+
885910
// End of api test
886911

887912
// All methods that begin with "Test" are run as tests within a

example/bucket/accelerate.go

Lines changed: 72 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
package main
2+
3+
import (
4+
"context"
5+
"fmt"
6+
"net/http"
7+
"net/url"
8+
"os"
9+
10+
"github.com/tencentyun/cos-go-sdk-v5"
11+
"github.com/tencentyun/cos-go-sdk-v5/debug"
12+
)
13+
14+
func log_status(err error) {
15+
if err == nil {
16+
return
17+
}
18+
if cos.IsNotFoundError(err) {
19+
// WARN
20+
fmt.Println("Resource is not existed")
21+
} else if e, ok := cos.IsCOSError(err); ok {
22+
fmt.Printf("Code: %v\n", e.Code)
23+
fmt.Printf("Message: %v\n", e.Message)
24+
fmt.Printf("Resource: %v\n", e.Resource)
25+
fmt.Printf("RequestId: %v\n", e.RequestID)
26+
// ERROR
27+
} else {
28+
fmt.Println(err)
29+
// ERROR
30+
}
31+
}
32+
33+
func main() {
34+
u, _ := url.Parse("https://test-1259654469.cos.ap-guangzhou.myqcloud.com")
35+
b := &cos.BaseURL{
36+
BucketURL: u,
37+
}
38+
c := cos.NewClient(b, &http.Client{
39+
Transport: &cos.AuthorizationTransport{
40+
SecretID: os.Getenv("COS_SECRETID"),
41+
SecretKey: os.Getenv("COS_SECRETKEY"),
42+
Transport: &debug.DebugRequestTransport{
43+
RequestHeader: true,
44+
RequestBody: true,
45+
ResponseHeader: true,
46+
ResponseBody: true,
47+
},
48+
},
49+
})
50+
res, _, err := c.Bucket.GetAccelerate(context.Background())
51+
log_status(err)
52+
fmt.Printf("%+v\n", res)
53+
54+
opt := &cos.BucketPutAccelerateOptions{
55+
Status: "Enabled",
56+
Type: "COS",
57+
}
58+
_, err = c.Bucket.PutAccelerate(context.Background(), opt)
59+
log_status(err)
60+
61+
res, _, err = c.Bucket.GetAccelerate(context.Background())
62+
log_status(err)
63+
fmt.Printf("%+v\n", res)
64+
65+
opt.Status = "Suspended"
66+
_, err = c.Bucket.PutAccelerate(context.Background(), opt)
67+
log_status(err)
68+
69+
res, _, err = c.Bucket.GetAccelerate(context.Background())
70+
log_status(err)
71+
fmt.Printf("%+v\n", res)
72+
}

example/object/uploadPart.go

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -79,10 +79,10 @@ func main() {
7979
resp, err := c.Object.UploadPart(
8080
context.Background(), name, uploadID, 1, fd, opt,
8181
)
82+
log_status(err)
8283
optcom.Parts = append(optcom.Parts, cos.Object{
8384
PartNumber: 1, ETag: resp.Header.Get("ETag"),
8485
})
85-
log_status(err)
8686

8787
f := strings.NewReader("test heoo")
8888
resp, err = c.Object.UploadPart(

helper.go

Lines changed: 12 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ import (
44
"bytes"
55
"crypto/md5"
66
"crypto/sha1"
7+
"errors"
78
"fmt"
89
"io"
910
"net/http"
@@ -12,6 +13,9 @@ import (
1213
"strings"
1314
)
1415

16+
// 单次上传文件最大为5GB
17+
const singleUploadMaxLength = 5 * 1024 * 1024 * 1024
18+
1519
// 计算 md5 或 sha1 时的分块大小
1620
const calDigestBlockSize = 1024 * 1024 * 10
1721

@@ -140,3 +144,11 @@ func GetReaderLen(reader io.Reader) (length int64, err error) {
140144
}
141145
return
142146
}
147+
148+
func CheckReaderLen(reader io.Reader) error {
149+
nlen, err := GetReaderLen(reader)
150+
if err != nil || nlen < singleUploadMaxLength {
151+
return nil
152+
}
153+
return errors.New("The single object size you upload can not be larger than 5GB")
154+
}

object.go

Lines changed: 28 additions & 22 deletions
Original file line numberDiff line numberDiff line change
@@ -127,7 +127,7 @@ func (s *ObjectService) GetPresignedURL(ctx context.Context, httpMethod, name, a
127127
authTime = NewAuthTime(expired)
128128
}
129129
authorization := newAuthorization(ak, sk, req, authTime)
130-
sign := encodeURIComponent(authorization, []byte{'&','='})
130+
sign := encodeURIComponent(authorization, []byte{'&', '='})
131131

132132
if req.URL.RawQuery == "" {
133133
req.URL.RawQuery = fmt.Sprintf("%s", sign)
@@ -181,6 +181,9 @@ type ObjectPutOptions struct {
181181
//
182182
// https://www.qcloud.com/document/product/436/7749
183183
func (s *ObjectService) Put(ctx context.Context, name string, r io.Reader, opt *ObjectPutOptions) (*Response, error) {
184+
if err := CheckReaderLen(r); err != nil {
185+
return nil, err
186+
}
184187
if opt != nil && opt.Listener != nil {
185188
totalBytes, err := GetReaderLen(r)
186189
if err != nil {
@@ -802,28 +805,30 @@ func (s *ObjectService) Upload(ctx context.Context, name string, filepath string
802805
progressCallback(listener, event)
803806

804807
// 4.Push jobs
805-
for _, chunk := range chunks {
806-
if chunk.Done {
807-
continue
808-
}
809-
partOpt := &ObjectUploadPartOptions{}
810-
if optini != nil && optini.ObjectPutHeaderOptions != nil {
811-
partOpt.XCosSSECustomerAglo = optini.XCosSSECustomerAglo
812-
partOpt.XCosSSECustomerKey = optini.XCosSSECustomerKey
813-
partOpt.XCosSSECustomerKeyMD5 = optini.XCosSSECustomerKeyMD5
814-
partOpt.XCosTrafficLimit = optini.XCosTrafficLimit
815-
}
816-
job := &Jobs{
817-
Name: name,
818-
RetryTimes: 3,
819-
FilePath: filepath,
820-
UploadId: uploadID,
821-
Chunk: chunk,
822-
Opt: partOpt,
808+
go func() {
809+
for _, chunk := range chunks {
810+
if chunk.Done {
811+
continue
812+
}
813+
partOpt := &ObjectUploadPartOptions{}
814+
if optini != nil && optini.ObjectPutHeaderOptions != nil {
815+
partOpt.XCosSSECustomerAglo = optini.XCosSSECustomerAglo
816+
partOpt.XCosSSECustomerKey = optini.XCosSSECustomerKey
817+
partOpt.XCosSSECustomerKeyMD5 = optini.XCosSSECustomerKeyMD5
818+
partOpt.XCosTrafficLimit = optini.XCosTrafficLimit
819+
}
820+
job := &Jobs{
821+
Name: name,
822+
RetryTimes: 3,
823+
FilePath: filepath,
824+
UploadId: uploadID,
825+
Chunk: chunk,
826+
Opt: partOpt,
827+
}
828+
chjobs <- job
823829
}
824-
chjobs <- job
825-
}
826-
close(chjobs)
830+
close(chjobs)
831+
}()
827832

828833
// 5.Recv the resp etag to complete
829834
for i := 0; i < partNum; i++ {
@@ -854,6 +859,7 @@ func (s *ObjectService) Upload(ctx context.Context, name string, filepath string
854859
event = newProgressEvent(ProgressDataEvent, chunks[res.PartNumber-1].Size, consumedBytes, totalBytes)
855860
progressCallback(listener, event)
856861
}
862+
close(chresults)
857863
sort.Sort(ObjectList(optcom.Parts))
858864

859865
event = newProgressEvent(ProgressCompletedEvent, 0, consumedBytes, totalBytes)

object_part.go

Lines changed: 8 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -41,16 +41,17 @@ func (s *ObjectService) InitiateMultipartUpload(ctx context.Context, name string
4141

4242
// ObjectUploadPartOptions is the options of upload-part
4343
type ObjectUploadPartOptions struct {
44-
Expect string `header:"Expect,omitempty" url:"-"`
45-
XCosContentSHA1 string `header:"x-cos-content-sha1,omitempty" url:"-"`
46-
ContentLength int `header:"Content-Length,omitempty" url:"-"`
47-
44+
Expect string `header:"Expect,omitempty" url:"-"`
45+
XCosContentSHA1 string `header:"x-cos-content-sha1,omitempty" url:"-"`
46+
ContentLength int `header:"Content-Length,omitempty" url:"-"`
47+
ContentMD5 string `header:"Content-MD5,omitempty" url:"-"`
4848
XCosSSECustomerAglo string `header:"x-cos-server-side-encryption-customer-algorithm,omitempty" url:"-" xml:"-"`
4949
XCosSSECustomerKey string `header:"x-cos-server-side-encryption-customer-key,omitempty" url:"-" xml:"-"`
5050
XCosSSECustomerKeyMD5 string `header:"x-cos-server-side-encryption-customer-key-MD5,omitempty" url:"-" xml:"-"`
5151

5252
XCosTrafficLimit int `header:"x-cos-traffic-limit,omitempty" url:"-" xml:"-"`
5353

54+
XOptionHeader *http.Header `header:"-,omitempty" url:"-" xml:"-"`
5455
// 上传进度, ProgressCompleteEvent不能表示对应API调用成功,API是否调用成功的判断标准为返回err==nil
5556
Listener ProgressListener `header:"-" url:"-" xml:"-"`
5657
}
@@ -64,6 +65,9 @@ type ObjectUploadPartOptions struct {
6465
//
6566
// https://www.qcloud.com/document/product/436/7750
6667
func (s *ObjectService) UploadPart(ctx context.Context, name, uploadID string, partNumber int, r io.Reader, opt *ObjectUploadPartOptions) (*Response, error) {
68+
if err := CheckReaderLen(r); err != nil {
69+
return nil, err
70+
}
6771
if opt != nil && opt.Listener != nil {
6872
totalBytes, err := GetReaderLen(r)
6973
if err != nil {

progress.go

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -97,6 +97,10 @@ func (r *teeReader) Close() error {
9797
return nil
9898
}
9999

100+
func (r *teeReader) Size() int64 {
101+
return r.totalBytes
102+
}
103+
100104
func TeeReader(reader io.Reader, writer io.Writer, total int64, listener ProgressListener) *teeReader {
101105
return &teeReader{
102106
reader: reader,

0 commit comments

Comments
 (0)