category_handlers.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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. includeCounts := request.QueryStringParam(r, "counts", "false")
  92. if includeCounts == "true" {
  93. categories, err = h.store.CategoriesWithFeedCount(request.UserID(r))
  94. } else {
  95. categories, err = h.store.Categories(request.UserID(r))
  96. }
  97. if err != nil {
  98. response.JSONServerError(w, r, err)
  99. return
  100. }
  101. response.JSON(w, r, categories)
  102. }
  103. func (h *handler) removeCategoryHandler(w http.ResponseWriter, r *http.Request) {
  104. userID := request.UserID(r)
  105. categoryID := request.RouteInt64Param(r, "categoryID")
  106. if categoryID == 0 {
  107. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  108. return
  109. }
  110. if !h.store.CategoryIDExists(userID, categoryID) {
  111. response.JSONNotFound(w, r)
  112. return
  113. }
  114. if err := h.store.RemoveCategory(userID, categoryID); err != nil {
  115. response.JSONServerError(w, r, err)
  116. return
  117. }
  118. response.NoContent(w, r)
  119. }
  120. func (h *handler) refreshCategoryHandler(w http.ResponseWriter, r *http.Request) {
  121. userID := request.UserID(r)
  122. categoryID := request.RouteInt64Param(r, "categoryID")
  123. if categoryID == 0 {
  124. response.JSONBadRequest(w, r, errors.New("invalid category ID"))
  125. return
  126. }
  127. batchBuilder := h.store.NewBatchBuilder()
  128. batchBuilder.WithErrorLimit(config.Opts.PollingParsingErrorLimit())
  129. batchBuilder.WithoutDisabledFeeds()
  130. batchBuilder.WithUserID(userID)
  131. batchBuilder.WithCategoryID(categoryID)
  132. batchBuilder.WithNextCheckExpired()
  133. batchBuilder.WithLimitPerHost(config.Opts.PollingLimitPerHost())
  134. jobs, err := batchBuilder.FetchJobs()
  135. if err != nil {
  136. response.JSONServerError(w, r, err)
  137. return
  138. }
  139. slog.Info(
  140. "Triggered a manual refresh of all feeds for a given category from the API",
  141. slog.Int64("user_id", userID),
  142. slog.Int64("category_id", categoryID),
  143. slog.Int("nb_jobs", len(jobs)),
  144. )
  145. go h.pool.Push(jobs)
  146. response.NoContent(w, r)
  147. }