tag.go 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. package v1
  2. import (
  3. "encoding/json"
  4. "fmt"
  5. "net/http"
  6. "regexp"
  7. "sort"
  8. "github.com/labstack/echo/v4"
  9. "github.com/pkg/errors"
  10. "github.com/usememos/memos/api/auth"
  11. "github.com/usememos/memos/store"
  12. "golang.org/x/exp/slices"
  13. )
  14. type Tag struct {
  15. Name string
  16. CreatorID int32
  17. }
  18. type UpsertTagRequest struct {
  19. Name string `json:"name"`
  20. }
  21. type DeleteTagRequest struct {
  22. Name string `json:"name"`
  23. }
  24. func (s *APIV1Service) registerTagRoutes(g *echo.Group) {
  25. g.POST("/tag", func(c echo.Context) error {
  26. ctx := c.Request().Context()
  27. userID, ok := c.Get(auth.UserIDContextKey).(int32)
  28. if !ok {
  29. return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
  30. }
  31. tagUpsert := &UpsertTagRequest{}
  32. if err := json.NewDecoder(c.Request().Body).Decode(tagUpsert); err != nil {
  33. return echo.NewHTTPError(http.StatusBadRequest, "Malformatted post tag request").SetInternal(err)
  34. }
  35. if tagUpsert.Name == "" {
  36. return echo.NewHTTPError(http.StatusBadRequest, "Tag name shouldn't be empty")
  37. }
  38. tag, err := s.Store.UpsertTag(ctx, &store.Tag{
  39. Name: tagUpsert.Name,
  40. CreatorID: userID,
  41. })
  42. if err != nil {
  43. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to upsert tag").SetInternal(err)
  44. }
  45. tagMessage := convertTagFromStore(tag)
  46. if err := s.createTagCreateActivity(c, tagMessage); err != nil {
  47. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to create activity").SetInternal(err)
  48. }
  49. return c.JSON(http.StatusOK, tagMessage.Name)
  50. })
  51. g.GET("/tag", func(c echo.Context) error {
  52. ctx := c.Request().Context()
  53. userID, ok := c.Get(auth.UserIDContextKey).(int32)
  54. if !ok {
  55. return echo.NewHTTPError(http.StatusBadRequest, "Missing user id to find tag")
  56. }
  57. list, err := s.Store.ListTags(ctx, &store.FindTag{
  58. CreatorID: userID,
  59. })
  60. if err != nil {
  61. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find tag list").SetInternal(err)
  62. }
  63. tagNameList := []string{}
  64. for _, tag := range list {
  65. tagNameList = append(tagNameList, tag.Name)
  66. }
  67. return c.JSON(http.StatusOK, tagNameList)
  68. })
  69. g.GET("/tag/suggestion", func(c echo.Context) error {
  70. ctx := c.Request().Context()
  71. userID, ok := c.Get(auth.UserIDContextKey).(int32)
  72. if !ok {
  73. return echo.NewHTTPError(http.StatusBadRequest, "Missing user session")
  74. }
  75. normalRowStatus := store.Normal
  76. memoFind := &store.FindMemo{
  77. CreatorID: &userID,
  78. ContentSearch: []string{"#"},
  79. RowStatus: &normalRowStatus,
  80. }
  81. memoMessageList, err := s.Store.ListMemos(ctx, memoFind)
  82. if err != nil {
  83. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
  84. }
  85. list, err := s.Store.ListTags(ctx, &store.FindTag{
  86. CreatorID: userID,
  87. })
  88. if err != nil {
  89. return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find tag list").SetInternal(err)
  90. }
  91. tagNameList := []string{}
  92. for _, tag := range list {
  93. tagNameList = append(tagNameList, tag.Name)
  94. }
  95. tagMapSet := make(map[string]bool)
  96. for _, memo := range memoMessageList {
  97. for _, tag := range findTagListFromMemoContent(memo.Content) {
  98. if !slices.Contains(tagNameList, tag) {
  99. tagMapSet[tag] = true
  100. }
  101. }
  102. }
  103. tagList := []string{}
  104. for tag := range tagMapSet {
  105. tagList = append(tagList, tag)
  106. }
  107. sort.Strings(tagList)
  108. return c.JSON(http.StatusOK, tagList)
  109. })
  110. g.POST("/tag/delete", func(c echo.Context) error {
  111. ctx := c.Request().Context()
  112. userID, ok := c.Get(auth.UserIDContextKey).(int32)
  113. if !ok {
  114. return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
  115. }
  116. tagDelete := &DeleteTagRequest{}
  117. if err := json.NewDecoder(c.Request().Body).Decode(tagDelete); err != nil {
  118. return echo.NewHTTPError(http.StatusBadRequest, "Malformatted post tag request").SetInternal(err)
  119. }
  120. if tagDelete.Name == "" {
  121. return echo.NewHTTPError(http.StatusBadRequest, "Tag name shouldn't be empty")
  122. }
  123. err := s.Store.DeleteTag(ctx, &store.DeleteTag{
  124. Name: tagDelete.Name,
  125. CreatorID: userID,
  126. })
  127. if err != nil {
  128. return echo.NewHTTPError(http.StatusInternalServerError, fmt.Sprintf("Failed to delete tag name: %v", tagDelete.Name)).SetInternal(err)
  129. }
  130. return c.JSON(http.StatusOK, true)
  131. })
  132. }
  133. func (s *APIV1Service) createTagCreateActivity(c echo.Context, tag *Tag) error {
  134. ctx := c.Request().Context()
  135. payload := ActivityTagCreatePayload{
  136. TagName: tag.Name,
  137. }
  138. payloadBytes, err := json.Marshal(payload)
  139. if err != nil {
  140. return errors.Wrap(err, "failed to marshal activity payload")
  141. }
  142. activity, err := s.Store.CreateActivity(ctx, &store.Activity{
  143. CreatorID: tag.CreatorID,
  144. Type: ActivityTagCreate.String(),
  145. Level: ActivityInfo.String(),
  146. Payload: string(payloadBytes),
  147. })
  148. if err != nil || activity == nil {
  149. return errors.Wrap(err, "failed to create activity")
  150. }
  151. return err
  152. }
  153. func convertTagFromStore(tag *store.Tag) *Tag {
  154. return &Tag{
  155. Name: tag.Name,
  156. CreatorID: tag.CreatorID,
  157. }
  158. }
  159. var tagRegexp = regexp.MustCompile(`#([^\s#,]+)`)
  160. func findTagListFromMemoContent(memoContent string) []string {
  161. tagMapSet := make(map[string]bool)
  162. matches := tagRegexp.FindAllStringSubmatch(memoContent, -1)
  163. for _, v := range matches {
  164. tagName := v[1]
  165. tagMapSet[tagName] = true
  166. }
  167. tagList := []string{}
  168. for tag := range tagMapSet {
  169. tagList = append(tagList, tag)
  170. }
  171. sort.Strings(tagList)
  172. return tagList
  173. }