category_handlers.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package api // import "miniflux.app/v2/internal/api"
  4. import (
  5. json_parser "encoding/json"
  6. "errors"
  7. "log/slog"
  8. "net/http"
  9. "time"
  10. "miniflux.app/v2/internal/config"
  11. "miniflux.app/v2/internal/http/request"
  12. "miniflux.app/v2/internal/http/response"
  13. "miniflux.app/v2/internal/model"
  14. "miniflux.app/v2/internal/validator"
  15. )
  16. func (h *handler) createCategoryHandler(w http.ResponseWriter, r *http.Request) {
  17. userID := request.UserID(r)
  18. var categoryCreationRequest model.CategoryCreationRequest
  19. if err := json_parser.NewDecoder(r.Body).Decode(&categoryCreationRequest); err != nil {
  20. response.JSONBadRequest(w, r, err)
  21. return
  22. }
  23. if validationErr := validator.ValidateCategoryCreation(h.store, userID, &categoryCreationRequest); validationErr != nil {
  24. response.JSONBadRequest(w, r, validationErr.Error())
  25. return
  26. }
  27. category, err := h.store.CreateCategory(userID, &categoryCreationRequest)
  28. if err != nil {
  29. response.JSONServerError(w, r, err)
  30. return
  31. }
  32. response.JSONCreated(w, r, category)
  33. }
  34. func (h *handler) updateCategoryHandler(w http.ResponseWriter, r *http.Request) {
  35. userID := request.UserID(r)
  36. categoryID := request.RouteInt64Param(r, "categoryID")
  37. if categoryID == 0 {
  38. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  39. return
  40. }
  41. category, err := h.store.Category(userID, categoryID)
  42. if err != nil {
  43. response.JSONServerError(w, r, err)
  44. return
  45. }
  46. if category == nil {
  47. response.JSONNotFound(w, r)
  48. return
  49. }
  50. var categoryModificationRequest model.CategoryModificationRequest
  51. if err := json_parser.NewDecoder(r.Body).Decode(&categoryModificationRequest); err != nil {
  52. response.JSONBadRequest(w, r, err)
  53. return
  54. }
  55. if validationErr := validator.ValidateCategoryModification(h.store, userID, category.ID, &categoryModificationRequest); validationErr != nil {
  56. response.JSONBadRequest(w, r, validationErr.Error())
  57. return
  58. }
  59. categoryModificationRequest.Patch(category)
  60. if err := h.store.UpdateCategory(category); err != nil {
  61. response.JSONServerError(w, r, err)
  62. return
  63. }
  64. response.JSONCreated(w, r, category)
  65. }
  66. func (h *handler) markCategoryAsReadHandler(w http.ResponseWriter, r *http.Request) {
  67. userID := request.UserID(r)
  68. categoryID := request.RouteInt64Param(r, "categoryID")
  69. if categoryID == 0 {
  70. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  71. return
  72. }
  73. category, err := h.store.Category(userID, categoryID)
  74. if err != nil {
  75. response.JSONServerError(w, r, err)
  76. return
  77. }
  78. if category == nil {
  79. response.JSONNotFound(w, r)
  80. return
  81. }
  82. if err = h.store.MarkCategoryAsRead(userID, categoryID, time.Now()); err != nil {
  83. response.JSONServerError(w, r, err)
  84. return
  85. }
  86. response.NoContent(w, r)
  87. }
  88. func (h *handler) getCategoriesHandler(w http.ResponseWriter, r *http.Request) {
  89. var categories model.Categories
  90. var err error
  91. if request.QueryBoolParam(r, "counts", false) {
  92. user, userErr := h.store.UserByID(request.UserID(r))
  93. if userErr != nil {
  94. response.JSONServerError(w, r, userErr)
  95. return
  96. }
  97. if user == nil {
  98. response.JSONNotFound(w, r)
  99. return
  100. }
  101. categories, err = h.store.CategoriesWithFeedCount(user.ID, user.CategoriesSortingOrder)
  102. } else {
  103. categories, err = h.store.Categories(request.UserID(r))
  104. }
  105. if err != nil {
  106. response.JSONServerError(w, r, err)
  107. return
  108. }
  109. response.JSON(w, r, categories)
  110. }
  111. func (h *handler) removeCategoryHandler(w http.ResponseWriter, r *http.Request) {
  112. userID := request.UserID(r)
  113. categoryID := request.RouteInt64Param(r, "categoryID")
  114. if categoryID == 0 {
  115. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  116. return
  117. }
  118. if !h.store.CategoryIDExists(userID, categoryID) {
  119. response.JSONNotFound(w, r)
  120. return
  121. }
  122. if err := h.store.RemoveCategory(userID, categoryID); err != nil {
  123. response.JSONServerError(w, r, err)
  124. return
  125. }
  126. response.NoContent(w, r)
  127. }
  128. func (h *handler) refreshCategoryHandler(w http.ResponseWriter, r *http.Request) {
  129. userID := request.UserID(r)
  130. categoryID := request.RouteInt64Param(r, "categoryID")
  131. if categoryID == 0 {
  132. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  133. return
  134. }
  135. jobs, err := h.store.NewBatchBuilder().
  136. WithErrorLimit(config.Opts.PollingParsingErrorLimit()).
  137. WithoutDisabledFeeds().
  138. WithUserID(userID).
  139. WithCategoryID(categoryID).
  140. WithNextCheckExpired().
  141. WithLimitPerHost(config.Opts.PollingLimitPerHost()).
  142. FetchJobs()
  143. if err != nil {
  144. response.JSONServerError(w, r, err)
  145. return
  146. }
  147. slog.Info(
  148. "Triggered a manual refresh of all feeds for a given category from the API",
  149. slog.Int64("user_id", userID),
  150. slog.Int64("category_id", categoryID),
  151. slog.Int("nb_jobs", len(jobs)),
  152. )
  153. go h.pool.Push(jobs)
  154. response.NoContent(w, r)
  155. }