rss.go 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. package rss
  2. import (
  3. "context"
  4. "fmt"
  5. "net/http"
  6. "strconv"
  7. "time"
  8. "github.com/gorilla/feeds"
  9. "github.com/labstack/echo/v4"
  10. "github.com/usememos/gomark"
  11. "github.com/usememos/gomark/renderer"
  12. storepb "github.com/usememos/memos/proto/gen/store"
  13. "github.com/usememos/memos/server/profile"
  14. "github.com/usememos/memos/store"
  15. )
  16. const (
  17. maxRSSItemCount = 100
  18. )
  19. type RSSService struct {
  20. Profile *profile.Profile
  21. Store *store.Store
  22. }
  23. func NewRSSService(profile *profile.Profile, store *store.Store) *RSSService {
  24. return &RSSService{
  25. Profile: profile,
  26. Store: store,
  27. }
  28. }
  29. func (s *RSSService) RegisterRoutes(g *echo.Group) {
  30. g.GET("/explore/rss.xml", s.GetExploreRSS)
  31. g.GET("/u/:username/rss.xml", s.GetUserRSS)
  32. }
  33. func (s *RSSService) GetExploreRSS(c echo.Context) error {
  34. ctx := c.Request().Context()
  35. normalStatus := store.Normal
  36. memoFind := store.FindMemo{
  37. RowStatus: &normalStatus,
  38. VisibilityList: []store.Visibility{store.Public},
  39. }
  40. memoList, err := s.Store.ListMemos(ctx, &memoFind)
  41. if err != nil {
  42. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
  43. }
  44. baseURL := c.Scheme() + "://" + c.Request().Host
  45. rss, err := s.generateRSSFromMemoList(ctx, memoList, baseURL)
  46. if err != nil {
  47. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to generate rss").SetInternal(err)
  48. }
  49. c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationXMLCharsetUTF8)
  50. return c.String(http.StatusOK, rss)
  51. }
  52. func (s *RSSService) GetUserRSS(c echo.Context) error {
  53. ctx := c.Request().Context()
  54. username := c.Param("username")
  55. user, err := s.Store.GetUser(ctx, &store.FindUser{
  56. Username: &username,
  57. })
  58. if err != nil {
  59. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find user").SetInternal(err)
  60. }
  61. if user == nil {
  62. return echo.NewHTTPError(http.StatusNotFound, "User not found")
  63. }
  64. normalStatus := store.Normal
  65. memoFind := store.FindMemo{
  66. CreatorID: &user.ID,
  67. RowStatus: &normalStatus,
  68. VisibilityList: []store.Visibility{store.Public},
  69. }
  70. memoList, err := s.Store.ListMemos(ctx, &memoFind)
  71. if err != nil {
  72. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
  73. }
  74. baseURL := c.Scheme() + "://" + c.Request().Host
  75. rss, err := s.generateRSSFromMemoList(ctx, memoList, baseURL)
  76. if err != nil {
  77. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to generate rss").SetInternal(err)
  78. }
  79. c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationXMLCharsetUTF8)
  80. return c.String(http.StatusOK, rss)
  81. }
  82. func (s *RSSService) generateRSSFromMemoList(ctx context.Context, memoList []*store.Memo, baseURL string) (string, error) {
  83. feed := &feeds.Feed{
  84. Title: "Memos",
  85. Link: &feeds.Link{Href: baseURL},
  86. Description: "An open source, lightweight note-taking service. Easily capture and share your great thoughts.",
  87. Created: time.Now(),
  88. }
  89. var itemCountLimit = min(len(memoList), maxRSSItemCount)
  90. feed.Items = make([]*feeds.Item, itemCountLimit)
  91. for i := 0; i < itemCountLimit; i++ {
  92. memo := memoList[i]
  93. description, err := getRSSItemDescription(memo.Content)
  94. if err != nil {
  95. return "", err
  96. }
  97. feed.Items[i] = &feeds.Item{
  98. Link: &feeds.Link{Href: baseURL + "/m/" + memo.UID},
  99. Description: description,
  100. Created: time.Unix(memo.CreatedTs, 0),
  101. }
  102. resources, err := s.Store.ListResources(ctx, &store.FindResource{
  103. MemoID: &memo.ID,
  104. })
  105. if err != nil {
  106. return "", err
  107. }
  108. if len(resources) > 0 {
  109. resource := resources[0]
  110. enclosure := feeds.Enclosure{}
  111. if resource.StorageType == storepb.ResourceStorageType_EXTERNAL || resource.StorageType == storepb.ResourceStorageType_S3 {
  112. enclosure.Url = resource.Reference
  113. } else {
  114. enclosure.Url = fmt.Sprintf("%s/file/resources/%d/%s", baseURL, resource.ID, resource.Filename)
  115. }
  116. enclosure.Length = strconv.Itoa(int(resource.Size))
  117. enclosure.Type = resource.Type
  118. feed.Items[i].Enclosure = &enclosure
  119. }
  120. }
  121. rss, err := feed.ToRss()
  122. if err != nil {
  123. return "", err
  124. }
  125. return rss, nil
  126. }
  127. func getRSSItemDescription(content string) (string, error) {
  128. nodes, err := gomark.Parse(content)
  129. if err != nil {
  130. return "", err
  131. }
  132. result := renderer.NewHTMLRenderer().Render(nodes)
  133. return result, nil
  134. }