bucket.go 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639
  1. package oss
  2. import (
  3. "bytes"
  4. "crypto/md5"
  5. "encoding/base64"
  6. "encoding/xml"
  7. "hash"
  8. "hash/crc64"
  9. "io"
  10. "io/ioutil"
  11. "net/http"
  12. "net/url"
  13. "os"
  14. "strconv"
  15. )
  16. // Bucket implements the operations of object.
  17. type Bucket struct {
  18. Client Client
  19. BucketName string
  20. }
  21. //
  22. // PutObject 新建Object,如果Object已存在,覆盖原有Object。
  23. //
  24. // objectKey 上传对象的名称,使用UTF-8编码、长度必须在1-1023字节之间、不能以“/”或者“\”字符开头。
  25. // reader io.Reader读取object的数据。
  26. // options 上传对象时可以指定对象的属性,可用选项有CacheControl、ContentDisposition、ContentEncoding、
  27. // Expires、ServerSideEncryption、ObjectACL、Meta,具体含义请参看
  28. // https://help.aliyun.com/document_detail/oss/api-reference/object/PutObject.html
  29. //
  30. // error 操作无错误为nil,非nil为错误信息。
  31. //
  32. func (bucket Bucket) PutObject(objectKey string, reader io.Reader, options ...Option) error {
  33. opts := addContentType(options, objectKey)
  34. request := &PutObjectRequest{
  35. ObjectKey: objectKey,
  36. Reader: reader,
  37. }
  38. resp, err := bucket.DoPutObject(request, opts)
  39. if err != nil {
  40. return err
  41. }
  42. defer resp.Body.Close()
  43. return err
  44. }
  45. //
  46. // PutObjectFromFile 新建Object,内容从本地文件中读取。
  47. //
  48. // objectKey 上传对象的名称。
  49. // filePath 本地文件,上传对象的值为该文件内容。
  50. // options 上传对象时可以指定对象的属性。详见PutObject的options。
  51. //
  52. // error 操作无错误为nil,非nil为错误信息。
  53. //
  54. func (bucket Bucket) PutObjectFromFile(objectKey, filePath string, options ...Option) error {
  55. fd, err := os.Open(filePath)
  56. if err != nil {
  57. return err
  58. }
  59. defer fd.Close()
  60. opts := addContentType(options, filePath, objectKey)
  61. request := &PutObjectRequest{
  62. ObjectKey: objectKey,
  63. Reader: fd,
  64. }
  65. resp, err := bucket.DoPutObject(request, opts)
  66. if err != nil {
  67. return err
  68. }
  69. defer resp.Body.Close()
  70. return err
  71. }
  72. //
  73. // DoPutObject 上传文件。
  74. //
  75. // request 上传请求。
  76. // options 上传选项。
  77. //
  78. // Response 上传请求返回值。
  79. // error 操作无错误为nil,非nil为错误信息。
  80. //
  81. func (bucket Bucket) DoPutObject(request *PutObjectRequest, options []Option) (*Response, error) {
  82. isOptSet, _, _ := isOptionSet(options, HTTPHeaderContentType)
  83. if !isOptSet {
  84. options = addContentType(options, request.ObjectKey)
  85. }
  86. if request.Listener == nil {
  87. request.Listener = getProgressListener(options)
  88. }
  89. resp, err := bucket.do("PUT", request.ObjectKey, "", "", options, request.Reader, request.Listener)
  90. if err != nil {
  91. return nil, err
  92. }
  93. if bucket.getConfig().IsEnableCRC {
  94. err = checkCRC(resp, "DoPutObject")
  95. if err != nil {
  96. return resp, err
  97. }
  98. }
  99. err = checkRespCode(resp.StatusCode, []int{http.StatusOK})
  100. return resp, err
  101. }
  102. //
  103. // GetObject 下载文件。
  104. //
  105. // objectKey 下载的文件名称。
  106. // options 对象的属性限制项,可选值有Range、IfModifiedSince、IfUnmodifiedSince、IfMatch、
  107. // IfNoneMatch、AcceptEncoding,详细请参考
  108. // https://help.aliyun.com/document_detail/oss/api-reference/object/GetObject.html
  109. //
  110. // io.ReadCloser reader,读取数据后需要close。error为nil时有效。
  111. // error 操作无错误为nil,非nil为错误信息。
  112. //
  113. func (bucket Bucket) GetObject(objectKey string, options ...Option) (io.ReadCloser, error) {
  114. result, err := bucket.DoGetObject(&GetObjectRequest{objectKey, nil}, options)
  115. if err != nil {
  116. return nil, err
  117. }
  118. return result.Response.Body, nil
  119. }
  120. //
  121. // GetObjectToFile 下载文件。
  122. //
  123. // objectKey 下载的文件名称。
  124. // filePath 下载对象的内容写到该本地文件。
  125. // options 对象的属性限制项。详见GetObject的options。
  126. //
  127. // error 操作无错误时返回error为nil,非nil为错误说明。
  128. //
  129. func (bucket Bucket) GetObjectToFile(objectKey, filePath string, options ...Option) error {
  130. tempFilePath := filePath + TempFileSuffix
  131. // 读取Object内容
  132. result, err := bucket.DoGetObject(&GetObjectRequest{objectKey, nil}, options)
  133. if err != nil {
  134. return err
  135. }
  136. defer result.Response.Body.Close()
  137. // 如果文件不存在则创建,存在则清空
  138. fd, err := os.OpenFile(tempFilePath, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, FilePermMode)
  139. if err != nil {
  140. return err
  141. }
  142. // 存储数据到文件
  143. _, err = io.Copy(fd, result.Response.Body)
  144. fd.Close()
  145. if err != nil {
  146. return err
  147. }
  148. // 比较CRC值
  149. hasRange, _, _ := isOptionSet(options, HTTPHeaderRange)
  150. if bucket.getConfig().IsEnableCRC && !hasRange {
  151. result.Response.ClientCRC = result.ClientCRC.Sum64()
  152. err = checkCRC(result.Response, "GetObjectToFile")
  153. if err != nil {
  154. os.Remove(tempFilePath)
  155. return err
  156. }
  157. }
  158. return os.Rename(tempFilePath, filePath)
  159. }
  160. //
  161. // DoGetObject 下载文件
  162. //
  163. // request 下载请求
  164. // options 对象的属性限制项。详见GetObject的options。
  165. //
  166. // GetObjectResult 下载请求返回值。
  167. // error 操作无错误为nil,非nil为错误信息。
  168. //
  169. func (bucket Bucket) DoGetObject(request *GetObjectRequest, options []Option) (*GetObjectResult, error) {
  170. resp, err := bucket.do("GET", request.ObjectKey, "", "", options, nil, nil)
  171. if err != nil {
  172. return nil, err
  173. }
  174. result := &GetObjectResult{
  175. Response: resp,
  176. }
  177. // crc
  178. var crcCalc hash.Hash64
  179. hasRange, _, _ := isOptionSet(options, HTTPHeaderRange)
  180. if bucket.getConfig().IsEnableCRC && !hasRange {
  181. crcCalc = crc64.New(crcTable())
  182. result.ServerCRC = resp.ServerCRC
  183. result.ClientCRC = crcCalc
  184. }
  185. // progress
  186. if request.Listener == nil {
  187. request.Listener = getProgressListener(options)
  188. }
  189. contentLen, _ := strconv.ParseInt(resp.Headers.Get(HTTPHeaderContentLength), 10, 64)
  190. resp.Body = ioutil.NopCloser(TeeReader(resp.Body, crcCalc, contentLen, request.Listener, nil))
  191. return result, nil
  192. }
  193. //
  194. // CopyObject 同一个bucket内拷贝Object。
  195. //
  196. // srcObjectKey Copy的源对象。
  197. // destObjectKey Copy的目标对象。
  198. // options Copy对象时,您可以指定源对象的限制条件,满足限制条件时copy,不满足时返回错误,您可以选择如下选项CopySourceIfMatch、
  199. // CopySourceIfNoneMatch、CopySourceIfModifiedSince、CopySourceIfUnmodifiedSince、MetadataDirective。
  200. // Copy对象时,您可以指定目标对象的属性,如CacheControl、ContentDisposition、ContentEncoding、Expires、
  201. // ServerSideEncryption、ObjectACL、Meta,选项的含义请参看
  202. // https://help.aliyun.com/document_detail/oss/api-reference/object/CopyObject.html
  203. //
  204. // error 操作无错误为nil,非nil为错误信息。
  205. //
  206. func (bucket Bucket) CopyObject(srcObjectKey, destObjectKey string, options ...Option) (CopyObjectResult, error) {
  207. var out CopyObjectResult
  208. options = append(options, CopySource(bucket.BucketName, url.QueryEscape(srcObjectKey)))
  209. resp, err := bucket.do("PUT", destObjectKey, "", "", options, nil, nil)
  210. if err != nil {
  211. return out, err
  212. }
  213. defer resp.Body.Close()
  214. err = xmlUnmarshal(resp.Body, &out)
  215. return out, err
  216. }
  217. //
  218. // CopyObjectTo bucket间拷贝object。
  219. //
  220. // srcObjectKey 源Object名称。源Bucket名称为Bucket.BucketName。
  221. // destBucketName 目标Bucket名称。
  222. // destObjectKey 目标Object名称。
  223. // options Copy选项,详见CopyObject的options。
  224. //
  225. // error 操作无错误为nil,非nil为错误信息。
  226. //
  227. func (bucket Bucket) CopyObjectTo(destBucketName, destObjectKey, srcObjectKey string, options ...Option) (CopyObjectResult, error) {
  228. return bucket.copy(srcObjectKey, destBucketName, destObjectKey, options...)
  229. }
  230. //
  231. // CopyObjectFrom bucket间拷贝object。
  232. //
  233. // srcBucketName 源Bucket名称。
  234. // srcObjectKey 源Object名称。
  235. // destObjectKey 目标Object名称。目标Bucket名称为Bucket.BucketName。
  236. // options Copy选项,详见CopyObject的options。
  237. //
  238. // error 操作无错误为nil,非nil为错误信息。
  239. //
  240. func (bucket Bucket) CopyObjectFrom(srcBucketName, srcObjectKey, destObjectKey string, options ...Option) (CopyObjectResult, error) {
  241. destBucketName := bucket.BucketName
  242. var out CopyObjectResult
  243. srcBucket, err := bucket.Client.Bucket(srcBucketName)
  244. if err != nil {
  245. return out, err
  246. }
  247. return srcBucket.copy(srcObjectKey, destBucketName, destObjectKey, options...)
  248. }
  249. func (bucket Bucket) copy(srcObjectKey, destBucketName, destObjectKey string, options ...Option) (CopyObjectResult, error) {
  250. var out CopyObjectResult
  251. options = append(options, CopySource(bucket.BucketName, url.QueryEscape(srcObjectKey)))
  252. headers := make(map[string]string)
  253. err := handleOptions(headers, options)
  254. if err != nil {
  255. return out, err
  256. }
  257. resp, err := bucket.Client.Conn.Do("PUT", destBucketName, destObjectKey, "", "", headers, nil, 0, nil)
  258. if err != nil {
  259. return out, err
  260. }
  261. defer resp.Body.Close()
  262. err = xmlUnmarshal(resp.Body, &out)
  263. return out, err
  264. }
  265. //
  266. // AppendObject 追加方式上传。
  267. //
  268. // AppendObject参数必须包含position,其值指定从何处进行追加。首次追加操作的position必须为0,
  269. // 后续追加操作的position是Object的当前长度。例如,第一次Append Object请求指定position值为0,
  270. // content-length是65536;那么,第二次Append Object需要指定position为65536。
  271. // 每次操作成功后,响应头部x-oss-next-append-position也会标明下一次追加的position。
  272. //
  273. // objectKey 需要追加的Object。
  274. // reader io.Reader,读取追的内容。
  275. // appendPosition object追加的起始位置。
  276. // destObjectProperties 第一次追加时指定新对象的属性,如CacheControl、ContentDisposition、ContentEncoding、
  277. // Expires、ServerSideEncryption、ObjectACL。
  278. //
  279. // int64 下次追加的开始位置,error为nil空时有效。
  280. // error 操作无错误为nil,非nil为错误信息。
  281. //
  282. func (bucket Bucket) AppendObject(objectKey string, reader io.Reader, appendPosition int64, options ...Option) (int64, error) {
  283. request := &AppendObjectRequest{
  284. ObjectKey: objectKey,
  285. Reader: reader,
  286. Position: appendPosition,
  287. }
  288. result, err := bucket.DoAppendObject(request, options)
  289. return result.NextPosition, err
  290. }
  291. //
  292. // DoAppendObject 追加上传。
  293. //
  294. // request 追加上传请求。
  295. // options 追加上传选项。
  296. //
  297. // AppendObjectResult 追加上传请求返回值。
  298. // error 操作无错误为nil,非nil为错误信息。
  299. //
  300. func (bucket Bucket) DoAppendObject(request *AppendObjectRequest, options []Option) (*AppendObjectResult, error) {
  301. params := "append&position=" + strconv.FormatInt(request.Position, 10)
  302. headers := make(map[string]string)
  303. opts := addContentType(options, request.ObjectKey)
  304. handleOptions(headers, opts)
  305. var initCRC uint64
  306. isCRCSet, initCRCStr, _ := isOptionSet(options, initCRC64)
  307. if isCRCSet {
  308. initCRC, _ = strconv.ParseUint(initCRCStr.(string), 10, 64)
  309. }
  310. if request.Listener == nil {
  311. request.Listener = getProgressListener(options)
  312. }
  313. handleOptions(headers, opts)
  314. resp, err := bucket.Client.Conn.Do("POST", bucket.BucketName, request.ObjectKey, params, params, headers,
  315. request.Reader, initCRC, request.Listener)
  316. if err != nil {
  317. return nil, err
  318. }
  319. defer resp.Body.Close()
  320. nextPosition, _ := strconv.ParseInt(resp.Headers.Get(HTTPHeaderOssNextAppendPosition), 10, 64)
  321. result := &AppendObjectResult{
  322. NextPosition: nextPosition,
  323. CRC: resp.ServerCRC,
  324. }
  325. if bucket.getConfig().IsEnableCRC && isCRCSet {
  326. err = checkCRC(resp, "AppendObject")
  327. if err != nil {
  328. return result, err
  329. }
  330. }
  331. return result, nil
  332. }
  333. //
  334. // DeleteObject 删除Object。
  335. //
  336. // objectKey 待删除Object。
  337. //
  338. // error 操作无错误为nil,非nil为错误信息。
  339. //
  340. func (bucket Bucket) DeleteObject(objectKey string) error {
  341. resp, err := bucket.do("DELETE", objectKey, "", "", nil, nil, nil)
  342. if err != nil {
  343. return err
  344. }
  345. defer resp.Body.Close()
  346. return checkRespCode(resp.StatusCode, []int{http.StatusNoContent})
  347. }
  348. //
  349. // DeleteObjects 批量删除object。
  350. //
  351. // objectKeys 待删除object类表。
  352. // options 删除选项,DeleteObjectsQuiet,是否是安静模式,默认不使用。
  353. //
  354. // DeleteObjectsResult 非安静模式的的返回值。
  355. // error 操作无错误为nil,非nil为错误信息。
  356. //
  357. func (bucket Bucket) DeleteObjects(objectKeys []string, options ...Option) (DeleteObjectsResult, error) {
  358. out := DeleteObjectsResult{}
  359. dxml := deleteXML{}
  360. for _, key := range objectKeys {
  361. dxml.Objects = append(dxml.Objects, DeleteObject{Key: key})
  362. }
  363. isQuietStr, _ := findOption(options, deleteObjectsQuiet, "FALSE")
  364. isQuiet, _ := strconv.ParseBool(isQuietStr.(string))
  365. dxml.Quiet = isQuiet
  366. encode := "&encoding-type=url"
  367. bs, err := xml.Marshal(dxml)
  368. if err != nil {
  369. return out, err
  370. }
  371. buffer := new(bytes.Buffer)
  372. buffer.Write(bs)
  373. contentType := http.DetectContentType(buffer.Bytes())
  374. options = append(options, ContentType(contentType))
  375. sum := md5.Sum(bs)
  376. b64 := base64.StdEncoding.EncodeToString(sum[:])
  377. options = append(options, ContentMD5(b64))
  378. resp, err := bucket.do("POST", "", "delete"+encode, "delete", options, buffer, nil)
  379. if err != nil {
  380. return out, err
  381. }
  382. defer resp.Body.Close()
  383. if !dxml.Quiet {
  384. if err = xmlUnmarshal(resp.Body, &out); err == nil {
  385. err = decodeDeleteObjectsResult(&out)
  386. }
  387. }
  388. return out, err
  389. }
  390. //
  391. // IsObjectExist object是否存在。
  392. //
  393. // bool object是否存在,true存在,false不存在。error为nil时有效。
  394. //
  395. // error 操作无错误为nil,非nil为错误信息。
  396. //
  397. func (bucket Bucket) IsObjectExist(objectKey string) (bool, error) {
  398. listRes, err := bucket.ListObjects(Prefix(objectKey), MaxKeys(1))
  399. if err != nil {
  400. return false, err
  401. }
  402. if len(listRes.Objects) == 1 && listRes.Objects[0].Key == objectKey {
  403. return true, nil
  404. }
  405. return false, nil
  406. }
  407. //
  408. // ListObjects 获得Bucket下筛选后所有的object的列表。
  409. //
  410. // options ListObject的筛选行为。Prefix指定的前缀、MaxKeys最大数目、Marker第一个开始、Delimiter对Object名字进行分组的字符。
  411. //
  412. // 您有如下8个object,my-object-1, my-object-11, my-object-2, my-object-21,
  413. // my-object-22, my-object-3, my-object-31, my-object-32。如果您指定了Prefix为my-object-2,
  414. // 则返回my-object-2, my-object-21, my-object-22三个object。如果您指定了Marker为my-object-22,
  415. // 则返回my-object-3, my-object-31, my-object-32三个object。如果您指定MaxKeys则每次最多返回MaxKeys个,
  416. // 最后一次可能不足。这三个参数可以组合使用,实现分页等功能。如果把prefix设为某个文件夹名,就可以罗列以此prefix开头的文件,
  417. // 即该文件夹下递归的所有的文件和子文件夹。如果再把delimiter设置为"/"时,返回值就只罗列该文件夹下的文件,该文件夹下的子文件名
  418. // 返回在CommonPrefixes部分,子文件夹下递归的文件和文件夹不被显示。例如一个bucket存在三个object,fun/test.jpg、
  419. // fun/movie/001.avi、fun/movie/007.avi。若设定prefix为"fun/",则返回三个object;如果增加设定
  420. // delimiter为"/",则返回文件"fun/test.jpg"和前缀"fun/movie/",即实现了文件夹的逻辑。
  421. //
  422. // 常用场景,请参数示例sample/list_object.go。
  423. //
  424. // ListObjectsResponse 操作成功后的返回值,成员Objects为bucket中对象列表。error为nil时该返回值有效。
  425. //
  426. func (bucket Bucket) ListObjects(options ...Option) (ListObjectsResult, error) {
  427. var out ListObjectsResult
  428. options = append(options, EncodingType("url"))
  429. params, err := handleParams(options)
  430. if err != nil {
  431. return out, err
  432. }
  433. resp, err := bucket.do("GET", "", params, "", nil, nil, nil)
  434. if err != nil {
  435. return out, err
  436. }
  437. defer resp.Body.Close()
  438. err = xmlUnmarshal(resp.Body, &out)
  439. if err != nil {
  440. return out, err
  441. }
  442. err = decodeListObjectsResult(&out)
  443. return out, err
  444. }
  445. //
  446. // SetObjectMeta 设置Object的Meta。
  447. //
  448. // objectKey object
  449. // options 指定对象的属性,有以下可选项CacheControl、ContentDisposition、ContentEncoding、Expires、
  450. // ServerSideEncryption、Meta。
  451. //
  452. // error 操作无错误时error为nil,非nil为错误信息。
  453. //
  454. func (bucket Bucket) SetObjectMeta(objectKey string, options ...Option) error {
  455. options = append(options, MetadataDirective(MetaReplace))
  456. _, err := bucket.CopyObject(objectKey, objectKey, options...)
  457. return err
  458. }
  459. //
  460. // GetObjectDetailedMeta 查询Object的头信息。
  461. //
  462. // objectKey object名称。
  463. // objectPropertyConstraints 对象的属性限制项,满足时正常返回,不满足时返回错误。现在项有IfModifiedSince、IfUnmodifiedSince、
  464. // IfMatch、IfNoneMatch。具体含义请参看 https://help.aliyun.com/document_detail/oss/api-reference/object/HeadObject.html
  465. //
  466. // http.Header 对象的meta,error为nil时有效。
  467. // error 操作无错误为nil,非nil为错误信息。
  468. //
  469. func (bucket Bucket) GetObjectDetailedMeta(objectKey string, options ...Option) (http.Header, error) {
  470. resp, err := bucket.do("HEAD", objectKey, "", "", options, nil, nil)
  471. if err != nil {
  472. return nil, err
  473. }
  474. defer resp.Body.Close()
  475. return resp.Headers, nil
  476. }
  477. //
  478. // GetObjectMeta 查询Object的头信息。
  479. //
  480. // GetObjectMeta相比GetObjectDetailedMeta更轻量,仅返回指定Object的少量基本meta信息,
  481. // 包括该Object的ETag、Size(对象大小)、LastModified,其中Size由响应头Content-Length的数值表示。
  482. //
  483. // objectKey object名称。
  484. //
  485. // http.Header 对象的meta,error为nil时有效。
  486. // error 操作无错误为nil,非nil为错误信息。
  487. //
  488. func (bucket Bucket) GetObjectMeta(objectKey string) (http.Header, error) {
  489. resp, err := bucket.do("GET", objectKey, "?objectMeta", "", nil, nil, nil)
  490. if err != nil {
  491. return nil, err
  492. }
  493. defer resp.Body.Close()
  494. return resp.Headers, nil
  495. }
  496. //
  497. // SetObjectACL 修改Object的ACL权限。
  498. //
  499. // 只有Bucket Owner才有权限调用PutObjectACL来修改Object的ACL。Object ACL优先级高于Bucket ACL。
  500. // 例如Bucket ACL是private的,而Object ACL是public-read-write的,则访问这个Object时,
  501. // 先判断Object的ACL,所以所有用户都拥有这个Object的访问权限,即使这个Bucket是private bucket。
  502. // 如果某个Object从来没设置过ACL,则访问权限遵循Bucket ACL。
  503. //
  504. // Object的读操作包括GetObject,HeadObject,CopyObject和UploadPartCopy中的对source object的读;
  505. // Object的写操作包括:PutObject,PostObject,AppendObject,DeleteObject,
  506. // DeleteMultipleObjects,CompleteMultipartUpload以及CopyObject对新的Object的写。
  507. //
  508. // objectKey 设置权限的object。
  509. // objectAcl 对象权限。可选值PrivateACL(私有读写)、PublicReadACL(公共读私有写)、PublicReadWriteACL(公共读写)。
  510. //
  511. // error 操作无错误为nil,非nil为错误信息。
  512. //
  513. func (bucket Bucket) SetObjectACL(objectKey string, objectACL ACLType) error {
  514. options := []Option{ObjectACL(objectACL)}
  515. resp, err := bucket.do("PUT", objectKey, "acl", "acl", options, nil, nil)
  516. if err != nil {
  517. return err
  518. }
  519. defer resp.Body.Close()
  520. return checkRespCode(resp.StatusCode, []int{http.StatusOK})
  521. }
  522. //
  523. // GetObjectACL 获取对象的ACL权限。
  524. //
  525. // objectKey 获取权限的object。
  526. //
  527. // GetObjectAclResponse 获取权限操作返回值,error为nil时有效。GetObjectAclResponse.Acl为对象的权限。
  528. // error 操作无错误为nil,非nil为错误信息。
  529. //
  530. func (bucket Bucket) GetObjectACL(objectKey string) (GetObjectACLResult, error) {
  531. var out GetObjectACLResult
  532. resp, err := bucket.do("GET", objectKey, "acl", "acl", nil, nil, nil)
  533. if err != nil {
  534. return out, err
  535. }
  536. defer resp.Body.Close()
  537. err = xmlUnmarshal(resp.Body, &out)
  538. return out, err
  539. }
  540. // Private
  541. func (bucket Bucket) do(method, objectName, urlParams, subResource string, options []Option,
  542. data io.Reader, listener ProgressListener) (*Response, error) {
  543. headers := make(map[string]string)
  544. err := handleOptions(headers, options)
  545. if err != nil {
  546. return nil, err
  547. }
  548. return bucket.Client.Conn.Do(method, bucket.BucketName, objectName,
  549. urlParams, subResource, headers, data, 0, listener)
  550. }
  551. func (bucket Bucket) getConfig() *Config {
  552. return bucket.Client.Config
  553. }
  554. func addContentType(options []Option, keys ...string) []Option {
  555. typ := TypeByExtension("")
  556. for _, key := range keys {
  557. typ = TypeByExtension(key)
  558. if typ != "" {
  559. break
  560. }
  561. }
  562. if typ == "" {
  563. typ = "application/octet-stream"
  564. }
  565. opts := []Option{ContentType(typ)}
  566. opts = append(opts, options...)
  567. return opts
  568. }