filer_server_handlers_tagging.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. package weed_server
  2. import (
  3. "context"
  4. "net/http"
  5. "strings"
  6. "github.com/seaweedfs/seaweedfs/weed/glog"
  7. "github.com/seaweedfs/seaweedfs/weed/storage/needle"
  8. "github.com/seaweedfs/seaweedfs/weed/util"
  9. )
  10. // add or replace one file Seaweed- prefixed attributes
  11. // curl -X PUT -H "Seaweed-Name1: value1" http://localhost:8888/path/to/a/file?tagging
  12. func (fs *FilerServer) PutTaggingHandler(w http.ResponseWriter, r *http.Request) {
  13. ctx := context.Background()
  14. path := r.URL.Path
  15. if strings.HasSuffix(path, "/") {
  16. path = path[:len(path)-1]
  17. }
  18. existingEntry, err := fs.filer.FindEntry(ctx, util.FullPath(path))
  19. if err != nil {
  20. writeJsonError(w, r, http.StatusNotFound, err)
  21. return
  22. }
  23. if existingEntry == nil {
  24. writeJsonError(w, r, http.StatusNotFound, err)
  25. return
  26. }
  27. if existingEntry.Extended == nil {
  28. existingEntry.Extended = make(map[string][]byte)
  29. }
  30. for header, values := range r.Header {
  31. if strings.HasPrefix(header, needle.PairNamePrefix) {
  32. for _, value := range values {
  33. existingEntry.Extended[header] = []byte(value)
  34. }
  35. }
  36. }
  37. if dbErr := fs.filer.CreateEntry(ctx, existingEntry, false, false, nil, false, fs.filer.MaxFilenameLength); dbErr != nil {
  38. glog.V(0).Infof("failing to update %s tagging : %v", path, dbErr)
  39. writeJsonError(w, r, http.StatusInternalServerError, err)
  40. return
  41. }
  42. writeJsonQuiet(w, r, http.StatusAccepted, nil)
  43. return
  44. }
  45. // remove all Seaweed- prefixed attributes
  46. // curl -X DELETE http://localhost:8888/path/to/a/file?tagging
  47. func (fs *FilerServer) DeleteTaggingHandler(w http.ResponseWriter, r *http.Request) {
  48. ctx := context.Background()
  49. path := r.URL.Path
  50. if strings.HasSuffix(path, "/") {
  51. path = path[:len(path)-1]
  52. }
  53. existingEntry, err := fs.filer.FindEntry(ctx, util.FullPath(path))
  54. if err != nil {
  55. writeJsonError(w, r, http.StatusNotFound, err)
  56. return
  57. }
  58. if existingEntry == nil {
  59. writeJsonError(w, r, http.StatusNotFound, err)
  60. return
  61. }
  62. if existingEntry.Extended == nil {
  63. existingEntry.Extended = make(map[string][]byte)
  64. }
  65. // parse out tags to be deleted
  66. toDelete := strings.Split(r.URL.Query().Get("tagging"), ",")
  67. deletions := make(map[string]struct{})
  68. for _, deletion := range toDelete {
  69. if deletion != "" {
  70. deletions[deletion] = struct{}{}
  71. }
  72. }
  73. // delete all tags or specific tags
  74. hasDeletion := false
  75. for header, _ := range existingEntry.Extended {
  76. if strings.HasPrefix(header, needle.PairNamePrefix) {
  77. if len(deletions) == 0 {
  78. delete(existingEntry.Extended, header)
  79. hasDeletion = true
  80. } else {
  81. tag := header[len(needle.PairNamePrefix):]
  82. if _, found := deletions[tag]; found {
  83. delete(existingEntry.Extended, header)
  84. hasDeletion = true
  85. }
  86. }
  87. }
  88. }
  89. if !hasDeletion {
  90. writeJsonQuiet(w, r, http.StatusNotModified, nil)
  91. return
  92. }
  93. if dbErr := fs.filer.CreateEntry(ctx, existingEntry, false, false, nil, false, fs.filer.MaxFilenameLength); dbErr != nil {
  94. glog.V(0).Infof("failing to delete %s tagging : %v", path, dbErr)
  95. writeJsonError(w, r, http.StatusInternalServerError, err)
  96. return
  97. }
  98. writeJsonQuiet(w, r, http.StatusAccepted, nil)
  99. return
  100. }