api-put-object-multipart.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. /*
  2. * Minio Go Library for Amazon S3 Compatible Cloud Storage
  3. * Copyright 2015-2017 Minio, Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. */
  17. package minio
  18. import (
  19. "bytes"
  20. "context"
  21. "encoding/base64"
  22. "encoding/hex"
  23. "encoding/xml"
  24. "fmt"
  25. "io"
  26. "io/ioutil"
  27. "net/http"
  28. "net/url"
  29. "runtime/debug"
  30. "sort"
  31. "strconv"
  32. "strings"
  33. "github.com/minio/minio-go/pkg/s3utils"
  34. )
  35. func (c Client) putObjectMultipart(ctx context.Context, bucketName, objectName string, reader io.Reader, size int64,
  36. opts PutObjectOptions) (n int64, err error) {
  37. n, err = c.putObjectMultipartNoStream(ctx, bucketName, objectName, reader, opts)
  38. if err != nil {
  39. errResp := ToErrorResponse(err)
  40. // Verify if multipart functionality is not available, if not
  41. // fall back to single PutObject operation.
  42. if errResp.Code == "AccessDenied" && strings.Contains(errResp.Message, "Access Denied") {
  43. // Verify if size of reader is greater than '5GiB'.
  44. if size > maxSinglePutObjectSize {
  45. return 0, ErrEntityTooLarge(size, maxSinglePutObjectSize, bucketName, objectName)
  46. }
  47. // Fall back to uploading as single PutObject operation.
  48. return c.putObjectNoChecksum(ctx, bucketName, objectName, reader, size, opts)
  49. }
  50. }
  51. return n, err
  52. }
  53. func (c Client) putObjectMultipartNoStream(ctx context.Context, bucketName, objectName string, reader io.Reader, opts PutObjectOptions) (n int64, err error) {
  54. // Input validation.
  55. if err = s3utils.CheckValidBucketName(bucketName); err != nil {
  56. return 0, err
  57. }
  58. if err = s3utils.CheckValidObjectName(objectName); err != nil {
  59. return 0, err
  60. }
  61. // Total data read and written to server. should be equal to
  62. // 'size' at the end of the call.
  63. var totalUploadedSize int64
  64. // Complete multipart upload.
  65. var complMultipartUpload completeMultipartUpload
  66. // Calculate the optimal parts info for a given size.
  67. totalPartsCount, partSize, _, err := optimalPartInfo(-1)
  68. if err != nil {
  69. return 0, err
  70. }
  71. // Initiate a new multipart upload.
  72. uploadID, err := c.newUploadID(ctx, bucketName, objectName, opts)
  73. if err != nil {
  74. return 0, err
  75. }
  76. defer func() {
  77. if err != nil {
  78. c.abortMultipartUpload(ctx, bucketName, objectName, uploadID)
  79. }
  80. }()
  81. // Part number always starts with '1'.
  82. partNumber := 1
  83. // Initialize parts uploaded map.
  84. partsInfo := make(map[int]ObjectPart)
  85. // Create a buffer.
  86. buf := make([]byte, partSize)
  87. defer debug.FreeOSMemory()
  88. for partNumber <= totalPartsCount {
  89. // Choose hash algorithms to be calculated by hashCopyN,
  90. // avoid sha256 with non-v4 signature request or
  91. // HTTPS connection.
  92. hashAlgos, hashSums := c.hashMaterials()
  93. length, rErr := io.ReadFull(reader, buf)
  94. if rErr == io.EOF {
  95. break
  96. }
  97. if rErr != nil && rErr != io.ErrUnexpectedEOF {
  98. return 0, rErr
  99. }
  100. // Calculates hash sums while copying partSize bytes into cw.
  101. for k, v := range hashAlgos {
  102. v.Write(buf[:length])
  103. hashSums[k] = v.Sum(nil)
  104. }
  105. // Update progress reader appropriately to the latest offset
  106. // as we read from the source.
  107. rd := newHook(bytes.NewReader(buf[:length]), opts.Progress)
  108. // Checksums..
  109. var (
  110. md5Base64 string
  111. sha256Hex string
  112. )
  113. if hashSums["md5"] != nil {
  114. md5Base64 = base64.StdEncoding.EncodeToString(hashSums["md5"])
  115. }
  116. if hashSums["sha256"] != nil {
  117. sha256Hex = hex.EncodeToString(hashSums["sha256"])
  118. }
  119. // Proceed to upload the part.
  120. var objPart ObjectPart
  121. objPart, err = c.uploadPart(ctx, bucketName, objectName, uploadID, rd, partNumber,
  122. md5Base64, sha256Hex, int64(length), opts.UserMetadata)
  123. if err != nil {
  124. return totalUploadedSize, err
  125. }
  126. // Save successfully uploaded part metadata.
  127. partsInfo[partNumber] = objPart
  128. // Save successfully uploaded size.
  129. totalUploadedSize += int64(length)
  130. // Increment part number.
  131. partNumber++
  132. // For unknown size, Read EOF we break away.
  133. // We do not have to upload till totalPartsCount.
  134. if rErr == io.EOF {
  135. break
  136. }
  137. }
  138. // Loop over total uploaded parts to save them in
  139. // Parts array before completing the multipart request.
  140. for i := 1; i < partNumber; i++ {
  141. part, ok := partsInfo[i]
  142. if !ok {
  143. return 0, ErrInvalidArgument(fmt.Sprintf("Missing part number %d", i))
  144. }
  145. complMultipartUpload.Parts = append(complMultipartUpload.Parts, CompletePart{
  146. ETag: part.ETag,
  147. PartNumber: part.PartNumber,
  148. })
  149. }
  150. // Sort all completed parts.
  151. sort.Sort(completedParts(complMultipartUpload.Parts))
  152. if _, err = c.completeMultipartUpload(ctx, bucketName, objectName, uploadID, complMultipartUpload); err != nil {
  153. return totalUploadedSize, err
  154. }
  155. // Return final size.
  156. return totalUploadedSize, nil
  157. }
  158. // initiateMultipartUpload - Initiates a multipart upload and returns an upload ID.
  159. func (c Client) initiateMultipartUpload(ctx context.Context, bucketName, objectName string, opts PutObjectOptions) (initiateMultipartUploadResult, error) {
  160. // Input validation.
  161. if err := s3utils.CheckValidBucketName(bucketName); err != nil {
  162. return initiateMultipartUploadResult{}, err
  163. }
  164. if err := s3utils.CheckValidObjectName(objectName); err != nil {
  165. return initiateMultipartUploadResult{}, err
  166. }
  167. // Initialize url queries.
  168. urlValues := make(url.Values)
  169. urlValues.Set("uploads", "")
  170. // Set ContentType header.
  171. customHeader := opts.Header()
  172. reqMetadata := requestMetadata{
  173. bucketName: bucketName,
  174. objectName: objectName,
  175. queryValues: urlValues,
  176. customHeader: customHeader,
  177. }
  178. // Execute POST on an objectName to initiate multipart upload.
  179. resp, err := c.executeMethod(ctx, "POST", reqMetadata)
  180. defer closeResponse(resp)
  181. if err != nil {
  182. return initiateMultipartUploadResult{}, err
  183. }
  184. if resp != nil {
  185. if resp.StatusCode != http.StatusOK {
  186. return initiateMultipartUploadResult{}, httpRespToErrorResponse(resp, bucketName, objectName)
  187. }
  188. }
  189. // Decode xml for new multipart upload.
  190. initiateMultipartUploadResult := initiateMultipartUploadResult{}
  191. err = xmlDecoder(resp.Body, &initiateMultipartUploadResult)
  192. if err != nil {
  193. return initiateMultipartUploadResult, err
  194. }
  195. return initiateMultipartUploadResult, nil
  196. }
  197. const serverEncryptionKeyPrefix = "x-amz-server-side-encryption"
  198. // uploadPart - Uploads a part in a multipart upload.
  199. func (c Client) uploadPart(ctx context.Context, bucketName, objectName, uploadID string, reader io.Reader,
  200. partNumber int, md5Base64, sha256Hex string, size int64, metadata map[string]string) (ObjectPart, error) {
  201. // Input validation.
  202. if err := s3utils.CheckValidBucketName(bucketName); err != nil {
  203. return ObjectPart{}, err
  204. }
  205. if err := s3utils.CheckValidObjectName(objectName); err != nil {
  206. return ObjectPart{}, err
  207. }
  208. if size > maxPartSize {
  209. return ObjectPart{}, ErrEntityTooLarge(size, maxPartSize, bucketName, objectName)
  210. }
  211. if size <= -1 {
  212. return ObjectPart{}, ErrEntityTooSmall(size, bucketName, objectName)
  213. }
  214. if partNumber <= 0 {
  215. return ObjectPart{}, ErrInvalidArgument("Part number cannot be negative or equal to zero.")
  216. }
  217. if uploadID == "" {
  218. return ObjectPart{}, ErrInvalidArgument("UploadID cannot be empty.")
  219. }
  220. // Get resources properly escaped and lined up before using them in http request.
  221. urlValues := make(url.Values)
  222. // Set part number.
  223. urlValues.Set("partNumber", strconv.Itoa(partNumber))
  224. // Set upload id.
  225. urlValues.Set("uploadId", uploadID)
  226. // Set encryption headers, if any.
  227. customHeader := make(http.Header)
  228. for k, v := range metadata {
  229. if len(v) > 0 {
  230. if strings.HasPrefix(strings.ToLower(k), serverEncryptionKeyPrefix) {
  231. customHeader.Set(k, v)
  232. }
  233. }
  234. }
  235. reqMetadata := requestMetadata{
  236. bucketName: bucketName,
  237. objectName: objectName,
  238. queryValues: urlValues,
  239. customHeader: customHeader,
  240. contentBody: reader,
  241. contentLength: size,
  242. contentMD5Base64: md5Base64,
  243. contentSHA256Hex: sha256Hex,
  244. }
  245. // Execute PUT on each part.
  246. resp, err := c.executeMethod(ctx, "PUT", reqMetadata)
  247. defer closeResponse(resp)
  248. if err != nil {
  249. return ObjectPart{}, err
  250. }
  251. if resp != nil {
  252. if resp.StatusCode != http.StatusOK {
  253. return ObjectPart{}, httpRespToErrorResponse(resp, bucketName, objectName)
  254. }
  255. }
  256. // Once successfully uploaded, return completed part.
  257. objPart := ObjectPart{}
  258. objPart.Size = size
  259. objPart.PartNumber = partNumber
  260. // Trim off the odd double quotes from ETag in the beginning and end.
  261. objPart.ETag = strings.TrimPrefix(resp.Header.Get("ETag"), "\"")
  262. objPart.ETag = strings.TrimSuffix(objPart.ETag, "\"")
  263. return objPart, nil
  264. }
  265. // completeMultipartUpload - Completes a multipart upload by assembling previously uploaded parts.
  266. func (c Client) completeMultipartUpload(ctx context.Context, bucketName, objectName, uploadID string,
  267. complete completeMultipartUpload) (completeMultipartUploadResult, error) {
  268. // Input validation.
  269. if err := s3utils.CheckValidBucketName(bucketName); err != nil {
  270. return completeMultipartUploadResult{}, err
  271. }
  272. if err := s3utils.CheckValidObjectName(objectName); err != nil {
  273. return completeMultipartUploadResult{}, err
  274. }
  275. // Initialize url queries.
  276. urlValues := make(url.Values)
  277. urlValues.Set("uploadId", uploadID)
  278. // Marshal complete multipart body.
  279. completeMultipartUploadBytes, err := xml.Marshal(complete)
  280. if err != nil {
  281. return completeMultipartUploadResult{}, err
  282. }
  283. // Instantiate all the complete multipart buffer.
  284. completeMultipartUploadBuffer := bytes.NewReader(completeMultipartUploadBytes)
  285. reqMetadata := requestMetadata{
  286. bucketName: bucketName,
  287. objectName: objectName,
  288. queryValues: urlValues,
  289. contentBody: completeMultipartUploadBuffer,
  290. contentLength: int64(len(completeMultipartUploadBytes)),
  291. contentSHA256Hex: sum256Hex(completeMultipartUploadBytes),
  292. }
  293. // Execute POST to complete multipart upload for an objectName.
  294. resp, err := c.executeMethod(ctx, "POST", reqMetadata)
  295. defer closeResponse(resp)
  296. if err != nil {
  297. return completeMultipartUploadResult{}, err
  298. }
  299. if resp != nil {
  300. if resp.StatusCode != http.StatusOK {
  301. return completeMultipartUploadResult{}, httpRespToErrorResponse(resp, bucketName, objectName)
  302. }
  303. }
  304. // Read resp.Body into a []bytes to parse for Error response inside the body
  305. var b []byte
  306. b, err = ioutil.ReadAll(resp.Body)
  307. if err != nil {
  308. return completeMultipartUploadResult{}, err
  309. }
  310. // Decode completed multipart upload response on success.
  311. completeMultipartUploadResult := completeMultipartUploadResult{}
  312. err = xmlDecoder(bytes.NewReader(b), &completeMultipartUploadResult)
  313. if err != nil {
  314. // xml parsing failure due to presence an ill-formed xml fragment
  315. return completeMultipartUploadResult, err
  316. } else if completeMultipartUploadResult.Bucket == "" {
  317. // xml's Decode method ignores well-formed xml that don't apply to the type of value supplied.
  318. // In this case, it would leave completeMultipartUploadResult with the corresponding zero-values
  319. // of the members.
  320. // Decode completed multipart upload response on failure
  321. completeMultipartUploadErr := ErrorResponse{}
  322. err = xmlDecoder(bytes.NewReader(b), &completeMultipartUploadErr)
  323. if err != nil {
  324. // xml parsing failure due to presence an ill-formed xml fragment
  325. return completeMultipartUploadResult, err
  326. }
  327. return completeMultipartUploadResult, completeMultipartUploadErr
  328. }
  329. return completeMultipartUploadResult, nil
  330. }