PandaWiki/backend/handler/v1/node.go

417 lines
12 KiB
Go

package v1
import (
"errors"
"github.com/labstack/echo/v4"
v1 "github.com/chaitin/panda-wiki/api/node/v1"
"github.com/chaitin/panda-wiki/consts"
"github.com/chaitin/panda-wiki/domain"
"github.com/chaitin/panda-wiki/handler"
"github.com/chaitin/panda-wiki/log"
"github.com/chaitin/panda-wiki/middleware"
"github.com/chaitin/panda-wiki/usecase"
)
type NodeHandler struct {
*handler.BaseHandler
logger *log.Logger
usecase *usecase.NodeUsecase
auth middleware.AuthMiddleware
}
func NewNodeHandler(
baseHandler *handler.BaseHandler,
echo *echo.Echo,
usecase *usecase.NodeUsecase,
auth middleware.AuthMiddleware,
logger *log.Logger,
) *NodeHandler {
h := &NodeHandler{
BaseHandler: baseHandler,
logger: logger.WithModule("handler.v1.node"),
usecase: usecase,
auth: auth,
}
group := echo.Group("/api/v1/node", h.auth.Authorize, h.auth.ValidateKBUserPerm(consts.UserKBPermissionDocManage))
group.GET("/list", h.GetNodeList)
group.POST("", h.CreateNode)
group.GET("/detail", h.GetNodeDetail)
group.PUT("/detail", h.UpdateNodeDetail)
group.POST("/summary", h.SummaryNode)
group.POST("/action", h.NodeAction)
group.POST("/move", h.MoveNode)
group.POST("/batch_move", h.BatchMoveNode)
group.GET("/recommend_nodes", h.RecommendNodes)
group.POST("/restudy", h.NodeRestudy)
// node permission
group.GET("/permission", h.NodePermission)
group.PATCH("/permission/edit", h.NodePermissionEdit)
return h
}
// CreateNode
//
// @Summary Create Node
// @Description Create Node
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param body body domain.CreateNodeReq true "Node"
// @Success 200 {object} domain.PWResponse{data=map[string]string}
// @Router /api/v1/node [post]
func (h *NodeHandler) CreateNode(c echo.Context) error {
ctx := c.Request().Context()
authInfo := domain.GetAuthInfoFromCtx(ctx)
if authInfo == nil {
return h.NewResponseWithError(c, "authInfo not found in context", nil)
}
req := &domain.CreateNodeReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
req.MaxNode = 300
if maxNode := c.Get("max_node"); maxNode != nil {
req.MaxNode = maxNode.(int)
}
id, err := h.usecase.Create(c.Request().Context(), req, authInfo.UserId)
if err != nil {
if errors.Is(err, domain.ErrMaxNodeLimitReached) {
return h.NewResponseWithError(c, "已达到最大文档数量限制,请升级到联创版或企业版", nil)
}
return h.NewResponseWithError(c, "create node failed", err)
}
return h.NewResponseWithData(c, map[string]any{
"id": id,
})
}
// GetNodeList
//
// @Summary Get Node List
// @Description Get Node List
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param params query domain.GetNodeListReq true "Params"
// @Success 200 {object} domain.PWResponse{data=[]domain.NodeListItemResp}
// @Router /api/v1/node/list [get]
func (h *NodeHandler) GetNodeList(c echo.Context) error {
var req domain.GetNodeListReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "invalid request", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
ctx := c.Request().Context()
nodes, err := h.usecase.GetList(ctx, &req)
if err != nil {
return h.NewResponseWithError(c, "get node list failed", err)
}
return h.NewResponseWithData(c, nodes)
}
// GetNodeDetail
//
// @Summary Get Node Detail
// @Description Get Node Detail
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param param query v1.GetNodeDetailReq true "conversation id"
// @Success 200 {object} domain.PWResponse{data=v1.NodeDetailResp}
// @Router /api/v1/node/detail [get]
func (h *NodeHandler) GetNodeDetail(c echo.Context) error {
var req v1.GetNodeDetailReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "invalid request", err)
}
if err := c.Validate(&req); err != nil {
return h.NewResponseWithError(c, "validate request failed", err)
}
node, err := h.usecase.GetNodeByKBID(c.Request().Context(), req.ID, req.KbId, req.Format)
if err != nil {
return h.NewResponseWithError(c, "get node detail failed", err)
}
return h.NewResponseWithData(c, node)
}
// NodeAction
//
// @Summary Node Action
// @Description Node Action
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param action body domain.NodeActionReq true "Action"
// @Success 200 {object} domain.PWResponse{data=map[string]string}
// @Router /api/v1/node/action [post]
func (h *NodeHandler) NodeAction(c echo.Context) error {
req := &domain.NodeActionReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
ctx := c.Request().Context()
if err := h.usecase.NodeAction(ctx, req); err != nil {
return h.NewResponseWithError(c, "node action failed", err)
}
return h.NewResponseWithData(c, nil)
}
// UpdateNodeDetail
//
// @Summary Update Node Detail
// @Description Update Node Detail
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param body body domain.UpdateNodeReq true "Node"
// @Success 200 {object} domain.Response
// @Router /api/v1/node/detail [put]
func (h *NodeHandler) UpdateNodeDetail(c echo.Context) error {
ctx := c.Request().Context()
authInfo := domain.GetAuthInfoFromCtx(ctx)
if authInfo == nil {
return h.NewResponseWithError(c, "authInfo not found in context", nil)
}
req := &domain.UpdateNodeReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
if err := h.usecase.Update(ctx, req, authInfo.UserId); err != nil {
return h.NewResponseWithError(c, "update node detail failed", err)
}
return h.NewResponseWithData(c, nil)
}
// MoveNode
//
// @Summary Move Node
// @Description Move Node
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param body body domain.MoveNodeReq true "Move Node"
// @Success 200 {object} domain.Response
// @Router /api/v1/node/move [post]
func (h *NodeHandler) MoveNode(c echo.Context) error {
req := &domain.MoveNodeReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
ctx := c.Request().Context()
if err := h.usecase.MoveNode(ctx, req); err != nil {
return h.NewResponseWithError(c, "move node failed", err)
}
return h.NewResponseWithData(c, nil)
}
// SummaryNode
//
// @Summary Summary Node
// @Description Summary Node
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param body body domain.NodeSummaryReq true "Summary Node"
// @Success 200 {object} domain.Response
// @Router /api/v1/node/summary [post]
func (h *NodeHandler) SummaryNode(c echo.Context) error {
req := &domain.NodeSummaryReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
ctx := c.Request().Context()
summary, err := h.usecase.SummaryNode(ctx, req)
if err != nil {
if err == domain.ErrModelNotConfigured {
return h.NewResponseWithError(c, "请前往管理后台,点击右上角的“系统设置”配置推理大模型。", err)
}
return h.NewResponseWithError(c, "summary node failed", err)
}
return h.NewResponseWithData(c, map[string]any{
"summary": summary,
})
}
// RecommendNodes
//
// @Summary Recommend Nodes
// @Description Recommend Nodes
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param query query domain.GetRecommendNodeListReq true "Recommend Nodes"
// @Success 200 {object} domain.PWResponse{data=[]domain.RecommendNodeListResp}
// @Router /api/v1/node/recommend_nodes [get]
func (h *NodeHandler) RecommendNodes(c echo.Context) error {
var req domain.GetRecommendNodeListReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "request params is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
ctx := c.Request().Context()
nodes, err := h.usecase.GetRecommendNodeList(ctx, &req)
if err != nil {
return h.NewResponseWithError(c, "get recommend nodes failed", err)
}
return h.NewResponseWithData(c, nodes)
}
// BatchMoveNode
//
// @Summary Batch Move Node
// @Description Batch Move Node
// @Tags node
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param body body domain.BatchMoveReq true "Batch Move Node"
// @Success 200 {object} domain.Response
// @Router /api/v1/node/batch_move [post]
func (h *NodeHandler) BatchMoveNode(c echo.Context) error {
req := &domain.BatchMoveReq{}
if err := c.Bind(req); err != nil {
return h.NewResponseWithError(c, "request body is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request body failed", err)
}
ctx := c.Request().Context()
if err := h.usecase.BatchMoveNode(ctx, req); err != nil {
return h.NewResponseWithError(c, "batch move node failed", err)
}
return h.NewResponseWithData(c, nil)
}
// NodePermission 文档授权信息获取
//
// @Tags NodePermission
// @Summary 文档授权信息获取
// @Description 文档授权信息获取
// @ID v1-NodePermission
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param param query v1.NodePermissionReq true "para"
// @Success 200 {object} domain.Response{data=v1.NodePermissionResp}
// @Router /api/v1/node/permission [get]
func (h *NodeHandler) NodePermission(c echo.Context) error {
var req v1.NodePermissionReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "request params is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
ctx := c.Request().Context()
release, err := h.usecase.GetNodePermissionsByID(ctx, req.ID, req.KbId)
if err != nil {
return h.NewResponseWithError(c, "get node permission detail failed", err)
}
return h.NewResponseWithData(c, release)
}
// NodePermissionEdit 文档授权信息更新
//
// @Tags NodePermission
// @Summary 文档授权信息更新
// @Description 文档授权信息更新
// @ID v1-NodePermissionEdit
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param param body v1.NodePermissionEditReq true "para"
// @Success 200 {object} domain.Response{data=v1.NodePermissionEditResp}
// @Router /api/v1/node/permission/edit [patch]
func (h *NodeHandler) NodePermissionEdit(c echo.Context) error {
var req v1.NodePermissionEditReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "request params is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
if err := h.usecase.ValidateNodePermissionsEdit(req, consts.GetLicenseEdition(c)); err != nil {
return h.NewResponseWithError(c, "validate node permission failed", err)
}
ctx := c.Request().Context()
err := h.usecase.NodePermissionsEdit(ctx, req)
if err != nil {
return h.NewResponseWithError(c, "update node permission failed", err)
}
return h.NewResponseWithData(c, nil)
}
// NodeRestudy 文档重新学习
//
// @Tags Node
// @Summary 文档重新学习
// @Description 文档重新学习
// @ID v1-NodeRestudy
// @Accept json
// @Produce json
// @Security bearerAuth
// @Param param body v1.NodeRestudyReq true "para"
// @Success 200 {object} domain.Response{data=v1.NodeRestudyResp}
// @Router /api/v1/node/restudy [post]
func (h *NodeHandler) NodeRestudy(c echo.Context) error {
var req v1.NodeRestudyReq
if err := c.Bind(&req); err != nil {
return h.NewResponseWithError(c, "request params is invalid", err)
}
if err := c.Validate(req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
if err := h.usecase.NodeRestudy(c.Request().Context(), &req); err != nil {
return h.NewResponseWithError(c, "validate request params failed", err)
}
return h.NewResponseWithData(c, nil)
}