s3api_object_handlers_delete.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. package s3api
  2. import (
  3. "encoding/xml"
  4. "fmt"
  5. "io"
  6. "net/http"
  7. "slices"
  8. "strings"
  9. "github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
  10. "github.com/seaweedfs/seaweedfs/weed/filer"
  11. "github.com/seaweedfs/seaweedfs/weed/s3api/s3err"
  12. "github.com/seaweedfs/seaweedfs/weed/glog"
  13. "github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
  14. stats_collect "github.com/seaweedfs/seaweedfs/weed/stats"
  15. "github.com/seaweedfs/seaweedfs/weed/util"
  16. )
  17. const (
  18. deleteMultipleObjectsLimit = 1000
  19. )
  20. func (s3a *S3ApiServer) DeleteObjectHandler(w http.ResponseWriter, r *http.Request) {
  21. bucket, object := s3_constants.GetBucketAndObject(r)
  22. glog.V(3).Infof("DeleteObjectHandler %s %s", bucket, object)
  23. target := util.FullPath(fmt.Sprintf("%s/%s%s", s3a.option.BucketsPath, bucket, object))
  24. dir, name := target.DirAndName()
  25. err := s3a.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
  26. if err := doDeleteEntry(client, dir, name, true, false); err != nil {
  27. return err
  28. }
  29. if s3a.option.AllowEmptyFolder {
  30. return nil
  31. }
  32. directoriesWithDeletion := make(map[string]int)
  33. if strings.LastIndex(object, "/") > 0 {
  34. directoriesWithDeletion[dir]++
  35. // purge empty folders, only checking folders with deletions
  36. for len(directoriesWithDeletion) > 0 {
  37. directoriesWithDeletion = s3a.doDeleteEmptyDirectories(client, directoriesWithDeletion)
  38. }
  39. }
  40. return nil
  41. })
  42. if err != nil {
  43. s3err.WriteErrorResponse(w, r, s3err.ErrInternalError)
  44. return
  45. }
  46. stats_collect.RecordBucketActiveTime(bucket)
  47. stats_collect.S3DeletedObjectsCounter.WithLabelValues(bucket).Inc()
  48. w.WriteHeader(http.StatusNoContent)
  49. }
  50. // / ObjectIdentifier carries key name for the object to delete.
  51. type ObjectIdentifier struct {
  52. ObjectName string `xml:"Key"`
  53. }
  54. // DeleteObjectsRequest - xml carrying the object key names which needs to be deleted.
  55. type DeleteObjectsRequest struct {
  56. // Element to enable quiet mode for the request
  57. Quiet bool
  58. // List of objects to be deleted
  59. Objects []ObjectIdentifier `xml:"Object"`
  60. }
  61. // DeleteError structure.
  62. type DeleteError struct {
  63. Code string
  64. Message string
  65. Key string
  66. }
  67. // DeleteObjectsResponse container for multiple object deletes.
  68. type DeleteObjectsResponse struct {
  69. XMLName xml.Name `xml:"http://s3.amazonaws.com/doc/2006-03-01/ DeleteResult" json:"-"`
  70. // Collection of all deleted objects
  71. DeletedObjects []ObjectIdentifier `xml:"Deleted,omitempty"`
  72. // Collection of errors deleting certain objects.
  73. Errors []DeleteError `xml:"Error,omitempty"`
  74. }
  75. // DeleteMultipleObjectsHandler - Delete multiple objects
  76. func (s3a *S3ApiServer) DeleteMultipleObjectsHandler(w http.ResponseWriter, r *http.Request) {
  77. bucket, _ := s3_constants.GetBucketAndObject(r)
  78. glog.V(3).Infof("DeleteMultipleObjectsHandler %s", bucket)
  79. deleteXMLBytes, err := io.ReadAll(r.Body)
  80. if err != nil {
  81. s3err.WriteErrorResponse(w, r, s3err.ErrInternalError)
  82. return
  83. }
  84. deleteObjects := &DeleteObjectsRequest{}
  85. if err := xml.Unmarshal(deleteXMLBytes, deleteObjects); err != nil {
  86. s3err.WriteErrorResponse(w, r, s3err.ErrMalformedXML)
  87. return
  88. }
  89. if len(deleteObjects.Objects) > deleteMultipleObjectsLimit {
  90. s3err.WriteErrorResponse(w, r, s3err.ErrInvalidMaxDeleteObjects)
  91. return
  92. }
  93. var deletedObjects []ObjectIdentifier
  94. var deleteErrors []DeleteError
  95. var auditLog *s3err.AccessLog
  96. directoriesWithDeletion := make(map[string]int)
  97. if s3err.Logger != nil {
  98. auditLog = s3err.GetAccessLog(r, http.StatusNoContent, s3err.ErrNone)
  99. }
  100. s3a.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
  101. // delete file entries
  102. for _, object := range deleteObjects.Objects {
  103. if object.ObjectName == "" {
  104. continue
  105. }
  106. lastSeparator := strings.LastIndex(object.ObjectName, "/")
  107. parentDirectoryPath, entryName, isDeleteData, isRecursive := "", object.ObjectName, true, false
  108. if lastSeparator > 0 && lastSeparator+1 < len(object.ObjectName) {
  109. entryName = object.ObjectName[lastSeparator+1:]
  110. parentDirectoryPath = "/" + object.ObjectName[:lastSeparator]
  111. }
  112. parentDirectoryPath = fmt.Sprintf("%s/%s%s", s3a.option.BucketsPath, bucket, parentDirectoryPath)
  113. err := doDeleteEntry(client, parentDirectoryPath, entryName, isDeleteData, isRecursive)
  114. if err == nil {
  115. directoriesWithDeletion[parentDirectoryPath]++
  116. deletedObjects = append(deletedObjects, object)
  117. } else if strings.Contains(err.Error(), filer.MsgFailDelNonEmptyFolder) {
  118. deletedObjects = append(deletedObjects, object)
  119. } else {
  120. delete(directoriesWithDeletion, parentDirectoryPath)
  121. deleteErrors = append(deleteErrors, DeleteError{
  122. Code: "",
  123. Message: err.Error(),
  124. Key: object.ObjectName,
  125. })
  126. }
  127. if auditLog != nil {
  128. auditLog.Key = entryName
  129. s3err.PostAccessLog(*auditLog)
  130. }
  131. }
  132. if s3a.option.AllowEmptyFolder {
  133. return nil
  134. }
  135. // purge empty folders, only checking folders with deletions
  136. for len(directoriesWithDeletion) > 0 {
  137. directoriesWithDeletion = s3a.doDeleteEmptyDirectories(client, directoriesWithDeletion)
  138. }
  139. return nil
  140. })
  141. deleteResp := DeleteObjectsResponse{}
  142. if !deleteObjects.Quiet {
  143. deleteResp.DeletedObjects = deletedObjects
  144. }
  145. deleteResp.Errors = deleteErrors
  146. stats_collect.RecordBucketActiveTime(bucket)
  147. stats_collect.S3DeletedObjectsCounter.WithLabelValues(bucket).Add(float64(len(deletedObjects)))
  148. writeSuccessResponseXML(w, r, deleteResp)
  149. }
  150. func (s3a *S3ApiServer) doDeleteEmptyDirectories(client filer_pb.SeaweedFilerClient, directoriesWithDeletion map[string]int) (newDirectoriesWithDeletion map[string]int) {
  151. var allDirs []string
  152. for dir := range directoriesWithDeletion {
  153. allDirs = append(allDirs, dir)
  154. }
  155. slices.SortFunc(allDirs, func(a, b string) int {
  156. return len(b) - len(a)
  157. })
  158. newDirectoriesWithDeletion = make(map[string]int)
  159. for _, dir := range allDirs {
  160. parentDir, dirName := util.FullPath(dir).DirAndName()
  161. if parentDir == s3a.option.BucketsPath {
  162. continue
  163. }
  164. if err := doDeleteEntry(client, parentDir, dirName, false, false); err != nil {
  165. glog.V(4).Infof("directory %s has %d deletion but still not empty: %v", dir, directoriesWithDeletion[dir], err)
  166. } else {
  167. newDirectoriesWithDeletion[parentDir]++
  168. }
  169. }
  170. return
  171. }