mirror of
				https://github.com/usememos/memos.git
				synced 2025-10-25 22:07:19 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			196 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			196 lines
		
	
	
	
		
			5.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package server
 | |
| 
 | |
| import (
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"net/http"
 | |
| 	"net/url"
 | |
| 	"regexp"
 | |
| 	"sort"
 | |
| 
 | |
| 	"github.com/pkg/errors"
 | |
| 	"github.com/usememos/memos/api"
 | |
| 	"github.com/usememos/memos/common"
 | |
| 	"golang.org/x/exp/slices"
 | |
| 
 | |
| 	"github.com/labstack/echo/v4"
 | |
| )
 | |
| 
 | |
| func (s *Server) registerTagRoutes(g *echo.Group) {
 | |
| 	g.POST("/tag", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 		userID, ok := c.Get(getUserIDContextKey()).(int)
 | |
| 		if !ok {
 | |
| 			return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
 | |
| 		}
 | |
| 
 | |
| 		tagUpsert := &api.TagUpsert{}
 | |
| 		if err := json.NewDecoder(c.Request().Body).Decode(tagUpsert); err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Malformatted post tag request").SetInternal(err)
 | |
| 		}
 | |
| 		if tagUpsert.Name == "" {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Tag name shouldn't be empty")
 | |
| 		}
 | |
| 
 | |
| 		tagUpsert.CreatorID = userID
 | |
| 		tag, err := s.Store.UpsertTag(ctx, tagUpsert)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to upsert tag").SetInternal(err)
 | |
| 		}
 | |
| 		if err := s.createTagCreateActivity(c, tag); err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to create activity").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSONCharsetUTF8)
 | |
| 		if err := json.NewEncoder(c.Response().Writer).Encode(composeResponse(tag.Name)); err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to encode tag response").SetInternal(err)
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 
 | |
| 	g.GET("/tag", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 		userID, ok := c.Get(getUserIDContextKey()).(int)
 | |
| 		if !ok {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Missing user id to find tag")
 | |
| 		}
 | |
| 
 | |
| 		tagFind := &api.TagFind{
 | |
| 			CreatorID: userID,
 | |
| 		}
 | |
| 		tagList, err := s.Store.FindTagList(ctx, tagFind)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find tag list").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		tagNameList := []string{}
 | |
| 		for _, tag := range tagList {
 | |
| 			tagNameList = append(tagNameList, tag.Name)
 | |
| 		}
 | |
| 
 | |
| 		c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSONCharsetUTF8)
 | |
| 		if err := json.NewEncoder(c.Response().Writer).Encode(composeResponse(tagNameList)); err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to encode tags response").SetInternal(err)
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 
 | |
| 	g.GET("/tag/suggestion", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 		userID, ok := c.Get(getUserIDContextKey()).(int)
 | |
| 		if !ok {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Missing user session")
 | |
| 		}
 | |
| 		contentSearch := "#"
 | |
| 		normalRowStatus := api.Normal
 | |
| 		memoFind := api.MemoFind{
 | |
| 			CreatorID:     &userID,
 | |
| 			ContentSearch: &contentSearch,
 | |
| 			RowStatus:     &normalRowStatus,
 | |
| 		}
 | |
| 
 | |
| 		memoList, err := s.Store.FindMemoList(ctx, &memoFind)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find memo list").SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		tagFind := &api.TagFind{
 | |
| 			CreatorID: userID,
 | |
| 		}
 | |
| 		existTagList, err := s.Store.FindTagList(ctx, tagFind)
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to find tag list").SetInternal(err)
 | |
| 		}
 | |
| 		tagNameList := []string{}
 | |
| 		for _, tag := range existTagList {
 | |
| 			tagNameList = append(tagNameList, tag.Name)
 | |
| 		}
 | |
| 
 | |
| 		tagMapSet := make(map[string]bool)
 | |
