opml_upload.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. // SPDX-FileCopyrightText: Copyright The Miniflux Authors. All rights reserved.
  2. // SPDX-License-Identifier: Apache-2.0
  3. package ui // import "miniflux.app/v2/internal/ui"
  4. import (
  5. "log/slog"
  6. "net/http"
  7. "strings"
  8. "miniflux.app/v2/internal/config"
  9. "miniflux.app/v2/internal/http/request"
  10. "miniflux.app/v2/internal/http/response/html"
  11. "miniflux.app/v2/internal/http/route"
  12. "miniflux.app/v2/internal/locale"
  13. "miniflux.app/v2/internal/proxyrotator"
  14. "miniflux.app/v2/internal/reader/fetcher"
  15. "miniflux.app/v2/internal/reader/opml"
  16. "miniflux.app/v2/internal/ui/session"
  17. "miniflux.app/v2/internal/ui/view"
  18. )
  19. func (h *handler) uploadOPML(w http.ResponseWriter, r *http.Request) {
  20. loggedUserID := request.UserID(r)
  21. user, err := h.store.UserByID(loggedUserID)
  22. if err != nil {
  23. html.ServerError(w, r, err)
  24. return
  25. }
  26. file, fileHeader, err := r.FormFile("file")
  27. if err != nil {
  28. slog.Error("OPML file upload error",
  29. slog.Int64("user_id", loggedUserID),
  30. slog.Any("error", err),
  31. )
  32. html.Redirect(w, r, route.Path(h.router, "import"))
  33. return
  34. }
  35. defer file.Close()
  36. slog.Info("OPML file uploaded",
  37. slog.Int64("user_id", loggedUserID),
  38. slog.String("file_name", fileHeader.Filename),
  39. slog.Int64("file_size", fileHeader.Size),
  40. )
  41. sess := session.New(h.store, request.SessionID(r))
  42. view := view.New(h.tpl, r, sess)
  43. view.Set("menu", "feeds")
  44. view.Set("user", user)
  45. view.Set("countUnread", h.store.CountUnreadEntries(user.ID))
  46. view.Set("countErrorFeeds", h.store.CountUserFeedsWithErrors(user.ID))
  47. if fileHeader.Size == 0 {
  48. view.Set("errorMessage", locale.NewLocalizedError("error.empty_file").Translate(user.Language))
  49. html.OK(w, r, view.Render("import"))
  50. return
  51. }
  52. if impErr := opml.NewHandler(h.store).Import(user.ID, file); impErr != nil {
  53. view.Set("errorMessage", impErr)
  54. html.OK(w, r, view.Render("import"))
  55. return
  56. }
  57. html.Redirect(w, r, route.Path(h.router, "feeds"))
  58. }
  59. func (h *handler) fetchOPML(w http.ResponseWriter, r *http.Request) {
  60. loggedUserID := request.UserID(r)
  61. user, err := h.store.UserByID(loggedUserID)
  62. if err != nil {
  63. html.ServerError(w, r, err)
  64. return
  65. }
  66. opmlFileURL := strings.TrimSpace(r.FormValue("url"))
  67. if opmlFileURL == "" {
  68. html.Redirect(w, r, route.Path(h.router, "import"))
  69. return
  70. }
  71. slog.Info("Fetching OPML file remotely",
  72. slog.Int64("user_id", loggedUserID),
  73. slog.String("opml_file_url", opmlFileURL),
  74. )
  75. sess := session.New(h.store, request.SessionID(r))
  76. view := view.New(h.tpl, r, sess)
  77. view.Set("menu", "feeds")
  78. view.Set("user", user)
  79. view.Set("countUnread", h.store.CountUnreadEntries(user.ID))
  80. view.Set("countErrorFeeds", h.store.CountUserFeedsWithErrors(user.ID))
  81. requestBuilder := fetcher.NewRequestBuilder()
  82. requestBuilder.WithTimeout(config.Opts.HTTPClientTimeout())
  83. requestBuilder.WithProxyRotator(proxyrotator.ProxyRotatorInstance)
  84. responseHandler := fetcher.NewResponseHandler(requestBuilder.ExecuteRequest(opmlFileURL))
  85. defer responseHandler.Close()
  86. if localizedError := responseHandler.LocalizedError(); localizedError != nil {
  87. slog.Warn("Unable to fetch OPML file", slog.String("opml_file_url", opmlFileURL), slog.Any("error", localizedError.Error()))
  88. view.Set("errorMessage", localizedError.Translate(user.Language))
  89. html.OK(w, r, view.Render("import"))
  90. return
  91. }
  92. if impErr := opml.NewHandler(h.store).Import(user.ID, responseHandler.Body(config.Opts.HTTPClientMaxBodySize())); impErr != nil {
  93. view.Set("errorMessage", impErr)
  94. html.OK(w, r, view.Render("import"))
  95. return
  96. }
  97. html.Redirect(w, r, route.Path(h.router, "feeds"))
  98. }