You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

918 lines
30 KiB

5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
6 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
  1. package cos
  2. // Basic imports
  3. import (
  4. "context"
  5. "errors"
  6. "fmt"
  7. "io/ioutil"
  8. "math/rand"
  9. "net/http"
  10. "net/url"
  11. "os"
  12. "strings"
  13. "testing"
  14. "time"
  15. "github.com/google/uuid"
  16. "github.com/stretchr/testify/assert"
  17. "github.com/stretchr/testify/suite"
  18. "github.com/tencentyun/cos-go-sdk-v5"
  19. )
  20. // Define the suite, and absorb the built-in basic suite
  21. // functionality from testify - including a T() method which
  22. // returns the current testing context
  23. type CosTestSuite struct {
  24. suite.Suite
  25. VariableThatShouldStartAtFive int
  26. // CI client
  27. Client *cos.Client
  28. // Copy source client
  29. CClient *cos.Client
  30. Region string
  31. Bucket string
  32. Appid string
  33. // test_object
  34. TestObject string
  35. // special_file_name
  36. SepFileName string
  37. }
  38. // 请替换成您的账号及存储桶信息
  39. const (
  40. //uin
  41. kUin = "100010805041"
  42. kAppid = 1259654469
  43. // 常规测试需要的存储桶
  44. kBucket = "cosgosdktest-1259654469"
  45. kRegion = "ap-guangzhou"
  46. // 跨区域复制需要的目标存储桶,地域不能与kBucket存储桶相同。
  47. kRepBucket = "cosgosdkreptest"
  48. kRepRegion = "ap-chengdu"
  49. // Batch测试需要的源存储桶和目标存储桶,目前只在成都、重庆地域公测
  50. kBatchBucket = "cosgosdktest-1259654469"
  51. kTargetBatchBucket = "cosgosdktest-1259654469" //复用了存储桶
  52. kBatchRegion = "ap-guangzhou"
  53. )
  54. func (s *CosTestSuite) SetupSuite() {
  55. fmt.Println("Set up test")
  56. // init
  57. s.TestObject = "test.txt"
  58. s.SepFileName = "中文" + "→↓←→↖↗↙↘! \"#$%&'()*+,-./0123456789:;<=>@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~"
  59. // CI client for test interface
  60. // URL like this http://test-1253846586.cos.ap-guangzhou.myqcloud.com
  61. u := "https://" + kBucket + ".cos." + kRegion + ".myqcloud.com"
  62. u2 := "https://" + kUin + ".cos-control." + kBatchRegion + ".myqcloud.com"
  63. // Get the region
  64. bucketurl, _ := url.Parse(u)
  65. batchurl, _ := url.Parse(u2)
  66. p := strings.Split(bucketurl.Host, ".")
  67. assert.Equal(s.T(), 5, len(p), "Bucket host is not right")
  68. s.Region = p[2]
  69. // Bucket name
  70. pp := strings.Split(p[0], "-")
  71. s.Bucket = pp[0]
  72. s.Appid = pp[1]
  73. ib := &cos.BaseURL{BucketURL: bucketurl, BatchURL: batchurl}
  74. s.Client = cos.NewClient(ib, &http.Client{
  75. Transport: &cos.AuthorizationTransport{
  76. SecretID: os.Getenv("COS_SECRETID"),
  77. SecretKey: os.Getenv("COS_SECRETKEY"),
  78. },
  79. })
  80. opt := &cos.BucketPutOptions{
  81. XCosACL: "public-read",
  82. }
  83. r, err := s.Client.Bucket.Put(context.Background(), opt)
  84. if err != nil && r.StatusCode == 409 {
  85. fmt.Println("BucketAlreadyOwnedByYou")
  86. } else if err != nil {
  87. assert.Nil(s.T(), err, "PutBucket Failed")
  88. }
  89. }
  90. // Begin of api test
  91. // Service API
  92. func (s *CosTestSuite) TestGetService() {
  93. _, _, err := s.Client.Service.Get(context.Background())
  94. assert.Nil(s.T(), err, "GetService Failed")
  95. }
  96. // Bucket API
  97. func (s *CosTestSuite) TestPutHeadDeleteBucket() {
  98. // Notic sometimes the bucket host can not analyis, may has i/o timeout problem
  99. u := "http://" + "testgosdkbucket-create-head-del-" + s.Appid + ".cos." + kRegion + ".myqcloud.com"
  100. iu, _ := url.Parse(u)
  101. ib := &cos.BaseURL{BucketURL: iu}
  102. client := cos.NewClient(ib, &http.Client{
  103. Transport: &cos.AuthorizationTransport{
  104. SecretID: os.Getenv("COS_SECRETID"),
  105. SecretKey: os.Getenv("COS_SECRETKEY"),
  106. },
  107. })
  108. r, err := client.Bucket.Put(context.Background(), nil)
  109. if err != nil && r.StatusCode == 409 {
  110. fmt.Println("BucketAlreadyOwnedByYou")
  111. } else if err != nil {
  112. assert.Nil(s.T(), err, "PutBucket Failed")
  113. }
  114. if err != nil {
  115. panic(err)
  116. }
  117. time.Sleep(3 * time.Second)
  118. _, err = client.Bucket.Head(context.Background())
  119. assert.Nil(s.T(), err, "HeadBucket Failed")
  120. if err == nil {
  121. _, err = client.Bucket.Delete(context.Background())
  122. assert.Nil(s.T(), err, "DeleteBucket Failed")
  123. }
  124. }
  125. func (s *CosTestSuite) TestPutBucketACLIllegal() {
  126. opt := &cos.BucketPutACLOptions{
  127. Header: &cos.ACLHeaderOptions{
  128. XCosACL: "public-read-writ",
  129. },
  130. }
  131. _, err := s.Client.Bucket.PutACL(context.Background(), opt)
  132. assert.NotNil(s.T(), err, "PutBucketACL illegal Failed")
  133. }
  134. func (s *CosTestSuite) TestPutGetBucketACLNormal() {
  135. // with header
  136. opt := &cos.BucketPutACLOptions{
  137. Header: &cos.ACLHeaderOptions{
  138. XCosACL: "private",
  139. },
  140. }
  141. _, err := s.Client.Bucket.PutACL(context.Background(), opt)
  142. assert.Nil(s.T(), err, "PutBucketACL normal Failed")
  143. v, _, err := s.Client.Bucket.GetACL(context.Background())
  144. assert.Nil(s.T(), err, "GetBucketACL normal Failed")
  145. assert.Equal(s.T(), 1, len(v.AccessControlList), "GetBucketACL normal Failed, must be private")
  146. }
  147. func (s *CosTestSuite) TestGetBucket() {
  148. opt := &cos.BucketGetOptions{
  149. Prefix: "中文",
  150. MaxKeys: 3,
  151. }
  152. _, _, err := s.Client.Bucket.Get(context.Background(), opt)
  153. assert.Nil(s.T(), err, "GetBucket Failed")
  154. }
  155. func (s *CosTestSuite) TestGetObjectVersions() {
  156. opt := &cos.BucketGetObjectVersionsOptions {
  157. Prefix: "中文",
  158. MaxKeys: 3,
  159. }
  160. _, _, err := s.Client.Bucket.GetObjectVersions(context.Background(), opt)
  161. assert.Nil(s.T(), err, "GetObjectVersions Failed")
  162. }
  163. func (s *CosTestSuite) TestGetBucketLocation() {
  164. v, _, err := s.Client.Bucket.GetLocation(context.Background())
  165. assert.Nil(s.T(), err, "GetLocation Failed")
  166. assert.Equal(s.T(), s.Region, v.Location, "GetLocation wrong region")
  167. }
  168. func (s *CosTestSuite) TestPutGetDeleteCORS() {
  169. opt := &cos.BucketPutCORSOptions{
  170. Rules: []cos.BucketCORSRule{
  171. {
  172. AllowedOrigins: []string{"http://www.qq.com"},
  173. AllowedMethods: []string{"PUT", "GET"},
  174. AllowedHeaders: []string{"x-cos-meta-test", "x-cos-xx"},
  175. MaxAgeSeconds: 500,
  176. ExposeHeaders: []string{"x-cos-meta-test1"},
  177. },
  178. },
  179. }
  180. _, err := s.Client.Bucket.PutCORS(context.Background(), opt)
  181. assert.Nil(s.T(), err, "PutBucketCORS Failed")
  182. v, _, err := s.Client.Bucket.GetCORS(context.Background())
  183. assert.Nil(s.T(), err, "GetBucketCORS Failed")
  184. assert.Equal(s.T(), 1, len(v.Rules), "GetBucketCORS wrong number rules")
  185. }
  186. func (s *CosTestSuite) TestVersionAndReplication() {
  187. opt := &cos.BucketPutVersionOptions{
  188. // Enabled or Suspended, the versioning once opened can not close.
  189. Status: "Enabled",
  190. }
  191. _, err := s.Client.Bucket.PutVersioning(context.Background(), opt)
  192. assert.Nil(s.T(), err, "PutVersioning Failed")
  193. v, _, err := s.Client.Bucket.GetVersioning(context.Background())
  194. assert.Nil(s.T(), err, "GetVersioning Failed")
  195. assert.Equal(s.T(), "Enabled", v.Status, "Get Wrong Version status")
  196. repOpt := &cos.PutBucketReplicationOptions{
  197. // qcs::cam::uin/[UIN]:uin/[Subaccount]
  198. Role: "qcs::cam::uin/" + kUin + ":uin/" + kUin,
  199. Rule: []cos.BucketReplicationRule{
  200. {
  201. ID: "1",
  202. // Enabled or Disabled
  203. Status: "Enabled",
  204. Destination: &cos.ReplicationDestination{
  205. // qcs::cos:[Region]::[Bucketname-Appid]
  206. Bucket: "qcs::cos:" + kRepRegion + "::" + kRepBucket + "-" + s.Appid,
  207. },
  208. },
  209. },
  210. }
  211. _, err = s.Client.Bucket.PutBucketReplication(context.Background(), repOpt)
  212. assert.Nil(s.T(), err, "PutBucketReplication Failed")
  213. vr, _, err := s.Client.Bucket.GetBucketReplication(context.Background())
  214. assert.Nil(s.T(), err, "GetBucketReplication Failed")
  215. for _, r := range vr.Rule {
  216. assert.Equal(s.T(), "Enabled", r.Status, "Get Wrong Version status")
  217. assert.Equal(s.T(), "qcs::cos:"+kRepRegion+"::"+kRepBucket+"-"+s.Appid, r.Destination.Bucket, "Get Wrong Version status")
  218. }
  219. _, err = s.Client.Bucket.DeleteBucketReplication(context.Background())
  220. assert.Nil(s.T(), err, "DeleteBucketReplication Failed")
  221. }
  222. func (s *CosTestSuite) TestBucketInventory() {
  223. id := "test1"
  224. dBucket := "qcs::cos:" + s.Region + "::" + s.Bucket + "-" + s.Appid
  225. opt := &cos.BucketPutInventoryOptions{
  226. ID: id,
  227. // True or False
  228. IsEnabled: "True",
  229. IncludedObjectVersions: "All",
  230. Filter: &cos.BucketInventoryFilter{
  231. Prefix: "test",
  232. },
  233. OptionalFields: &cos.BucketInventoryOptionalFields{
  234. BucketInventoryFields: []string{
  235. "Size", "LastModifiedDate",
  236. },
  237. },
  238. Schedule: &cos.BucketInventorySchedule{
  239. // Weekly or Daily
  240. Frequency: "Daily",
  241. },
  242. Destination: &cos.BucketInventoryDestination{
  243. Bucket: dBucket,
  244. Format: "CSV",
  245. },
  246. }
  247. _, err := s.Client.Bucket.PutInventory(context.Background(), id, opt)
  248. assert.Nil(s.T(), err, "PutBucketInventory Failed")
  249. v, _, err := s.Client.Bucket.GetInventory(context.Background(), id)
  250. assert.Nil(s.T(), err, "GetBucketInventory Failed")
  251. assert.Equal(s.T(), "test1", v.ID, "Get Wrong inventory id")
  252. assert.Equal(s.T(), "true", v.IsEnabled, "Get Wrong inventory isenabled")
  253. assert.Equal(s.T(), dBucket, v.Destination.Bucket, "Get Wrong inventory isenabled")
  254. _, err = s.Client.Bucket.DeleteInventory(context.Background(), id)
  255. assert.Nil(s.T(), err, "DeleteBucketInventory Failed")
  256. }
  257. func (s *CosTestSuite) TestBucketLogging() {
  258. tBucket := s.Bucket + "-" + s.Appid
  259. opt := &cos.BucketPutLoggingOptions{
  260. LoggingEnabled: &cos.BucketLoggingEnabled{
  261. TargetBucket: tBucket,
  262. },
  263. }
  264. _, err := s.Client.Bucket.PutLogging(context.Background(), opt)
  265. assert.Nil(s.T(), err, "PutLogging Failed")
  266. v, _, err := s.Client.Bucket.GetLogging(context.Background())
  267. assert.Nil(s.T(), err, "GetLogging Failed")
  268. assert.Equal(s.T(), tBucket, v.LoggingEnabled.TargetBucket, "Get Wrong Version status")
  269. }
  270. func (s *CosTestSuite) TestBucketTagging() {
  271. opt := &cos.BucketPutTaggingOptions{
  272. TagSet: []cos.BucketTaggingTag{
  273. {
  274. Key: "testk1",
  275. Value: "testv1",
  276. },
  277. {
  278. Key: "testk2",
  279. Value: "testv2",
  280. },
  281. },
  282. }
  283. _, err := s.Client.Bucket.PutTagging(context.Background(), opt)
  284. assert.Nil(s.T(), err, "Put Tagging Failed")
  285. v, _, err := s.Client.Bucket.GetTagging(context.Background())
  286. assert.Nil(s.T(), err, "Get Tagging Failed")
  287. assert.Equal(s.T(), v.TagSet[0].Key, opt.TagSet[0].Key, "Get Wrong Tag key")
  288. assert.Equal(s.T(), v.TagSet[0].Value, opt.TagSet[0].Value, "Get Wrong Tag value")
  289. assert.Equal(s.T(), v.TagSet[1].Key, opt.TagSet[1].Key, "Get Wrong Tag key")
  290. assert.Equal(s.T(), v.TagSet[1].Value, opt.TagSet[1].Value, "Get Wrong Tag value")
  291. }
  292. func (s *CosTestSuite) TestPutGetDeleteLifeCycle() {
  293. lc := &cos.BucketPutLifecycleOptions{
  294. Rules: []cos.BucketLifecycleRule{
  295. {
  296. ID: "1234",
  297. Filter: &cos.BucketLifecycleFilter{Prefix: "test"},
  298. Status: "Enabled",
  299. Transition: &cos.BucketLifecycleTransition{
  300. Days: 10,
  301. StorageClass: "Standard",
  302. },
  303. },
  304. },
  305. }
  306. _, err := s.Client.Bucket.PutLifecycle(context.Background(), lc)
  307. assert.Nil(s.T(), err, "PutBucketLifecycle Failed")
  308. _, r, err := s.Client.Bucket.GetLifecycle(context.Background())
  309. // Might cleaned by other case concrrent
  310. if err != nil && 404 != r.StatusCode {
  311. assert.Nil(s.T(), err, "GetBucketLifecycle Failed")
  312. }
  313. _, err = s.Client.Bucket.DeleteLifecycle(context.Background())
  314. assert.Nil(s.T(), err, "DeleteBucketLifecycle Failed")
  315. }
  316. func (s *CosTestSuite) TestPutGetDeleteWebsite() {
  317. opt := &cos.BucketPutWebsiteOptions{
  318. Index: "index.html",
  319. Error: &cos.ErrorDocument{"index_backup.html"},
  320. RoutingRules: &cos.WebsiteRoutingRules{
  321. []cos.WebsiteRoutingRule{
  322. {
  323. ConditionErrorCode: "404",
  324. RedirectProtocol: "https",
  325. RedirectReplaceKey: "404.html",
  326. },
  327. {
  328. ConditionPrefix: "docs/",
  329. RedirectProtocol: "https",
  330. RedirectReplaceKeyPrefix: "documents/",
  331. },
  332. },
  333. },
  334. }
  335. _, err := s.Client.Bucket.PutWebsite(context.Background(), opt)
  336. assert.Nil(s.T(), err, "PutBucketWebsite Failed")
  337. res, rsp, err := s.Client.Bucket.GetWebsite(context.Background())
  338. if err != nil && 404 != rsp.StatusCode {
  339. assert.Nil(s.T(), err, "GetBucketWebsite Failed")
  340. }
  341. assert.Equal(s.T(), opt.Index, res.Index, "GetBucketWebsite Failed")
  342. assert.Equal(s.T(), opt.Error, res.Error, "GetBucketWebsite Failed")
  343. assert.Equal(s.T(), opt.RedirectProtocol, res.RedirectProtocol, "GetBucketWebsite Failed")
  344. _, err = s.Client.Bucket.DeleteWebsite(context.Background())
  345. assert.Nil(s.T(), err, "DeleteBucketWebsite Failed")
  346. }
  347. func (s *CosTestSuite) TestListMultipartUploads() {
  348. // Create new upload
  349. name := "test_multipart" + time.Now().Format(time.RFC3339)
  350. flag := false
  351. v, _, err := s.Client.Object.InitiateMultipartUpload(context.Background(), name, nil)
  352. assert.Nil(s.T(), err, "InitiateMultipartUpload Failed")
  353. id := v.UploadID
  354. // List
  355. r, _, err := s.Client.Bucket.ListMultipartUploads(context.Background(), nil)
  356. assert.Nil(s.T(), err, "ListMultipartUploads Failed")
  357. for _, p := range r.Uploads {
  358. if p.Key == name {
  359. assert.Equal(s.T(), id, p.UploadID, "ListMultipartUploads wrong uploadid")
  360. flag = true
  361. }
  362. }
  363. assert.Equal(s.T(), true, flag, "ListMultipartUploads wrong key")
  364. // Abort
  365. _, err = s.Client.Object.AbortMultipartUpload(context.Background(), name, id)
  366. assert.Nil(s.T(), err, "AbortMultipartUpload Failed")
  367. }
  368. // Object API
  369. func (s *CosTestSuite) TestPutHeadGetDeleteObject_10MB() {
  370. name := "test/objectPut" + time.Now().Format(time.RFC3339)
  371. b := make([]byte, 1024*1024*10)
  372. _, err := rand.Read(b)
  373. content := fmt.Sprintf("%X", b)
  374. f := strings.NewReader(content)
  375. _, err = s.Client.Object.Put(context.Background(), name, f, nil)
  376. assert.Nil(s.T(), err, "PutObject Failed")
  377. _, err = s.Client.Object.Head(context.Background(), name, nil)
  378. assert.Nil(s.T(), err, "HeadObject Failed")
  379. _, err = s.Client.Object.Delete(context.Background(), name)
  380. assert.Nil(s.T(), err, "DeleteObject Failed")
  381. }
  382. func (s *CosTestSuite) TestPutGetDeleteObjectByFile_10MB() {
  383. // Create tmp file
  384. filePath := "tmpfile" + time.Now().Format(time.RFC3339)
  385. newfile, err := os.Create(filePath)
  386. assert.Nil(s.T(), err, "create tmp file Failed")
  387. defer newfile.Close()
  388. name := "test/objectPutByFile" + time.Now().Format(time.RFC3339)
  389. b := make([]byte, 1024*1024*10)
  390. _, err = rand.Read(b)
  391. newfile.Write(b)
  392. _, err = s.Client.Object.PutFromFile(context.Background(), name, filePath, nil)
  393. assert.Nil(s.T(), err, "PutObject Failed")
  394. // Over write tmp file
  395. _, err = s.Client.Object.GetToFile(context.Background(), name, filePath, nil)
  396. assert.Nil(s.T(), err, "HeadObject Failed")
  397. _, err = s.Client.Object.Delete(context.Background(), name)
  398. assert.Nil(s.T(), err, "DeleteObject Failed")
  399. // remove the local tmp file
  400. err = os.Remove(filePath)
  401. assert.Nil(s.T(), err, "remove local file Failed")
  402. }
  403. func (s *CosTestSuite) TestPutGetDeleteObjectSpecialName() {
  404. f := strings.NewReader("test")
  405. name := s.SepFileName + time.Now().Format(time.RFC3339)
  406. _, err := s.Client.Object.Put(context.Background(), name, f, nil)
  407. assert.Nil(s.T(), err, "PutObject Failed")
  408. resp, err := s.Client.Object.Get(context.Background(), name, nil)
  409. assert.Nil(s.T(), err, "GetObject Failed")
  410. defer resp.Body.Close()
  411. bs, _ := ioutil.ReadAll(resp.Body)
  412. assert.Equal(s.T(), "test", string(bs), "GetObject failed content wrong")
  413. _, err = s.Client.Object.Delete(context.Background(), name)
  414. assert.Nil(s.T(), err, "DeleteObject Failed")
  415. }
  416. func (s *CosTestSuite) TestPutObjectToNonExistBucket() {
  417. u := "http://gosdknonexistbucket-" + s.Appid + ".cos." + s.Region + ".myqcloud.com"
  418. iu, _ := url.Parse(u)
  419. ib := &cos.BaseURL{BucketURL: iu}
  420. client := cos.NewClient(ib, &http.Client{
  421. Transport: &cos.AuthorizationTransport{
  422. SecretID: os.Getenv("COS_SECRETID"),
  423. SecretKey: os.Getenv("COS_SECRETKEY"),
  424. },
  425. })
  426. name := "test/objectPut.go"
  427. f := strings.NewReader("test")
  428. r, err := client.Object.Put(context.Background(), name, f, nil)
  429. assert.NotNil(s.T(), err, "PutObject ToNonExistBucket Failed")
  430. assert.Equal(s.T(), 404, r.StatusCode, "PutObject ToNonExistBucket, not 404")
  431. }
  432. func (s *CosTestSuite) TestPutGetObjectACL() {
  433. name := "test/objectACL.go" + time.Now().Format(time.RFC3339)
  434. f := strings.NewReader("test")
  435. _, err := s.Client.Object.Put(context.Background(), name, f, nil)
  436. assert.Nil(s.T(), err, "PutObject Failed")
  437. // Put acl
  438. opt := &cos.ObjectPutACLOptions{
  439. Header: &cos.ACLHeaderOptions{
  440. XCosACL: "public-read",
  441. },
  442. }
  443. _, err = s.Client.Object.PutACL(context.Background(), name, opt)
  444. assert.Nil(s.T(), err, "PutObjectACL Failed")
  445. v, _, err := s.Client.Object.GetACL(context.Background(), name)
  446. assert.Nil(s.T(), err, "GetObjectACL Failed")
  447. assert.Equal(s.T(), 2, len(v.AccessControlList), "GetLifecycle wrong number rules")
  448. _, err = s.Client.Object.Delete(context.Background(), name)
  449. assert.Nil(s.T(), err, "DeleteObject Failed")
  450. }
  451. func (s *CosTestSuite) TestPutObjectRestore() {
  452. name := "archivetest"
  453. putOpt := &cos.ObjectPutOptions{
  454. ObjectPutHeaderOptions: &cos.ObjectPutHeaderOptions{
  455. XCosStorageClass: "ARCHIVE",
  456. },
  457. }
  458. f := strings.NewReader("test")
  459. _, err := s.Client.Object.Put(context.Background(), name, f, putOpt)
  460. assert.Nil(s.T(), err, "PutObject Archive faild")
  461. opt := &cos.ObjectRestoreOptions{
  462. Days: 2,
  463. Tier: &cos.CASJobParameters{
  464. // Standard, Exepdited and Bulk
  465. Tier: "Expedited",
  466. },
  467. }
  468. resp, _ := s.Client.Object.PostRestore(context.Background(), name, opt)
  469. retCode := resp.StatusCode
  470. if retCode != 200 && retCode != 202 && retCode != 409 {
  471. right := false
  472. fmt.Println("PutObjectRestore get code is:", retCode)
  473. assert.Equal(s.T(), true, right, "PutObjectRestore Failed")
  474. }
  475. }
  476. func (s *CosTestSuite) TestCopyObject() {
  477. u := "http://" + kRepBucket + "-" + s.Appid + ".cos." + kRepRegion + ".myqcloud.com"
  478. iu, _ := url.Parse(u)
  479. ib := &cos.BaseURL{BucketURL: iu}
  480. c := cos.NewClient(ib, &http.Client{
  481. Transport: &cos.AuthorizationTransport{
  482. SecretID: os.Getenv("COS_SECRETID"),
  483. SecretKey: os.Getenv("COS_SECRETKEY"),
  484. },
  485. })
  486. opt := &cos.BucketPutOptions{
  487. XCosACL: "public-read",
  488. }
  489. // Notice in intranet the bucket host sometimes has i/o timeout problem
  490. r, err := c.Bucket.Put(context.Background(), opt)
  491. if err != nil && r.StatusCode == 409 {
  492. fmt.Println("BucketAlreadyOwnedByYou")
  493. } else if err != nil {
  494. assert.Nil(s.T(), err, "PutBucket Failed")
  495. }
  496. source := "test/objectMove1" + time.Now().Format(time.RFC3339)
  497. expected := "test"
  498. f := strings.NewReader(expected)
  499. r, err = c.Object.Put(context.Background(), source, f, nil)
  500. assert.Nil(s.T(), err, "PutObject Failed")
  501. var version_id string
  502. if r.Header["X-Cos-Version-Id"] != nil {
  503. version_id = r.Header.Get("X-Cos-Version-Id")
  504. }
  505. time.Sleep(3 * time.Second)
  506. // Copy file
  507. soruceURL := fmt.Sprintf("%s/%s", iu.Host, source)
  508. dest := "test/objectMove1" + time.Now().Format(time.RFC3339)
  509. //opt := &cos.ObjectCopyOptions{}
  510. if version_id == "" {
  511. _, _, err = s.Client.Object.Copy(context.Background(), dest, soruceURL, nil)
  512. } else {
  513. _, _, err = s.Client.Object.Copy(context.Background(), dest, soruceURL, nil, version_id)
  514. }
  515. assert.Nil(s.T(), err, "PutObjectCopy Failed")
  516. // Check content
  517. resp, err := s.Client.Object.Get(context.Background(), dest, nil)
  518. assert.Nil(s.T(), err, "GetObject Failed")
  519. bs, _ := ioutil.ReadAll(resp.Body)
  520. resp.Body.Close()
  521. result := string(bs)
  522. assert.Equal(s.T(), expected, result, "PutObjectCopy Failed, wrong content")
  523. }
  524. func (s *CosTestSuite) TestCreateAbortMultipartUpload() {
  525. name := "test_multipart" + time.Now().Format(time.RFC3339)
  526. v, _, err := s.Client.Object.InitiateMultipartUpload(context.Background(), name, nil)
  527. assert.Nil(s.T(), err, "InitiateMultipartUpload Failed")
  528. _, err = s.Client.Object.AbortMultipartUpload(context.Background(), name, v.UploadID)
  529. assert.Nil(s.T(), err, "AbortMultipartUpload Failed")
  530. }
  531. func (s *CosTestSuite) TestCreateCompleteMultipartUpload() {
  532. name := "test/test_complete_upload" + time.Now().Format(time.RFC3339)
  533. v, _, err := s.Client.Object.InitiateMultipartUpload(context.Background(), name, nil)
  534. uploadID := v.UploadID
  535. blockSize := 1024 * 1024 * 3
  536. opt := &cos.CompleteMultipartUploadOptions{}
  537. for i := 1; i < 3; i++ {
  538. b := make([]byte, blockSize)
  539. _, err := rand.Read(b)
  540. content := fmt.Sprintf("%X", b)
  541. f := strings.NewReader(content)
  542. resp, err := s.Client.Object.UploadPart(
  543. context.Background(), name, uploadID, i, f, nil,
  544. )
  545. assert.Nil(s.T(), err, "UploadPart Failed")
  546. etag := resp.Header.Get("Etag")
  547. opt.Parts = append(opt.Parts, cos.Object{
  548. PartNumber: i, ETag: etag},
  549. )
  550. }
  551. _, _, err = s.Client.Object.CompleteMultipartUpload(
  552. context.Background(), name, uploadID, opt,
  553. )
  554. assert.Nil(s.T(), err, "CompleteMultipartUpload Failed")
  555. }
  556. func (s *CosTestSuite) TestSSE_C() {
  557. name := "test/TestSSE_C"
  558. content := "test sse-c " + time.Now().Format(time.RFC3339)
  559. f := strings.NewReader(content)
  560. putOpt := &cos.ObjectPutOptions{
  561. ObjectPutHeaderOptions: &cos.ObjectPutHeaderOptions{
  562. ContentType: "text/html",
  563. //XCosServerSideEncryption: "AES256",
  564. XCosSSECustomerAglo: "AES256",
  565. XCosSSECustomerKey: "MDEyMzQ1Njc4OUFCQ0RFRjAxMjM0NTY3ODlBQkNERUY=",
  566. XCosSSECustomerKeyMD5: "U5L61r7jcwdNvT7frmUG8g==",
  567. },
  568. ACLHeaderOptions: &cos.ACLHeaderOptions{
  569. XCosACL: "public-read",
  570. //XCosACL: "private",
  571. },
  572. }
  573. _, err := s.Client.Object.Put(context.Background(), name, f, putOpt)
  574. assert.Nil(s.T(), err, "PutObject with SSE failed")
  575. headOpt := &cos.ObjectHeadOptions{
  576. XCosSSECustomerAglo: "AES256",
  577. XCosSSECustomerKey: "MDEyMzQ1Njc4OUFCQ0RFRjAxMjM0NTY3ODlBQkNERUY=",
  578. XCosSSECustomerKeyMD5: "U5L61r7jcwdNvT7frmUG8g==",
  579. }
  580. _, err = s.Client.Object.Head(context.Background(), name, headOpt)
  581. assert.Nil(s.T(), err, "HeadObject with SSE failed")
  582. getOpt := &cos.ObjectGetOptions{
  583. XCosSSECustomerAglo: "AES256",
  584. XCosSSECustomerKey: "MDEyMzQ1Njc4OUFCQ0RFRjAxMjM0NTY3ODlBQkNERUY=",
  585. XCosSSECustomerKeyMD5: "U5L61r7jcwdNvT7frmUG8g==",
  586. }
  587. var resp *cos.Response
  588. resp, err = s.Client.Object.Get(context.Background(), name, getOpt)
  589. assert.Nil(s.T(), err, "GetObject with SSE failed")
  590. bodyBytes, _ := ioutil.ReadAll(resp.Body)
  591. bodyContent := string(bodyBytes)
  592. assert.Equal(s.T(), content, bodyContent, "GetObject with SSE failed, want: %+v, res: %+v", content, bodyContent)
  593. copyOpt := &cos.ObjectCopyOptions{
  594. &cos.ObjectCopyHeaderOptions{
  595. XCosCopySourceSSECustomerAglo: "AES256",
  596. XCosCopySourceSSECustomerKey: "MDEyMzQ1Njc4OUFCQ0RFRjAxMjM0NTY3ODlBQkNERUY=",
  597. XCosCopySourceSSECustomerKeyMD5: "U5L61r7jcwdNvT7frmUG8g==",
  598. },
  599. &cos.ACLHeaderOptions{},
  600. }
  601. copySource := s.Bucket + "-" + s.Appid + ".cos." + s.Region + ".myqcloud.com/" + name
  602. _, _, err = s.Client.Object.Copy(context.Background(), "test/TestSSE_C_Copy", copySource, copyOpt)
  603. assert.Nil(s.T(), err, "CopyObject with SSE failed")
  604. partIni := &cos.MultiUploadOptions{
  605. OptIni: &cos.InitiateMultipartUploadOptions{
  606. &cos.ACLHeaderOptions{},
  607. &cos.ObjectPutHeaderOptions{
  608. XCosSSECustomerAglo: "AES256",
  609. XCosSSECustomerKey: "MDEyMzQ1Njc4OUFCQ0RFRjAxMjM0NTY3ODlBQkNERUY=",
  610. XCosSSECustomerKeyMD5: "U5L61r7jcwdNvT7frmUG8g==",
  611. },
  612. },
  613. PartSize: 1,
  614. }
  615. filePath := "tmpfile" + time.Now().Format(time.RFC3339)
  616. newFile, err := os.Create(filePath)
  617. assert.Nil(s.T(), err, "create tmp file Failed")
  618. defer newFile.Close()
  619. b := make([]byte, 1024*10)
  620. _, err = rand.Read(b)
  621. newFile.Write(b)
  622. _, _, err = s.Client.Object.MultiUpload(context.Background(), "test/TestSSE_C_MultiUpload", filePath, partIni)
  623. assert.Nil(s.T(), err, "MultiUpload with SSE failed")
  624. err = os.Remove(filePath)
  625. assert.Nil(s.T(), err, "remove local file Failed")
  626. }
  627. func (s *CosTestSuite) TestMultiUpload() {
  628. filePath := "tmpfile" + time.Now().Format(time.RFC3339)
  629. newFile, err := os.Create(filePath)
  630. assert.Nil(s.T(), err, "create tmp file Failed")
  631. defer newFile.Close()
  632. b := make([]byte, 1024*1024*10)
  633. _, err = rand.Read(b)
  634. newFile.Write(b)
  635. partIni := &cos.MultiUploadOptions{}
  636. _, _, err = s.Client.Object.MultiUpload(context.Background(), "test/Test_MultiUpload", filePath, partIni)
  637. err = os.Remove(filePath)
  638. assert.Nil(s.T(), err, "remove tmp file failed")
  639. }
  640. func (s *CosTestSuite) TestBatch() {
  641. client := cos.NewClient(s.Client.BaseURL, &http.Client{
  642. Transport: &cos.AuthorizationTransport{
  643. SecretID: os.Getenv("COS_SECRETID"),
  644. SecretKey: os.Getenv("COS_SECRETKEY"),
  645. },
  646. })
  647. source_name := "test/1.txt"
  648. sf := strings.NewReader("batch test content")
  649. _, err := client.Object.Put(context.Background(), source_name, sf, nil)
  650. assert.Nil(s.T(), err, "object put Failed")
  651. manifest_name := "test/manifest.csv"
  652. f := strings.NewReader(kBatchBucket + "," + source_name)
  653. resp, err := client.Object.Put(context.Background(), manifest_name, f, nil)
  654. assert.Nil(s.T(), err, "object put Failed")
  655. etag := resp.Header.Get("ETag")
  656. uuid_str := uuid.New().String()
  657. opt := &cos.BatchCreateJobOptions{
  658. ClientRequestToken: uuid_str,
  659. ConfirmationRequired: "true",
  660. Description: "test batch",
  661. Manifest: &cos.BatchJobManifest{
  662. Location: &cos.BatchJobManifestLocation{
  663. ETag: etag,
  664. ObjectArn: "qcs::cos:" + kBatchRegion + ":uid/" + s.Appid + ":" + kBatchBucket + "/" + manifest_name,
  665. },
  666. Spec: &cos.BatchJobManifestSpec{
  667. Fields: []string{"Bucket", "Key"},
  668. Format: "COSBatchOperations_CSV_V1",
  669. },
  670. },
  671. Operation: &cos.BatchJobOperation{
  672. PutObjectCopy: &cos.BatchJobOperationCopy{
  673. TargetResource: "qcs::cos:" + kBatchRegion + ":uid/" + s.Appid + ":" + kTargetBatchBucket,
  674. },
  675. },
  676. Priority: 1,
  677. Report: &cos.BatchJobReport{
  678. Bucket: "qcs::cos:" + kBatchRegion + ":uid/" + s.Appid + ":" + kBatchBucket,
  679. Enabled: "true",
  680. Format: "Report_CSV_V1",
  681. Prefix: "job-result",
  682. ReportScope: "AllTasks",
  683. },
  684. RoleArn: "qcs::cam::uin/" + kUin + ":roleName/COSBatch_QcsRole",
  685. }
  686. headers := &cos.BatchRequestHeaders{
  687. XCosAppid: kAppid,
  688. }
  689. res1, _, err := client.Batch.CreateJob(context.Background(), opt, headers)
  690. assert.Nil(s.T(), err, "create job Failed")
  691. jobid := res1.JobId
  692. res2, _, err := client.Batch.DescribeJob(context.Background(), jobid, headers)
  693. assert.Nil(s.T(), err, "describe job Failed")
  694. assert.Equal(s.T(), res2.Job.ConfirmationRequired, "true", "ConfirmationRequired not right")
  695. assert.Equal(s.T(), res2.Job.Description, "test batch", "Description not right")
  696. assert.Equal(s.T(), res2.Job.JobId, jobid, "jobid not right")
  697. assert.Equal(s.T(), res2.Job.Priority, 1, "priority not right")
  698. assert.Equal(s.T(), res2.Job.RoleArn, "qcs::cam::uin/"+kUin+":roleName/COSBatch_QcsRole", "priority not right")
  699. _, _, err = client.Batch.ListJobs(context.Background(), nil, headers)
  700. assert.Nil(s.T(), err, "list jobs failed")
  701. up_opt := &cos.BatchUpdatePriorityOptions{
  702. JobId: jobid,
  703. Priority: 3,
  704. }
  705. res3, _, err := client.Batch.UpdateJobPriority(context.Background(), up_opt, headers)
  706. assert.Nil(s.T(), err, "list jobs failed")
  707. assert.Equal(s.T(), res3.JobId, jobid, "jobid failed")
  708. assert.Equal(s.T(), res3.Priority, 3, "priority not right")
  709. // 等待状态变成Suspended
  710. for i := 0; i < 50; i = i + 1 {
  711. res, _, err := client.Batch.DescribeJob(context.Background(), jobid, headers)
  712. assert.Nil(s.T(), err, "describe job Failed")
  713. assert.Equal(s.T(), res2.Job.ConfirmationRequired, "true", "ConfirmationRequired not right")
  714. assert.Equal(s.T(), res2.Job.Description, "test batch", "Description not right")
  715. assert.Equal(s.T(), res2.Job.JobId, jobid, "jobid not right")
  716. assert.Equal(s.T(), res2.Job.Priority, 1, "priority not right")
  717. assert.Equal(s.T(), res2.Job.RoleArn, "qcs::cam::uin/"+kUin+":roleName/COSBatch_QcsRole", "priority not right")
  718. if res.Job.Status == "Suspended" {
  719. break
  720. }
  721. if i == 9 {
  722. assert.Error(s.T(), errors.New("Job status is not Suspended or timeout"))
  723. }
  724. time.Sleep(time.Second * 2)
  725. }
  726. us_opt := &cos.BatchUpdateStatusOptions{
  727. JobId: jobid,
  728. RequestedJobStatus: "Ready", // 允许状态转换见 https://cloud.tencent.com/document/product/436/38604
  729. StatusUpdateReason: "to test",
  730. }
  731. res4, _, err := client.Batch.UpdateJobStatus(context.Background(), us_opt, headers)
  732. assert.Nil(s.T(), err, "list jobs failed")
  733. assert.Equal(s.T(), res4.JobId, jobid, "jobid failed")
  734. assert.Equal(s.T(), res4.Status, "Ready", "status failed")
  735. assert.Equal(s.T(), res4.StatusUpdateReason, "to test", "StatusUpdateReason failed")
  736. }
  737. func (s *CosTestSuite) TestEncryption() {
  738. opt := &cos.BucketPutEncryptionOptions{
  739. Rule: &cos.BucketEncryptionConfiguration{
  740. SSEAlgorithm: "AES256",
  741. },
  742. }
  743. _, err := s.Client.Bucket.PutEncryption(context.Background(), opt)
  744. assert.Nil(s.T(), err, "PutEncryption Failed")
  745. res, _, err := s.Client.Bucket.GetEncryption(context.Background())
  746. assert.Nil(s.T(), err, "GetEncryption Failed")
  747. assert.Equal(s.T(), opt.Rule.SSEAlgorithm, res.Rule.SSEAlgorithm, "GetEncryption Failed")
  748. _, err = s.Client.Bucket.DeleteEncryption(context.Background())
  749. assert.Nil(s.T(), err, "DeleteEncryption Failed")
  750. }
  751. func (s *CosTestSuite) TestReferer() {
  752. opt := &cos.BucketPutRefererOptions{
  753. Status: "Enabled",
  754. RefererType: "White-List",
  755. DomainList: []string{
  756. "*.qq.com",
  757. "*.qcloud.com",
  758. },
  759. EmptyReferConfiguration: "Allow",
  760. }
  761. _, err := s.Client.Bucket.PutReferer(context.Background(), opt)
  762. assert.Nil(s.T(), err, "PutReferer Failed")
  763. res, _, err := s.Client.Bucket.GetReferer(context.Background())
  764. assert.Nil(s.T(), err, "GetReferer Failed")
  765. assert.Equal(s.T(), opt.Status, res.Status, "GetReferer Failed")
  766. assert.Equal(s.T(), opt.RefererType, res.RefererType, "GetReferer Failed")
  767. assert.Equal(s.T(), opt.DomainList, res.DomainList, "GetReferer Failed")
  768. assert.Equal(s.T(), opt.EmptyReferConfiguration, res.EmptyReferConfiguration, "GetReferer Failed")
  769. }
  770. // End of api test
  771. // All methods that begin with "Test" are run as tests within a
  772. // suite.
  773. // In order for 'go test' to run this suite, we need to create
  774. // a normal test function and pass our suite to suite.Run
  775. func TestCosTestSuite(t *testing.T) {
  776. suite.Run(t, new(CosTestSuite))
  777. }
  778. func (s *CosTestSuite) TearDownSuite() {
  779. // Clean the file in bucket
  780. // r, _, err := s.Client.Bucket.ListMultipartUploads(context.Background(), nil)
  781. // assert.Nil(s.T(), err, "ListMultipartUploads Failed")
  782. // for _, p := range r.Uploads {
  783. // // Abort
  784. // _, err = s.Client.Object.AbortMultipartUpload(context.Background(), p.Key, p.UploadID)
  785. // assert.Nil(s.T(), err, "AbortMultipartUpload Failed")
  786. // }
  787. // // Delete objects
  788. // opt := &cos.BucketGetOptions{
  789. // MaxKeys: 500,
  790. // }
  791. // v, _, err := s.Client.Bucket.Get(context.Background(), opt)
  792. // assert.Nil(s.T(), err, "GetBucket Failed")
  793. // for _, c := range v.Contents {
  794. // _, err := s.Client.Object.Delete(context.Background(), c.Key)
  795. // assert.Nil(s.T(), err, "DeleteObject Failed")
  796. // }
  797. // When clean up these infos, can not solve the concurrent test problem
  798. fmt.Println("tear down~")
  799. }