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.

355 lines
10 KiB

3 years ago
6 years ago
4 years ago
4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
4 years ago
3 years ago
4 years ago
3 years ago
3 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
3 years ago
3 years ago
4 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
4 years ago
4 years ago
3 years ago
3 years ago
3 years ago
4 years ago
3 years ago
3 years ago
2 years ago
  1. package s3api
  2. import (
  3. "crypto/sha1"
  4. "encoding/xml"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "net/url"
  9. "strconv"
  10. "strings"
  11. "github.com/google/uuid"
  12. "github.com/seaweedfs/seaweedfs/weed/glog"
  13. "github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
  14. "github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
  15. weed_server "github.com/seaweedfs/seaweedfs/weed/server"
  16. stats_collect "github.com/seaweedfs/seaweedfs/weed/stats"
  17. "github.com/aws/aws-sdk-go/aws"
  18. "github.com/aws/aws-sdk-go/service/s3"
  19. )
  20. const (
  21. maxObjectListSizeLimit = 10000 // Limit number of objects in a listObjectsResponse.
  22. maxUploadsList = 10000 // Limit number of uploads in a listUploadsResponse.
  23. maxPartsList = 10000 // Limit number of parts in a listPartsResponse.
  24. globalMaxPartID = 100000
  25. )
  26. // NewMultipartUploadHandler - New multipart upload.
  27. func (s3a *S3ApiServer) NewMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
  28. bucket, object := s3_constants.GetBucketAndObject(r)
  29. createMultipartUploadInput := &s3.CreateMultipartUploadInput{
  30. Bucket: aws.String(bucket),
  31. Key: objectKey(aws.String(object)),
  32. Metadata: make(map[string]*string),
  33. }
  34. metadata := weed_server.SaveAmzMetaData(r, nil, false)
  35. for k, v := range metadata {
  36. createMultipartUploadInput.Metadata[k] = aws.String(string(v))
  37. }
  38. contentType := r.Header.Get("Content-Type")
  39. if contentType != "" {
  40. createMultipartUploadInput.ContentType = &contentType
  41. }
  42. response, errCode := s3a.createMultipartUpload(createMultipartUploadInput)
  43. glog.V(2).Info("NewMultipartUploadHandler", string(s3err.EncodeXMLResponse(response)), errCode)
  44. if errCode != s3err.ErrNone {
  45. s3err.WriteErrorResponse(w, r, errCode)
  46. return
  47. }
  48. writeSuccessResponseXML(w, r, response)
  49. }
  50. // CompleteMultipartUploadHandler - Completes multipart upload.
  51. func (s3a *S3ApiServer) CompleteMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
  52. // https://docs.aws.amazon.com/AmazonS3/latest/API/API_CompleteMultipartUpload.html
  53. bucket, object := s3_constants.GetBucketAndObject(r)
  54. parts := &CompleteMultipartUpload{}
  55. if err := xmlDecoder(r.Body, parts, r.ContentLength); err != nil {
  56. s3err.WriteErrorResponse(w, r, s3err.ErrMalformedXML)
  57. return
  58. }
  59. // Get upload id.
  60. uploadID, _, _, _ := getObjectResources(r.URL.Query())
  61. err := s3a.checkUploadId(object, uploadID)
  62. if err != nil {
  63. s3err.WriteErrorResponse(w, r, s3err.ErrNoSuchUpload)
  64. return
  65. }
  66. response, errCode := s3a.completeMultipartUpload(&s3.CompleteMultipartUploadInput{
  67. Bucket: aws.String(bucket),
  68. Key: objectKey(aws.String(object)),
  69. UploadId: aws.String(uploadID),
  70. }, parts)
  71. glog.V(2).Info("CompleteMultipartUploadHandler", string(s3err.EncodeXMLResponse(response)), errCode)
  72. if errCode != s3err.ErrNone {
  73. s3err.WriteErrorResponse(w, r, errCode)
  74. return
  75. }
  76. stats_collect.S3UploadedObjectsCounter.WithLabelValues(bucket).Inc()
  77. writeSuccessResponseXML(w, r, response)
  78. }
  79. // AbortMultipartUploadHandler - Aborts multipart upload.
  80. func (s3a *S3ApiServer) AbortMultipartUploadHandler(w http.ResponseWriter, r *http.Request) {
  81. bucket, object := s3_constants.GetBucketAndObject(r)
  82. // Get upload id.
  83. uploadID, _, _, _ := getObjectResources(r.URL.Query())
  84. err := s3a.checkUploadId(object, uploadID)
  85. if err != nil {
  86. s3err.WriteErrorResponse(w, r, s3err.ErrNoSuchUpload)
  87. return
  88. }
  89. response, errCode := s3a.abortMultipartUpload(&s3.AbortMultipartUploadInput{
  90. Bucket: aws.String(bucket),
  91. Key: objectKey(aws.String(object)),
  92. UploadId: aws.String(uploadID),
  93. })
  94. if errCode != s3err.ErrNone {
  95. s3err.WriteErrorResponse(w, r, errCode)
  96. return
  97. }
  98. glog.V(2).Info("AbortMultipartUploadHandler", string(s3err.EncodeXMLResponse(response)))
  99. //https://docs.aws.amazon.com/AmazonS3/latest/API/API_AbortMultipartUpload.html
  100. s3err.WriteEmptyResponse(w, r, http.StatusNoContent)
  101. s3err.PostLog(r, http.StatusNoContent, s3err.ErrNone)
  102. }
  103. // ListMultipartUploadsHandler - Lists multipart uploads.
  104. func (s3a *S3ApiServer) ListMultipartUploadsHandler(w http.ResponseWriter, r *http.Request) {
  105. bucket, _ := s3_constants.GetBucketAndObject(r)
  106. prefix, keyMarker, uploadIDMarker, delimiter, maxUploads, encodingType := getBucketMultipartResources(r.URL.Query())
  107. if maxUploads < 0 {
  108. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidMaxUploads)
  109. return
  110. }
  111. if keyMarker != "" {
  112. // Marker not common with prefix is not implemented.
  113. if !strings.HasPrefix(keyMarker, prefix) {
  114. s3err.WriteErrorResponse(w, r, s3err.ErrNotImplemented)
  115. return
  116. }
  117. }
  118. response, errCode := s3a.listMultipartUploads(&s3.ListMultipartUploadsInput{
  119. Bucket: aws.String(bucket),
  120. Delimiter: aws.String(delimiter),
  121. EncodingType: aws.String(encodingType),
  122. KeyMarker: aws.String(keyMarker),
  123. MaxUploads: aws.Int64(int64(maxUploads)),
  124. Prefix: aws.String(prefix),
  125. UploadIdMarker: aws.String(uploadIDMarker),
  126. })
  127. glog.V(2).Infof("ListMultipartUploadsHandler %s errCode=%d", string(s3err.EncodeXMLResponse(response)), errCode)
  128. if errCode != s3err.ErrNone {
  129. s3err.WriteErrorResponse(w, r, errCode)
  130. return
  131. }
  132. // TODO handle encodingType
  133. writeSuccessResponseXML(w, r, response)
  134. }
  135. // ListObjectPartsHandler - Lists object parts in a multipart upload.
  136. func (s3a *S3ApiServer) ListObjectPartsHandler(w http.ResponseWriter, r *http.Request) {
  137. bucket, object := s3_constants.GetBucketAndObject(r)
  138. uploadID, partNumberMarker, maxParts, _ := getObjectResources(r.URL.Query())
  139. if partNumberMarker < 0 {
  140. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidPartNumberMarker)
  141. return
  142. }
  143. if maxParts < 0 {
  144. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidMaxParts)
  145. return
  146. }
  147. err := s3a.checkUploadId(object, uploadID)
  148. if err != nil {
  149. s3err.WriteErrorResponse(w, r, s3err.ErrNoSuchUpload)
  150. return
  151. }
  152. response, errCode := s3a.listObjectParts(&s3.ListPartsInput{
  153. Bucket: aws.String(bucket),
  154. Key: objectKey(aws.String(object)),
  155. MaxParts: aws.Int64(int64(maxParts)),
  156. PartNumberMarker: aws.Int64(int64(partNumberMarker)),
  157. UploadId: aws.String(uploadID),
  158. })
  159. if errCode != s3err.ErrNone {
  160. s3err.WriteErrorResponse(w, r, errCode)
  161. return
  162. }
  163. glog.V(2).Infof("ListObjectPartsHandler %s count=%d", string(s3err.EncodeXMLResponse(response)), len(response.Part))
  164. writeSuccessResponseXML(w, r, response)
  165. }
  166. // PutObjectPartHandler - Put an object part in a multipart upload.
  167. func (s3a *S3ApiServer) PutObjectPartHandler(w http.ResponseWriter, r *http.Request) {
  168. bucket, object := s3_constants.GetBucketAndObject(r)
  169. uploadID := r.URL.Query().Get("uploadId")
  170. err := s3a.checkUploadId(object, uploadID)
  171. if err != nil {
  172. s3err.WriteErrorResponse(w, r, s3err.ErrNoSuchUpload)
  173. return
  174. }
  175. partIDString := r.URL.Query().Get("partNumber")
  176. partID, err := strconv.Atoi(partIDString)
  177. if err != nil {
  178. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidPart)
  179. return
  180. }
  181. if partID > globalMaxPartID {
  182. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidMaxParts)
  183. return
  184. }
  185. dataReader := r.Body
  186. rAuthType := getRequestAuthType(r)
  187. if s3a.iam.isEnabled() {
  188. var s3ErrCode s3err.ErrorCode
  189. switch rAuthType {
  190. case authTypeStreamingSigned:
  191. dataReader, s3ErrCode = s3a.iam.newSignV4ChunkedReader(r)
  192. case authTypeSignedV2, authTypePresignedV2:
  193. _, s3ErrCode = s3a.iam.isReqAuthenticatedV2(r)
  194. case authTypePresigned, authTypeSigned:
  195. _, s3ErrCode = s3a.iam.reqSignatureV4Verify(r)
  196. }
  197. if s3ErrCode != s3err.ErrNone {
  198. s3err.WriteErrorResponse(w, r, s3ErrCode)
  199. return
  200. }
  201. } else {
  202. if authTypeStreamingSigned == rAuthType {
  203. s3err.WriteErrorResponse(w, r, s3err.ErrAuthNotSetup)
  204. return
  205. }
  206. }
  207. defer dataReader.Close()
  208. glog.V(2).Infof("PutObjectPartHandler %s %s %04d", bucket, uploadID, partID)
  209. uploadUrl := s3a.genPartUploadUrl(bucket, uploadID, partID)
  210. if partID == 1 && r.Header.Get("Content-Type") == "" {
  211. dataReader = mimeDetect(r, dataReader)
  212. }
  213. destination := fmt.Sprintf("%s/%s%s", s3a.option.BucketsPath, bucket, object)
  214. etag, errCode := s3a.putToFiler(r, uploadUrl, dataReader, destination, bucket)
  215. if errCode != s3err.ErrNone {
  216. s3err.WriteErrorResponse(w, r, errCode)
  217. return
  218. }
  219. setEtag(w, etag)
  220. writeSuccessResponseEmpty(w, r)
  221. }
  222. func (s3a *S3ApiServer) genUploadsFolder(bucket string) string {
  223. return fmt.Sprintf("%s/%s/%s", s3a.option.BucketsPath, bucket, s3_constants.MultipartUploadsFolder)
  224. }
  225. func (s3a *S3ApiServer) genPartUploadUrl(bucket, uploadID string, partID int) string {
  226. return fmt.Sprintf("http://%s%s/%s/%04d_%s.part",
  227. s3a.option.Filer.ToHttpAddress(), s3a.genUploadsFolder(bucket), uploadID, partID, uuid.NewString())
  228. }
  229. // Generate uploadID hash string from object
  230. func (s3a *S3ApiServer) generateUploadID(object string) string {
  231. if strings.HasPrefix(object, "/") {
  232. object = object[1:]
  233. }
  234. h := sha1.New()
  235. h.Write([]byte(object))
  236. return fmt.Sprintf("%x", h.Sum(nil))
  237. }
  238. // Check object name and uploadID when processing multipart uploading
  239. func (s3a *S3ApiServer) checkUploadId(object string, id string) error {
  240. hash := s3a.generateUploadID(object)
  241. if !strings.HasPrefix(id, hash) {
  242. glog.Errorf("object %s and uploadID %s are not matched", object, id)
  243. return fmt.Errorf("object %s and uploadID %s are not matched", object, id)
  244. }
  245. return nil
  246. }
  247. // Parse bucket url queries for ?uploads
  248. func getBucketMultipartResources(values url.Values) (prefix, keyMarker, uploadIDMarker, delimiter string, maxUploads int, encodingType string) {
  249. prefix = values.Get("prefix")
  250. keyMarker = values.Get("key-marker")
  251. uploadIDMarker = values.Get("upload-id-marker")
  252. delimiter = values.Get("delimiter")
  253. if values.Get("max-uploads") != "" {
  254. maxUploads, _ = strconv.Atoi(values.Get("max-uploads"))
  255. } else {
  256. maxUploads = maxUploadsList
  257. }
  258. encodingType = values.Get("encoding-type")
  259. return
  260. }
  261. // Parse object url queries
  262. func getObjectResources(values url.Values) (uploadID string, partNumberMarker, maxParts int, encodingType string) {
  263. uploadID = values.Get("uploadId")
  264. partNumberMarker, _ = strconv.Atoi(values.Get("part-number-marker"))
  265. if values.Get("max-parts") != "" {
  266. maxParts, _ = strconv.Atoi(values.Get("max-parts"))
  267. } else {
  268. maxParts = maxPartsList
  269. }
  270. encodingType = values.Get("encoding-type")
  271. return
  272. }
  273. func xmlDecoder(body io.Reader, v interface{}, size int64) error {
  274. var lbody io.Reader
  275. if size > 0 {
  276. lbody = io.LimitReader(body, size)
  277. } else {
  278. lbody = body
  279. }
  280. d := xml.NewDecoder(lbody)
  281. d.CharsetReader = func(label string, input io.Reader) (io.Reader, error) {
  282. return input, nil
  283. }
  284. return d.Decode(v)
  285. }
  286. type CompleteMultipartUpload struct {
  287. Parts []CompletedPart `xml:"Part"`
  288. }
  289. type CompletedPart struct {
  290. ETag string
  291. PartNumber int
  292. }