mirror of https://github.com/usememos/memos
feat: implement memo relation server (#1618)
parent
6e6aae6649
commit
b6564bcd77
@ -0,0 +1,19 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
type MemoRelationType string
|
||||||
|
|
||||||
|
const (
|
||||||
|
MemoRelationReference MemoRelationType = "REFERENCE"
|
||||||
|
MemoRelationAdditional MemoRelationType = "ADDITIONAL"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MemoRelation struct {
|
||||||
|
MemoID int
|
||||||
|
RelatedMemoID int
|
||||||
|
Type MemoRelationType
|
||||||
|
}
|
||||||
|
|
||||||
|
type MemoRelationUpsert struct {
|
||||||
|
RelatedMemoID int
|
||||||
|
Type MemoRelationType
|
||||||
|
}
|
@ -0,0 +1,76 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/usememos/memos/api"
|
||||||
|
"github.com/usememos/memos/store"
|
||||||
|
|
||||||
|
"github.com/labstack/echo/v4"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (s *Server) registerMemoRelationRoutes(g *echo.Group) {
|
||||||
|
g.POST("/memo/:memoId/relation", func(c echo.Context) error {
|
||||||
|
ctx := c.Request().Context()
|
||||||
|
memoID, err := strconv.Atoi(c.Param("memoId"))
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusBadRequest, fmt.Sprintf("ID is not a number: %s", c.Param("memoId"))).SetInternal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
memoRelationUpsert := &api.MemoRelationUpsert{}
|
||||||
|
if err := json.NewDecoder(c.Request().Body).Decode(memoRelationUpsert); err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusBadRequest, "Malformatted post memo relation request").SetInternal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
memoRelation, err := s.Store.UpsertMemoRelation(ctx, &store.MemoRelationMessage{
|
||||||
|
MemoID: memoID,
|
||||||
|
RelatedMemoID: memoRelationUpsert.RelatedMemoID,
|
||||||
|
Type: store.MemoRelationType(memoRelationUpsert.Type),
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to upsert memo relation").SetInternal(err)
|
||||||
|
}
|
||||||
|
return c.JSON(http.StatusOK, composeResponse(memoRelation))
|
||||||
|
})
|
||||||
|
|
||||||
|
g.GET("/memo/:memoId/relation", func(c echo.Context) error {
|
||||||
|
ctx := c.Request().Context()
|
||||||
|
memoID, err := strconv.Atoi(c.Param("memoId"))
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusBadRequest, fmt.Sprintf("ID is not a number: %s", c.Param("memoId"))).SetInternal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
memoRelationList, err := s.Store.ListMemoRelations(ctx, &store.FindMemoRelationMessage{
|
||||||
|
MemoID: &memoID,
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to list memo relations").SetInternal(err)
|
||||||
|
}
|
||||||
|
return c.JSON(http.StatusOK, composeResponse(memoRelationList))
|
||||||
|
})
|
||||||
|
|
||||||
|
g.DELETE("/memo/:memoId/relation/:relatedMemoId/type/:relationType", func(c echo.Context) error {
|
||||||
|
ctx := c.Request().Context()
|
||||||
|
memoID, err := strconv.Atoi(c.Param("memoId"))
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusBadRequest, fmt.Sprintf("Memo ID is not a number: %s", c.Param("memoId"))).SetInternal(err)
|
||||||
|
}
|
||||||
|
relatedMemoID, err := strconv.Atoi(c.Param("relatedMemoId"))
|
||||||
|
if err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusBadRequest, fmt.Sprintf("Related memo ID is not a number: %s", c.Param("resourceId"))).SetInternal(err)
|
||||||
|
}
|
||||||
|
relationType := store.MemoRelationType(c.Param("relationType"))
|
||||||
|
|
||||||
|
if err := s.Store.DeleteMemoRelation(ctx, &store.DeleteMemoRelationMessage{
|
||||||
|
MemoID: &memoID,
|
||||||
|
RelatedMemoID: &relatedMemoID,
|
||||||
|
Type: &relationType,
|
||||||
|
}); err != nil {
|
||||||
|
return echo.NewHTTPError(http.StatusInternalServerError, "Failed to delete memo relation").SetInternal(err)
|
||||||
|
}
|
||||||
|
return c.JSON(http.StatusOK, true)
|
||||||
|
})
|
||||||
|
}
|
@ -0,0 +1,97 @@
|
|||||||
|
package testserver
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"context"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/pkg/errors"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
"github.com/usememos/memos/api"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestMemoRelationServer(t *testing.T) {
|
||||||
|
ctx := context.Background()
|
||||||
|
s, err := NewTestingServer(ctx, t)
|
||||||
|
require.NoError(t, err)
|
||||||
|
defer s.Shutdown(ctx)
|
||||||
|
|
||||||
|
signup := &api.SignUp{
|
||||||
|
Username: "testuser",
|
||||||
|
Password: "testpassword",
|
||||||
|
}
|
||||||
|
user, err := s.postAuthSignup(signup)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, signup.Username, user.Username)
|
||||||
|
memoList, err := s.getMemoList()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Len(t, memoList, 0)
|
||||||
|
memo, err := s.postMemoCreate(&api.MemoCreate{
|
||||||
|
Content: "test memo",
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, "test memo", memo.Content)
|
||||||
|
memo2, err := s.postMemoCreate(&api.MemoCreate{
|
||||||
|
Content: "test memo2",
|
||||||
|
MemoRelationList: []*api.MemoRelationUpsert{
|
||||||
|
{
|
||||||
|
RelatedMemoID: memo.ID,
|
||||||
|
Type: api.MemoRelationReference,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, "test memo2", memo2.Content)
|
||||||
|
memoList, err = s.getMemoList()
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Len(t, memoList, 2)
|
||||||
|
require.Len(t, memo2.RelationList, 1)
|
||||||
|
err = s.deleteMemoRelation(memo2.ID, memo.ID, api.MemoRelationReference)
|
||||||
|
require.NoError(t, err)
|
||||||
|
memo2, err = s.getMemo(memo2.ID)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Len(t, memo2.RelationList, 0)
|
||||||
|
memoRelation, err := s.postMemoRelationUpsert(memo2.ID, &api.MemoRelationUpsert{
|
||||||
|
RelatedMemoID: memo.ID,
|
||||||
|
Type: api.MemoRelationReference,
|
||||||
|
})
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, memo.ID, memoRelation.RelatedMemoID)
|
||||||
|
memo2, err = s.getMemo(memo2.ID)
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Len(t, memo2.RelationList, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *TestingServer) postMemoRelationUpsert(memoID int, memoRelationUpsert *api.MemoRelationUpsert) (*api.MemoRelation, error) {
|
||||||
|
rawData, err := json.Marshal(&memoRelationUpsert)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "failed to marshal memo relation upsert")
|
||||||
|
}
|
||||||
|
reader := bytes.NewReader(rawData)
|
||||||
|
body, err := s.post(fmt.Sprintf("/api/memo/%d/relation", memoID), reader, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
_, err = buf.ReadFrom(body)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "fail to read response body")
|
||||||
|
}
|
||||||
|
|
||||||
|
type MemoCreateResponse struct {
|
||||||
|
Data *api.MemoRelation `json:"data"`
|
||||||
|
}
|
||||||
|
res := new(MemoCreateResponse)
|
||||||
|
if err = json.Unmarshal(buf.Bytes(), res); err != nil {
|
||||||
|
return nil, errors.Wrap(err, "fail to unmarshal post memo relation upsert response")
|
||||||
|
}
|
||||||
|
return res.Data, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (s *TestingServer) deleteMemoRelation(memoID int, relatedMemoID int, relationType api.MemoRelationType) error {
|
||||||
|
_, err := s.delete(fmt.Sprintf("/api/memo/%d/relation/%d/type/%s", memoID, relatedMemoID, relationType), nil)
|
||||||
|
return err
|
||||||
|
}
|
Loading…
Reference in New Issue