bucket.go 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. package oss
  2. import (
  3. "bytes"
  4. "encoding/xml"
  5. "io"
  6. "net/http"
  7. "os"
  8. "strconv"
  9. )
  10. // Bucket implements the operations of object.
  11. type Bucket struct {
  12. Client Client
  13. BucketName string
  14. }
  15. //
  16. // PutObject 新建Object,如果Object已存在,覆盖原有Object。
  17. //
  18. // objectKey 上传对象的名称,使用UTF-8编码、长度必须在1-1023字节之间、不能以“/”或者“\”字符开头。
  19. // reader io.Reader读取object的数据。
  20. // options 上传对象时可以指定对象的属性,可用选项有CacheControl、ContentDisposition、ContentEncoding、
  21. // Expires、ServerSideEncryption、ObjectACL、Meta,具体含义请参看
  22. // https://help.aliyun.com/document_detail/oss/api-reference/object/PutObject.html
  23. //
  24. // error 操作无错误为nil,非nil为错误信息。
  25. //
  26. func (bucket Bucket) PutObject(objectKey string, reader io.Reader, options ...Option) error {
  27. opts := addContentType(options, objectKey)
  28. resp, err := bucket.do("PUT", objectKey, "", "", opts, reader)
  29. if err != nil {
  30. return err
  31. }
  32. defer resp.body.Close()
  33. return checkRespCode(resp.statusCode, []int{http.StatusOK})
  34. }
  35. //
  36. // PutObjectFromFile 新建Object,内容从本地文件中读取。
  37. //
  38. // objectKey 上传对象的名称。
  39. // filePath 本地文件,上传对象的值为该文件内容。
  40. // options 上传对象时可以指定对象的属性。详见PutObject的options。
  41. //
  42. // error 操作无错误为nil,非nil为错误信息。
  43. //
  44. func (bucket Bucket) PutObjectFromFile(objectKey, filePath string, options ...Option) error {
  45. fd, err := os.Open(filePath)
  46. if err != nil {
  47. return err
  48. }
  49. defer fd.Close()
  50. opts := addContentType(options, filePath, objectKey)
  51. resp, err := bucket.do("PUT", objectKey, "", "", opts, fd)
  52. if err != nil {
  53. return err
  54. }
  55. defer resp.body.Close()
  56. return checkRespCode(resp.statusCode, []int{http.StatusOK})
  57. }
  58. //
  59. // GetObject 下载文件。
  60. //
  61. // objectKey 下载的文件名称。
  62. // options 对象的属性限制项,可选值有Range、IfModifiedSince、IfUnmodifiedSince、IfMatch、
  63. // IfNoneMatch、AcceptEncoding,详细请参考
  64. // https://help.aliyun.com/document_detail/oss/api-reference/object/GetObject.html
  65. //
  66. // io.ReadCloser reader,读取数据后需要close。error为nil时有效。
  67. // error 操作无错误为nil,非nil为错误信息。
  68. //
  69. func (bucket Bucket) GetObject(objectKey string, options ...Option) (io.ReadCloser, error) {
  70. resp, err := bucket.do("GET", objectKey, "", "", options, nil)
  71. if err != nil {
  72. return nil, err
  73. }
  74. return resp.body, nil
  75. }
  76. //
  77. // GetObjectToFile 下载文件。
  78. //
  79. // objectKey 下载的文件名称。
  80. // filePath 下载对象的内容写到该本地文件。
  81. // options 对象的属性限制项。详见GetObject的options。
  82. //
  83. // error 操作无错误时返回error为nil,非nil为错误说明。
  84. //
  85. func (bucket Bucket) GetObjectToFile(objectKey, filePath string, options ...Option) error {
  86. resp, err := bucket.do("GET", objectKey, "", "", options, nil)
  87. if err != nil {
  88. return err
  89. }
  90. defer resp.body.Close()
  91. fd, err := os.OpenFile(filePath, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0660)
  92. if err != nil {
  93. return err
  94. }
  95. defer fd.Close()
  96. _, err = io.Copy(fd, resp.body)
  97. if err != nil {
  98. return err
  99. }
  100. return nil
  101. }
  102. //
  103. // CopyObject 同一个bucket内拷贝Object。
  104. //
  105. // srcObjectKey Copy的源对象。
  106. // destObjectKey Copy的目标对象。
  107. // options Copy对象时,您可以指定源对象的限制条件,满足限制条件时copy,不满足时返回错误,您可以选择如下选项CopySourceIfMatch、
  108. // CopySourceIfNoneMatch、CopySourceIfModifiedSince、CopySourceIfUnmodifiedSince、MetadataDirective。
  109. // Copy对象时,您可以指定目标对象的属性,如CacheControl、ContentDisposition、ContentEncoding、Expires、
  110. // ServerSideEncryption、ObjectACL、Meta,选项的含义请参看
  111. // https://help.aliyun.com/document_detail/oss/api-reference/object/CopyObject.html
  112. //
  113. // error 操作无错误为nil,非nil为错误信息。
  114. //
  115. func (bucket Bucket) CopyObject(srcObjectKey, destObjectKey string, options ...Option) (CopyObjectResult, error) {
  116. var out CopyObjectResult
  117. options = append(options, CopySource(bucket.BucketName, srcObjectKey))
  118. resp, err := bucket.do("PUT", destObjectKey, "", "", options, nil)
  119. if err != nil {
  120. return out, err
  121. }
  122. defer resp.body.Close()
  123. err = xmlUnmarshal(resp.body, &out)
  124. return out, err
  125. }
  126. //
  127. // CopyObjectToBucket bucket间拷贝object。
  128. //
  129. // srcObjectKey Copy的源对象。
  130. // destBucket Copy的目标Bucket。
  131. // destObjectKey Copy的目标Object。
  132. // options Copy选项,详见CopyObject的options。
  133. //
  134. // error 操作无错误为nil,非nil为错误信息。
  135. //
  136. func (bucket Bucket) CopyObjectToBucket(srcObjectKey, destBucketName, destObjectKey string, options ...Option) (CopyObjectResult, error) {
  137. var out CopyObjectResult
  138. options = append(options, CopySource(bucket.BucketName, srcObjectKey))
  139. headers := make(map[string]string)
  140. err := handleOptions(headers, options)
  141. if err != nil {
  142. return out, err
  143. }
  144. resp, err := bucket.Client.Conn.Do("PUT", destBucketName, destObjectKey, "", "", headers, nil)
  145. if err != nil {
  146. return out, err
  147. }
  148. defer resp.body.Close()
  149. err = xmlUnmarshal(resp.body, &out)
  150. return out, err
  151. }
  152. //
  153. // AppendObject 追加方式上传。
  154. //
  155. // AppendObject参数必须包含position,其值指定从何处进行追加。首次追加操作的position必须为0,
  156. // 后续追加操作的position是Object的当前长度。例如,第一次Append Object请求指定position值为0,
  157. // content-length是65536;那么,第二次Append Object需要指定position为65536。
  158. // 每次操作成功后,响应头部x-oss-next-append-position也会标明下一次追加的position。
  159. //
  160. // objectKey 需要追加的Object。
  161. // reader io.Reader,读取追的内容。
  162. // appendPosition object追加的起始位置。
  163. // destObjectProperties 第一次追加时指定新对象的属性,如CacheControl、ContentDisposition、ContentEncoding、
  164. // Expires、ServerSideEncryption、ObjectACL。
  165. // int64 下次追加的开始位置,error为nil空时有效。
  166. //
  167. // error 操作无错误为nil,非nil为错误信息。
  168. //
  169. func (bucket Bucket) AppendObject(objectKey string, reader io.Reader, appendPosition int64, options ...Option) (int64, error) {
  170. var nextAppendPosition int64
  171. params := "append&position=" + strconv.Itoa(int(appendPosition))
  172. opts := addContentType(options, objectKey)
  173. resp, err := bucket.do("POST", objectKey, params, params, opts, reader)
  174. if err != nil {
  175. return nextAppendPosition, err
  176. }
  177. defer resp.body.Close()
  178. napint, err := strconv.Atoi(resp.headers.Get(HTTPHeaderOssNextAppendPosition))
  179. if err != nil {
  180. return nextAppendPosition, err
  181. }
  182. nextAppendPosition = int64(napint)
  183. return nextAppendPosition, nil
  184. }
  185. //
  186. // DeleteObject 删除Object。
  187. //
  188. // objectKey 待删除Object。
  189. //
  190. // error 操作无错误为nil,非nil为错误信息。
  191. //
  192. func (bucket Bucket) DeleteObject(objectKey string) error {
  193. resp, err := bucket.do("DELETE", objectKey, "", "", nil, nil)
  194. if err != nil {
  195. return err
  196. }
  197. defer resp.body.Close()
  198. return checkRespCode(resp.statusCode, []int{http.StatusNoContent})
  199. }
  200. //
  201. // DeleteObjects 批量删除object。
  202. //
  203. // objectKeys 待删除object类表。
  204. // options 删除选项,DeleteObjectsQuiet,是否是安静模式,默认不使用。
  205. //
  206. // DeleteObjectsResult 非安静模式的的返回值。
  207. // error 操作无错误为nil,非nil为错误信息。
  208. //
  209. func (bucket Bucket) DeleteObjects(objectKeys []string, options ...Option) (DeleteObjectsResult, error) {
  210. out := DeleteObjectsResult{}
  211. dxml := deleteXML{}
  212. for _, key := range objectKeys {
  213. dxml.Objects = append(dxml.Objects, DeleteObject{Key: key})
  214. }
  215. isQuietStr, _ := findOption(options, deleteObjectsQuiet, "FALSE")
  216. isQuiet, _ := strconv.ParseBool(isQuietStr)
  217. dxml.Quiet = isQuiet
  218. encode := "&encoding-type=url"
  219. bs, err := xml.Marshal(dxml)
  220. if err != nil {
  221. return out, err
  222. }
  223. buffer := new(bytes.Buffer)
  224. buffer.Write(bs)
  225. contentType := http.DetectContentType(buffer.Bytes())
  226. options = append(options, ContentType(contentType))
  227. resp, err := bucket.do("POST", "", "delete"+encode, "delete", options, buffer)
  228. if err != nil {
  229. return out, err
  230. }
  231. defer resp.body.Close()
  232. if !dxml.Quiet {
  233. if err = xmlUnmarshal(resp.body, &out); err == nil {
  234. err = decodeDeleteObjectsResult(&out)
  235. }
  236. }
  237. return out, err
  238. }
  239. //
  240. // IsObjectExist object是否存在。
  241. //
  242. // bool object是否存在,true存在,false不存在。error为nil时有效。
  243. //
  244. // error 操作无错误为nil,非nil为错误信息。
  245. //
  246. func (bucket Bucket) IsObjectExist(objectKey string) (bool, error) {
  247. listRes, err := bucket.ListObjects(Prefix(objectKey), MaxKeys(1))
  248. if err != nil {
  249. return false, err
  250. }
  251. if len(listRes.Objects) == 1 && listRes.Objects[0].Key == objectKey {
  252. return true, nil
  253. }
  254. return false, nil
  255. }
  256. //
  257. // ListObjects 获得Bucket下筛选后所有的object的列表。
  258. //
  259. // options ListObject的筛选行为。Prefix指定的前缀、MaxKeys最大数目、Marker第一个开始、Delimiter对Object名字进行分组的字符。
  260. //
  261. // 您有如下8个object,my-object-1, my-object-11, my-object-2, my-object-21,
  262. // my-object-22, my-object-3, my-object-31, my-object-32。如果您指定了Prefix为my-object-2,
  263. // 则返回my-object-2, my-object-21, my-object-22三个object。如果您指定了Marker为my-object-22,
  264. // 则返回my-object-3, my-object-31, my-object-32三个object。如果您指定MaxKeys则每次最多返回MaxKeys个,
  265. // 最后一次可能不足。这三个参数可以组合使用,实现分页等功能。如果把prefix设为某个文件夹名,就可以罗列以此prefix开头的文件,
  266. // 即该文件夹下递归的所有的文件和子文件夹。如果再把delimiter设置为"/"时,返回值就只罗列该文件夹下的文件,该文件夹下的子文件名
  267. // 返回在CommonPrefixes部分,子文件夹下递归的文件和文件夹不被显示。例如一个bucket存在三个object,fun/test.jpg、
  268. // fun/movie/001.avi、fun/movie/007.avi。若设定prefix为"fun/",则返回三个object;如果增加设定
  269. // delimiter为"/",则返回文件"fun/test.jpg"和前缀"fun/movie/",即实现了文件夹的逻辑。
  270. //
  271. // 常用场景,请参数示例sample/list_object.go。
  272. //
  273. // ListObjectsResponse 操作成功后的返回值,成员Objects为bucket中对象列表。error为nil时该返回值有效。
  274. //
  275. func (bucket Bucket) ListObjects(options ...Option) (ListObjectsResult, error) {
  276. var out ListObjectsResult
  277. options = append(options, EncodingType("url"))
  278. params, err := handleParams(options)
  279. if err != nil {
  280. return out, err
  281. }
  282. resp, err := bucket.do("GET", "", params, "", nil, nil)
  283. if err != nil {
  284. return out, err
  285. }
  286. defer resp.body.Close()
  287. err = xmlUnmarshal(resp.body, &out)
  288. if err != nil {
  289. return out, err
  290. }
  291. err = decodeListObjectsResult(&out)
  292. return out, err
  293. }
  294. //
  295. // SetObjectMeta 设置Object的Meta。
  296. //
  297. // objectKey object
  298. // options 指定对象的属性,有以下可选项CacheControl、ContentDisposition、ContentEncoding、Expires、
  299. // ServerSideEncryption、Meta。
  300. //
  301. // error 操作无错误时error为nil,非nil为错误信息。
  302. //
  303. func (bucket Bucket) SetObjectMeta(objectKey string, options ...Option) error {
  304. options = append(options, MetadataDirective(MetaReplace))
  305. _, err := bucket.CopyObject(objectKey, objectKey, options...)
  306. return err
  307. }
  308. //
  309. // GetObjectDetailedMeta 查询Object的头信息。
  310. //
  311. // objectKey object名称。
  312. // objectPropertyConstraints 对象的属性限制项,满足时正常返回,不满足时返回错误。现在项有IfModifiedSince、IfUnmodifiedSince、
  313. // IfMatch、IfNoneMatch。具体含义请参看 https://help.aliyun.com/document_detail/oss/api-reference/object/HeadObject.html
  314. //
  315. // http.Header 对象的meta,error为nil时有效。
  316. // error 操作无错误为nil,非nil为错误信息。
  317. //
  318. func (bucket Bucket) GetObjectDetailedMeta(objectKey string, options ...Option) (http.Header, error) {
  319. resp, err := bucket.do("HEAD", objectKey, "", "", options, nil)
  320. if err != nil {
  321. return nil, err
  322. }
  323. defer resp.body.Close()
  324. return resp.headers, nil
  325. }
  326. //
  327. // GetObjectMeta 查询Object的头信息。
  328. //
  329. // GetObjectMeta相比GetObjectDetailedMeta更轻量,仅返回指定Object的少量基本meta信息,
  330. // 包括该Object的ETag、Size(对象大小)、LastModified,其中Size由响应头Content-Length的数值表示。
  331. //
  332. // objectKey object名称。
  333. //
  334. // http.Header 对象的meta,error为nil时有效。
  335. // error 操作无错误为nil,非nil为错误信息。
  336. //
  337. func (bucket Bucket) GetObjectMeta(objectKey string) (http.Header, error) {
  338. resp, err := bucket.do("GET", objectKey, "?objectMeta", "", nil, nil)
  339. if err != nil {
  340. return nil, err
  341. }
  342. defer resp.body.Close()
  343. return resp.headers, nil
  344. }
  345. //
  346. // SetObjectACL 修改Object的ACL权限。
  347. //
  348. // 只有Bucket Owner才有权限调用PutObjectACL来修改Object的ACL。Object ACL优先级高于Bucket ACL。
  349. // 例如Bucket ACL是private的,而Object ACL是public-read-write的,则访问这个Object时,
  350. // 先判断Object的ACL,所以所有用户都拥有这个Object的访问权限,即使这个Bucket是private bucket。
  351. // 如果某个Object从来没设置过ACL,则访问权限遵循Bucket ACL。
  352. //
  353. // Object的读操作包括GetObject,HeadObject,CopyObject和UploadPartCopy中的对source object的读;
  354. // Object的写操作包括:PutObject,PostObject,AppendObject,DeleteObject,
  355. // DeleteMultipleObjects,CompleteMultipartUpload以及CopyObject对新的Object的写。
  356. //
  357. // objectKey 设置权限的object。
  358. // objectAcl 对象权限。可选值PrivateACL(私有读写)、PublicReadACL(公共读私有写)、PublicReadWriteACL(公共读写)。
  359. //
  360. // error 操作无错误为nil,非nil为错误信息。
  361. //
  362. func (bucket Bucket) SetObjectACL(objectKey string, objectACL ACLType) error {
  363. options := []Option{ObjectACL(objectACL)}
  364. resp, err := bucket.do("PUT", objectKey, "", "", options, nil)
  365. if err != nil {
  366. return err
  367. }
  368. defer resp.body.Close()
  369. return checkRespCode(resp.statusCode, []int{http.StatusOK})
  370. }
  371. //
  372. // GetObjectACL 获取对象的ACL权限。
  373. //
  374. // objectKey 获取权限的object。
  375. //
  376. // GetObjectAclResponse 获取权限操作返回值,error为nil时有效。GetObjectAclResponse.Acl为对象的权限。
  377. // error 操作无错误为nil,非nil为错误信息。
  378. //
  379. func (bucket Bucket) GetObjectACL(objectKey string) (GetObjectACLResult, error) {
  380. var out GetObjectACLResult
  381. resp, err := bucket.do("GET", objectKey, "acl", "acl", nil, nil)
  382. if err != nil {
  383. return out, err
  384. }
  385. defer resp.body.Close()
  386. err = xmlUnmarshal(resp.body, &out)
  387. return out, err
  388. }
  389. // Private
  390. func (bucket Bucket) do(method, objectName, urlParams, subResource string,
  391. options []Option, data io.Reader) (*Response, error) {
  392. headers := make(map[string]string)
  393. err := handleOptions(headers, options)
  394. if err != nil {
  395. return nil, err
  396. }
  397. return bucket.Client.Conn.Do(method, bucket.BucketName, objectName,
  398. urlParams, subResource, headers, data)
  399. }
  400. func (bucket Bucket) getConfig() *Config {
  401. return bucket.Client.Config
  402. }
  403. // Private
  404. func addContentType(options []Option, keys ...string) []Option {
  405. typ := TypeByExtension("")
  406. for _, key := range keys {
  407. typ = TypeByExtension(key)
  408. if typ != "" {
  409. break
  410. }
  411. }
  412. if typ == "" {
  413. typ = "application/octet-stream"
  414. }
  415. opts := []Option{ContentType(typ)}
  416. opts = append(opts, options...)
  417. return opts
  418. }