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.

179 lines
4.6 KiB

7 years ago
7 years ago
  1. package s3api
  2. import (
  3. "context"
  4. "fmt"
  5. "github.com/chrislusf/seaweedfs/weed/filer2"
  6. "github.com/chrislusf/seaweedfs/weed/glog"
  7. "github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
  8. "github.com/gorilla/mux"
  9. "net/http"
  10. "net/url"
  11. "path/filepath"
  12. "strconv"
  13. "time"
  14. )
  15. const (
  16. maxObjectListSizeLimit = 1000 // Limit number of objects in a listObjectsResponse.
  17. )
  18. func (s3a *S3ApiServer) ListObjectsV2Handler(w http.ResponseWriter, r *http.Request) {
  19. // https://docs.aws.amazon.com/AmazonS3/latest/API/v2-RESTBucketGET.html
  20. // collect parameters
  21. vars := mux.Vars(r)
  22. bucket := vars["bucket"]
  23. originalPrefix, marker, startAfter, delimiter, _, maxKeys := getListObjectsV2Args(r.URL.Query())
  24. if maxKeys < 0 {
  25. writeErrorResponse(w, ErrInvalidMaxKeys, r.URL)
  26. return
  27. }
  28. if delimiter != "" && delimiter != "/" {
  29. writeErrorResponse(w, ErrNotImplemented, r.URL)
  30. return
  31. }
  32. if marker == "" {
  33. marker = startAfter
  34. }
  35. response, err := s3a.listFilerEntries(bucket, originalPrefix, maxKeys, marker)
  36. if err != nil {
  37. writeErrorResponse(w, ErrInternalError, r.URL)
  38. return
  39. }
  40. writeSuccessResponseXML(w, encodeResponse(response))
  41. }
  42. func (s3a *S3ApiServer) ListObjectsV1Handler(w http.ResponseWriter, r *http.Request) {
  43. // https://docs.aws.amazon.com/AmazonS3/latest/API/RESTBucketGET.html
  44. // collect parameters
  45. vars := mux.Vars(r)
  46. bucket := vars["bucket"]
  47. originalPrefix, marker, delimiter, maxKeys := getListObjectsV1Args(r.URL.Query())
  48. if maxKeys < 0 {
  49. writeErrorResponse(w, ErrInvalidMaxKeys, r.URL)
  50. return
  51. }
  52. if delimiter != "" && delimiter != "/" {
  53. writeErrorResponse(w, ErrNotImplemented, r.URL)
  54. return
  55. }
  56. response, err := s3a.listFilerEntries(bucket, originalPrefix, maxKeys, marker)
  57. if err != nil {
  58. writeErrorResponse(w, ErrInternalError, r.URL)
  59. return
  60. }
  61. writeSuccessResponseXML(w, encodeResponse(response))
  62. }
  63. func (s3a *S3ApiServer) listFilerEntries(bucket, originalPrefix string, maxKeys int, marker string) (response ListBucketResponse, err error) {
  64. // convert full path prefix into directory name and prefix for entry name
  65. dir, prefix := filepath.Split(originalPrefix)
  66. // check filer
  67. err = s3a.withFilerClient(func(client filer_pb.SeaweedFilerClient) error {
  68. request := &filer_pb.ListEntriesRequest{
  69. Directory: fmt.Sprintf("%s/%s/%s", s3a.option.BucketsPath, bucket, dir),
  70. Prefix: prefix,
  71. Limit: uint32(maxKeys + 1),
  72. StartFromFileName: marker,
  73. InclusiveStartFrom: false,
  74. }
  75. glog.V(4).Infof("read directory: %v", request)
  76. resp, err := client.ListEntries(context.Background(), request)
  77. if err != nil {
  78. return fmt.Errorf("list buckets: %v", err)
  79. }
  80. var contents []ListEntry
  81. var commonPrefixes []PrefixEntry
  82. var counter int
  83. var lastEntryName string
  84. var isTruncated bool
  85. for _, entry := range resp.Entries {
  86. counter++
  87. if counter > maxKeys {
  88. isTruncated = true
  89. break
  90. }
  91. lastEntryName = entry.Name
  92. if entry.IsDirectory {
  93. commonPrefixes = append(commonPrefixes, PrefixEntry{
  94. Prefix: fmt.Sprintf("%s%s/", dir, entry.Name),
  95. })
  96. } else {
  97. contents = append(contents, ListEntry{
  98. Key: fmt.Sprintf("%s%s", dir, entry.Name),
  99. LastModified: time.Unix(entry.Attributes.Mtime, 0),
  100. ETag: "", // TODO add etag
  101. Size: int64(filer2.TotalSize(entry.Chunks)),
  102. Owner: CanonicalUser{
  103. ID: fmt.Sprintf("%d", entry.Attributes.Uid),
  104. },
  105. StorageClass: StorageClass("STANDARD"),
  106. })
  107. }
  108. }
  109. response = ListBucketResponse{
  110. ListBucketResponse: ListBucketResult{
  111. Name: bucket,
  112. Prefix: originalPrefix,
  113. Marker: marker,
  114. NextMarker: lastEntryName,
  115. MaxKeys: maxKeys,
  116. Delimiter: "/",
  117. IsTruncated: isTruncated,
  118. Contents: contents,
  119. CommonPrefixes: commonPrefixes,
  120. },
  121. }
  122. return nil
  123. })
  124. return
  125. }
  126. func getListObjectsV2Args(values url.Values) (prefix, token, startAfter, delimiter string, fetchOwner bool, maxkeys int) {
  127. prefix = values.Get("prefix")
  128. token = values.Get("continuation-token")
  129. startAfter = values.Get("start-after")
  130. delimiter = values.Get("delimiter")
  131. if values.Get("max-keys") != "" {
  132. maxkeys, _ = strconv.Atoi(values.Get("max-keys"))
  133. } else {
  134. maxkeys = maxObjectListSizeLimit
  135. }
  136. fetchOwner = values.Get("fetch-owner") == "true"
  137. return
  138. }
  139. func getListObjectsV1Args(values url.Values) (prefix, marker, delimiter string, maxkeys int) {
  140. prefix = values.Get("prefix")
  141. marker = values.Get("marker")
  142. delimiter = values.Get("delimiter")
  143. if values.Get("max-keys") != "" {
  144. maxkeys, _ = strconv.Atoi(values.Get("max-keys"))
  145. } else {
  146. maxkeys = maxObjectListSizeLimit
  147. }
  148. return
  149. }