2022-05-26 15:18:53 +08:00
|
|
|
|
// SiYuan - Build Your Eternal Digital Garden
|
|
|
|
|
// Copyright (c) 2020-present, b3log.org
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
package api
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"net/http"
|
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
|
|
"github.com/88250/gulu"
|
|
|
|
|
"github.com/gin-gonic/gin"
|
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/model"
|
|
|
|
|
"github.com/siyuan-note/siyuan/kernel/util"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func findReplace(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
k := arg["k"].(string)
|
|
|
|
|
r := arg["r"].(string)
|
2022-11-28 11:54:04 +08:00
|
|
|
|
methodArg := arg["method"]
|
|
|
|
|
var method int // 0:文本,1:查询语法,2:SQL,3:正则表达式
|
|
|
|
|
if nil != methodArg {
|
|
|
|
|
method = int(methodArg.(float64))
|
|
|
|
|
}
|
2022-05-26 15:18:53 +08:00
|
|
|
|
idsArg := arg["ids"].([]interface{})
|
|
|
|
|
var ids []string
|
|
|
|
|
for _, id := range idsArg {
|
|
|
|
|
ids = append(ids, id.(string))
|
|
|
|
|
}
|
2022-11-28 11:54:04 +08:00
|
|
|
|
err := model.FindReplace(k, r, ids, method)
|
2022-05-26 15:18:53 +08:00
|
|
|
|
if nil != err {
|
|
|
|
|
ret.Code = -1
|
|
|
|
|
ret.Msg = err.Error()
|
2022-11-28 16:30:08 +08:00
|
|
|
|
ret.Data = map[string]interface{}{"closeTimeout": 5000}
|
2022-05-26 15:18:53 +08:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchAsset(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
k := arg["k"].(string)
|
|
|
|
|
ret.Data = model.SearchAssetsByName(k)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchTag(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
k := arg["k"].(string)
|
|
|
|
|
tags := model.SearchTags(k)
|
2022-12-05 10:53:00 +08:00
|
|
|
|
if 1 > len(tags) {
|
|
|
|
|
tags = []string{}
|
|
|
|
|
}
|
2022-05-26 15:18:53 +08:00
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"tags": tags,
|
|
|
|
|
"k": k,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchWidget(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyword := arg["k"].(string)
|
|
|
|
|
blocks := model.SearchWidget(keyword)
|
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"blocks": blocks,
|
|
|
|
|
"k": keyword,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-23 19:29:36 +08:00
|
|
|
|
func removeTemplate(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
path := arg["path"].(string)
|
|
|
|
|
err := model.RemoveTemplate(path)
|
|
|
|
|
if nil != err {
|
|
|
|
|
ret.Code = -1
|
|
|
|
|
ret.Msg = err.Error()
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-26 15:18:53 +08:00
|
|
|
|
func searchTemplate(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
keyword := arg["k"].(string)
|
|
|
|
|
blocks := model.SearchTemplate(keyword)
|
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"blocks": blocks,
|
|
|
|
|
"k": keyword,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchEmbedBlock(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2022-10-18 20:41:20 +08:00
|
|
|
|
embedBlockID := arg["embedBlockID"].(string)
|
2022-05-26 15:18:53 +08:00
|
|
|
|
stmt := arg["stmt"].(string)
|
|
|
|
|
excludeIDsArg := arg["excludeIDs"].([]interface{})
|
|
|
|
|
var excludeIDs []string
|
|
|
|
|
for _, excludeID := range excludeIDsArg {
|
|
|
|
|
excludeIDs = append(excludeIDs, excludeID.(string))
|
|
|
|
|
}
|
|
|
|
|
headingMode := 0 // 0:带标题下方块
|
|
|
|
|
headingModeArg := arg["headingMode"]
|
|
|
|
|
if nil != headingModeArg {
|
|
|
|
|
headingMode = int(headingModeArg.(float64))
|
|
|
|
|
}
|
2022-10-17 10:52:27 +08:00
|
|
|
|
breadcrumb := false
|
|
|
|
|
breadcrumbArg := arg["breadcrumb"]
|
|
|
|
|
if nil != breadcrumbArg {
|
|
|
|
|
breadcrumb = breadcrumbArg.(bool)
|
|
|
|
|
}
|
2022-05-26 15:18:53 +08:00
|
|
|
|
|
2022-10-18 20:41:20 +08:00
|
|
|
|
blocks := model.SearchEmbedBlock(embedBlockID, stmt, excludeIDs, headingMode, breadcrumb)
|
2022-05-26 15:18:53 +08:00
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"blocks": blocks,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchRefBlock(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-31 11:50:18 +08:00
|
|
|
|
reqId := arg["reqId"]
|
|
|
|
|
ret.Data = map[string]interface{}{"reqId": reqId}
|
|
|
|
|
if nil == arg["id"] {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-17 22:47:25 +08:00
|
|
|
|
onlyDoc := false
|
|
|
|
|
if nil != arg["onlyDoc"] {
|
|
|
|
|
onlyDoc = arg["onlyDoc"].(bool)
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-26 15:18:53 +08:00
|
|
|
|
rootID := arg["rootID"].(string)
|
|
|
|
|
id := arg["id"].(string)
|
|
|
|
|
keyword := arg["k"].(string)
|
|
|
|
|
beforeLen := int(arg["beforeLen"].(float64))
|
2023-04-17 22:47:25 +08:00
|
|
|
|
blocks, newDoc := model.SearchRefBlock(id, rootID, keyword, beforeLen, onlyDoc)
|
2022-05-26 15:18:53 +08:00
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"blocks": blocks,
|
|
|
|
|
"newDoc": newDoc,
|
2022-12-28 17:07:59 +08:00
|
|
|
|
"k": util.EscapeHTML(keyword),
|
2022-05-26 15:18:53 +08:00
|
|
|
|
"reqId": arg["reqId"],
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func fullTextSearchBlock(c *gin.Context) {
|
|
|
|
|
ret := gulu.Ret.NewResult()
|
|
|
|
|
defer c.JSON(http.StatusOK, ret)
|
|
|
|
|
|
|
|
|
|
arg, ok := util.JsonArg(c, ret)
|
|
|
|
|
if !ok {
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
query := arg["query"].(string)
|
2022-11-29 23:21:19 +08:00
|
|
|
|
pathsArg := arg["paths"]
|
|
|
|
|
var paths, boxes []string
|
|
|
|
|
if nil != pathsArg {
|
|
|
|
|
for _, p := range pathsArg.([]interface{}) {
|
|
|
|
|
path := p.(string)
|
2023-01-03 23:10:28 +08:00
|
|
|
|
box := strings.TrimSpace(strings.Split(path, "/")[0])
|
|
|
|
|
if "" != box {
|
|
|
|
|
boxes = append(boxes, box)
|
|
|
|
|
}
|
|
|
|
|
path = strings.TrimSpace(strings.TrimPrefix(path, box))
|
|
|
|
|
if "" != path {
|
|
|
|
|
paths = append(paths, path)
|
|
|
|
|
}
|
2022-11-29 23:21:19 +08:00
|
|
|
|
}
|
|
|
|
|
paths = gulu.Str.RemoveDuplicatedElem(paths)
|
|
|
|
|
boxes = gulu.Str.RemoveDuplicatedElem(boxes)
|
2022-05-26 15:18:53 +08:00
|
|
|
|
}
|
|
|
|
|
var types map[string]bool
|
|
|
|
|
if nil != arg["types"] {
|
|
|
|
|
typesArg := arg["types"].(map[string]interface{})
|
|
|
|
|
types = map[string]bool{}
|
|
|
|
|
for t, b := range typesArg {
|
|
|
|
|
types[t] = b.(bool)
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-11-28 11:24:31 +08:00
|
|
|
|
methodArg := arg["method"]
|
2022-12-09 12:08:07 +08:00
|
|
|
|
var method int // 0:关键字,1:查询语法,2:SQL,3:正则表达式
|
2022-11-28 11:24:31 +08:00
|
|
|
|
if nil != methodArg {
|
|
|
|
|
method = int(methodArg.(float64))
|
2022-05-26 15:18:53 +08:00
|
|
|
|
}
|
2022-12-02 17:30:16 +08:00
|
|
|
|
orderByArg := arg["orderBy"]
|
2022-12-09 12:08:07 +08:00
|
|
|
|
var orderBy int // 0:按块类型(默认),1:按创建时间升序,2:按创建时间降序,3:按更新时间升序,4:按更新时间降序,5:按内容顺序(仅在按文档分组时),6:按相关度升序,7:按相关度降序
|
2022-12-02 17:30:16 +08:00
|
|
|
|
if nil != orderByArg {
|
|
|
|
|
orderBy = int(orderByArg.(float64))
|
|
|
|
|
}
|
2022-11-26 18:12:54 +08:00
|
|
|
|
groupByArg := arg["groupBy"]
|
2022-11-26 22:34:28 +08:00
|
|
|
|
var groupBy int // 0:不分组,1:按文档分组
|
2022-11-26 18:12:54 +08:00
|
|
|
|
if nil != groupByArg {
|
|
|
|
|
groupBy = int(groupByArg.(float64))
|
|
|
|
|
}
|
2022-12-02 17:30:16 +08:00
|
|
|
|
blocks, matchedBlockCount, matchedRootCount := model.FullTextSearchBlock(query, boxes, paths, types, method, orderBy, groupBy)
|
2022-07-28 01:14:49 +08:00
|
|
|
|
ret.Data = map[string]interface{}{
|
|
|
|
|
"blocks": blocks,
|
|
|
|
|
"matchedBlockCount": matchedBlockCount,
|
|
|
|
|
"matchedRootCount": matchedRootCount,
|
|
|
|
|
}
|
2022-05-26 15:18:53 +08:00
|
|
|
|
}
|