bucket.go 15 KB

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