mirror of
				https://github.com/usememos/memos.git
				synced 2025-10-25 05:46:03 +08:00 
			
		
		
		
	* feat: generate RSS item title and limit item count * Update server/rss.go Co-authored-by: boojack <stevenlgtm@gmail.com> * Update server/rss.go --------- Co-authored-by: boojack <stevenlgtm@gmail.com>
		
			
				
	
	
		
			204 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			204 lines
		
	
	
	
		
			5.6 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package server
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"encoding/json"
 | |
| 	"net/http"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/gorilla/feeds"
 | |
| 	"github.com/labstack/echo/v4"
 | |
| 	"github.com/usememos/memos/api"
 | |
| )
 | |
| 
 | |
| func (s *Server) registerRSSRoutes(g *echo.Group) {
 | |
| 	g.GET("/explore/rss.xml", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 
 | |
| 		systemCustomizedProfile, err := getSystemCustomizedProfile(ctx, s)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to get system customized profile").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		normalStatus := api.Normal
 | |
| 		memoFind := api.MemoFind{
 | |
| 			RowStatus: &normalStatus,
 | |
| 			VisibilityList: []api.Visibility{
 | |
| 				api.Public,
 | |
| 			},
 | |
| 		}
 | |
| 		memoList, err := s.Store.FindMemoList(ctx, &memoFind)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		baseURL := c.Scheme() + "://" + c.Request().Host
 | |
| 		rss, err := generateRSSFromMemoList(memoList, baseURL, &systemCustomizedProfile)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to generate rss").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationXMLCharsetUTF8)
 | |
| 		return c.String(http.StatusOK, rss)
 | |
| 	})
 | |
| 
 | |
| 	g.GET("/u/:id/rss.xml", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 
 | |
| 		systemCustomizedProfile, err := getSystemCustomizedProfile(ctx, s)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to get system customized profile").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		id, err := strconv.Atoi(c.Param("id"))
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "User id is not a number").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		normalStatus := api.Normal
 | |
| 		memoFind := api.MemoFind{
 | |
| 			CreatorID: &id,
 | |
| 			RowStatus: &normalStatus,
 | |
| 			VisibilityList: []api.Visibility{
 | |
| 				api.Public,
 | |
| 			},
 | |
| 		}
 | |
| 		memoList, err := s.Store.FindMemoList(ctx, &memoFind)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		baseURL := c.Scheme() + "://" + c.Request().Host
 | |
| 
 | |
| 		rss, err := generateRSSFromMemoList(memoList, baseURL, &systemCustomizedProfile)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to generate rss").SetInternal(err)
 | |
| 		}
 | |
| 		c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationXMLCharsetUTF8)
 | |
| 		return c.String(http.StatusOK, rss)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| const MaxRSSItemCount = 100
 | |
| const MaxRSSItemTitleLength = 100
 | |
| 
 | |