| 		for _, memo := range memoList {
 | |
| 			for _, tag := range findTagListFromMemoContent(memo.Content) {
 | |
| 				if !slices.Contains(tagNameList, tag) {
 | |
| 					tagMapSet[tag] = true
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		tagList := []string{}
 | |
| 		for tag := range tagMapSet {
 | |
| 			tagList = append(tagList, tag)
 | |
| 		}
 | |
| 		sort.Strings(tagList)
 | |
| 
 | |
| 		c.Response().Header().Set(echo.HeaderContentType, echo.MIMEApplicationJSONCharsetUTF8)
 | |
| 		if err := json.NewEncoder(c.Response().Writer).Encode(composeResponse(tagList)); err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, "Failed to encode tags response").SetInternal(err)
 | |
| 		}
 | |
| 		return nil
 | |
| 	})
 | |
| 
 | |
| 	g.DELETE("/tag/:tagName", func(c echo.Context) error {
 | |
| 		ctx := c.Request().Context()
 | |
| 		userID, ok := c.Get(getUserIDContextKey()).(int)
 | |
| 		if !ok {
 | |
| 			return echo.NewHTTPError(http.StatusUnauthorized, "Missing user in session")
 | |
| 		}
 | |
| 
 | |
| 		tagName, err := url.QueryUnescape(c.Param("tagName"))
 | |
| 		if err != nil {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Invalid tag name").SetInternal(err)
 | |
| 		} else if tagName == "" {
 | |
| 			return echo.NewHTTPError(http.StatusBadRequest, "Tag name cannot be empty")
 | |
| 		}
 | |
| 
 | |
| 		tagDelete := &api.TagDelete{
 | |
| 			Name:      tagName,
 | |
| 			CreatorID: userID,
 | |
| 		}
 | |
| 		if err := s.Store.DeleteTag(ctx, tagDelete); err != nil {
 | |
| 			if common.ErrorCode(err) == common.NotFound {
 | |
| 				return echo.NewHTTPError(http.StatusNotFound, fmt.Sprintf("Tag name not found: %s", tagName))
 | |
| 			}
 | |
| 			return echo.NewHTTPError(http.StatusInternalServerError, fmt.Sprintf("Failed to delete tag name: %v", tagName)).SetInternal(err)
 | |
| 		}
 | |
| 
 | |
| 		return c.JSON(http.StatusOK, true)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| var tagRegexp = regexp.MustCompile(`#([^\s#]+)`)
 | |
| 
 | |
| func findTagListFromMemoContent(memoContent string) []string {
 | |
| 	tagMapSet := make(map[string]bool)
 | |
| 	matches := tagRegexp.FindAllStringSubmatch(memoContent, -1)
 | |
| 	for _, v := range matches {
 | |
| 		tagName := v[1]
 | |
| 		tagMapSet[tagName] = true
 | |
| 	}
 | |
| 
 | |
| 	tagList := []string{}
 | |
| 	for tag := range tagMapSet {
 | |
| 		tagList = append(tagList, tag)
 | |
| 	}
 | |
| 	sort.Strings(tagList)
 | |
| 	return tagList
 | |
| }
 | |
| 
 | |
| func (s *Server) createTagCreateActivity(c echo.Context, tag *api.Tag) error {
 | |
| 	ctx := c.Request().Context()
 | |
| 	payload := api.ActivityTagCreatePayload{
 | |
| 		TagName: tag.Name,
 | |
| 	}
 | |
| 	payloadStr, err := json.Marshal(payload)
 | |
| 	if err != nil {
 | |
| 		return errors.Wrap(err, "failed to marshal activity payload")
 | |
| 	}
 | |
| 	activity, err := s.Store.CreateActivity(ctx, &api.ActivityCreate{
 | |
| 		CreatorID: tag.CreatorID,
 | |
| 		Type:      api.ActivityTagCreate,
 | |
| 		Level:     api.ActivityInfo,
 | |
| 		Payload:   string(payloadStr),
 | |
| 	})
 | |
| 	if err != nil || activity == nil {
 | |
| 		return errors.Wrap(err, "failed to create activity")
 | |
| 	}
 | |
| 	return err
 | |
| }
 |