| func generateRSSFromMemoList(memoList []*api.Memo, baseURL string, profile *api.CustomizedProfile) (string, error) {
 | |
| 	feed := &feeds.Feed{
 | |
| 		Title:       profile.Name,
 | |
| 		Link:        &feeds.Link{Href: baseURL},
 | |
| 		Description: profile.Description,
 | |
| 		Created:     time.Now(),
 | |
| 	}
 | |
| 
 | |
| 	var itemCountLimit = min(len(memoList), MaxRSSItemCount)
 | |
| 
 | |
| 	feed.Items = make([]*feeds.Item, itemCountLimit)
 | |
| 
 | |
| 	for i := 0; i < itemCountLimit; i++ {
 | |
| 		memo := memoList[i]
 | |
| 
 | |
| 		feed.Items[i] = &feeds.Item{
 | |
| 			Title:       getRSSItemTitle(memo.Content),
 | |
| 			Link:        &feeds.Link{Href: baseURL + "/m/" + strconv.Itoa(memo.ID)},
 | |
| 			Description: getRSSItemDescription(memo.Content),
 | |
| 			Created:     time.Unix(memo.CreatedTs, 0),
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	rss, err := feed.ToRss()
 | |
| 	if err != nil {
 | |
| 		return "", err
 | |
| 	}
 | |
| 	return rss, nil
 | |
| }
 | |
| 
 | |
| func getSystemCustomizedProfile(ctx context.Context, s *Server) (api.CustomizedProfile, error) {
 | |
| 	systemStatus := api.SystemStatus{
 | |
| 		CustomizedProfile: api.CustomizedProfile{
 | |
| 			Name:        "memos",
 | |
| 			LogoURL:     "",
 | |
| 			Description: "",
 | |
| 			Locale:      "en",
 | |
| 			Appearance:  "system",
 | |
| 			ExternalURL: "",
 | |
| 		},
 | |
| 	}
 | |
| 
 | |
| 	systemSettingList, err := s.Store.FindSystemSettingList(ctx, &api.SystemSettingFind{})
 | |
| 	if err != nil {
 | |
| 		return api.CustomizedProfile{}, err
 | |
| 	}
 | |
| 	for _, systemSetting := range systemSettingList {
 | |
| 		if systemSetting.Name == api.SystemSettingServerID || systemSetting.Name == api.SystemSettingSecretSessionName {
 | |
| 			continue
 | |
| 		}
 | |
| 
 | |
| 		var value interface{}
 | |
| 		err := json.Unmarshal([]byte(systemSetting.Value), &value)
 | |
| 		if err != nil {
 | |
| 			return api.CustomizedProfile{}, err
 | |
| 		}
 | |
| 
 | |
| 		if systemSetting.Name == api.SystemSettingCustomizedProfileName {
 | |
| 			valueMap := value.(map[string]interface{})
 | |
| 			systemStatus.CustomizedProfile = api.CustomizedProfile{}
 | |
| 			if v := valueMap["name"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.Name = v.(string)
 | |
| 			}
 | |
| 			if v := valueMap["logoUrl"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.LogoURL = v.(string)
 | |
| 			}
 | |
| 			if v := valueMap["description"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.Description = v.(string)
 | |
| 			}
 | |
| 			if v := valueMap["locale"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.Locale = v.(string)
 | |
| 			}
 | |
| 			if v := valueMap["appearance"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.Appearance = v.(string)
 | |
| 			}
 | |
| 			if v := valueMap["externalUrl"]; v != nil {
 | |
| 				systemStatus.CustomizedProfile.ExternalURL = v.(string)
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	return systemStatus.CustomizedProfile, nil
 | |
| }
 | |
| 
 | |
| func min(a, b int) int {
 | |
| 	if a < b {
 | |
| 		return a
 | |
| 	}
 | |
| 	return b
 | |
| }
 | |
| 
 | |
| func getRSSItemTitle(content string) string {
 | |
| 	var title string
 | |
| 	if isTitleDefined(content) {
 | |
| 		title = strings.Split(content, "\n")[0][2:]
 | |
| 	} else {
 | |
| 		title = strings.Split(content, "\n")[0]
 | |
| 		var titleLengthLimit = min(len(title), MaxRSSItemTitleLength)
 | |
| 		if titleLengthLimit < len(title) {
 | |
| 			title = title[:titleLengthLimit] + "..."
 | |
| 		}
 | |
| 	}
 | |
| 	return title
 | |
| }
 | |
| 
 | |
| func getRSSItemDescription(content string) string {
 | |
| 	var description string
 | |
| 	if isTitleDefined(content) {
 | |
| 		var firstLineEnd = strings.Index(content, "\n")
 | |
| 		description = strings.Trim(content[firstLineEnd+1:], " ")
 | |
| 	} else {
 | |
| 		description = content
 | |
| 	}
 | |
| 	return description
 | |
| }
 | |
| 
 | |
| func isTitleDefined(content string) bool {
 | |
| 	return strings.HasPrefix(content, "# ")
 | |
| }
 |