123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490 |
- // SiYuan - Refactor your thinking
- // 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 listInvalidBlockRefs(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- page := 1
- if nil != arg["page"] {
- page = int(arg["page"].(float64))
- }
- if 0 >= page {
- page = 1
- }
- pageSize := 32
- if nil != arg["pageSize"] {
- pageSize = int(arg["pageSize"].(float64))
- }
- if 0 >= pageSize {
- pageSize = 32
- }
- blocks, matchedBlockCount, matchedRootCount, pageCount := model.ListInvalidBlockRefs(page, pageSize)
- ret.Data = map[string]interface{}{
- "blocks": blocks,
- "matchedBlockCount": matchedBlockCount,
- "matchedRootCount": matchedRootCount,
- "pageCount": pageCount,
- }
- }
- func getAssetContent(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- id := arg["id"].(string)
- query := arg["query"].(string)
- queryMethod := int(arg["queryMethod"].(float64))
- ret.Data = map[string]interface{}{
- "assetContent": model.GetAssetContent(id, query, queryMethod),
- }
- return
- }
- func fullTextSearchAssetContent(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- if !model.IsPaidUser() {
- ret.Code = 1
- return
- }
- page, pageSize, query, types, method, orderBy := parseSearchAssetContentArgs(arg)
- assetContents, matchedAssetCount, pageCount := model.FullTextSearchAssetContent(query, types, method, orderBy, page, pageSize)
- ret.Data = map[string]interface{}{
- "assetContents": assetContents,
- "matchedAssetCount": matchedAssetCount,
- "pageCount": pageCount,
- }
- }
- func findReplace(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- _, _, _, paths, boxes, types, method, orderBy, groupBy := parseSearchBlockArgs(arg)
- k := arg["k"].(string)
- r := arg["r"].(string)
- idsArg := arg["ids"].([]interface{})
- var ids []string
- for _, id := range idsArg {
- ids = append(ids, id.(string))
- }
- replaceTypes := map[string]bool{}
- // text, imgText, imgTitle, imgSrc, aText, aTitle, aHref, code, em, strong, inlineMath, inlineMemo, kbd, mark, s, sub, sup, tag, u
- // docTitle, codeBlock, mathBlock, htmlBlock
- if nil != arg["replaceTypes"] {
- replaceTypesArg := arg["replaceTypes"].(map[string]interface{})
- for t, b := range replaceTypesArg {
- replaceTypes[t] = b.(bool)
- }
- }
- err := model.FindReplace(k, r, replaceTypes, ids, paths, boxes, types, method, orderBy, groupBy)
- if nil != err {
- ret.Code = 1
- ret.Msg = err.Error()
- ret.Data = map[string]interface{}{"closeTimeout": 5000}
- 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)
- var exts []string
- if extsArg := arg["exts"]; nil != extsArg {
- for _, ext := range extsArg.([]interface{}) {
- exts = append(exts, ext.(string))
- }
- }
- ret.Data = model.SearchAssetsByName(k, exts)
- 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)
- if 1 > len(tags) {
- tags = []string{}
- }
- 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,
- }
- }
- 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
- }
- }
- 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 getEmbedBlock(c *gin.Context) {
- // Query embed block supports executing JavaScript https://github.com/siyuan-note/siyuan/issues/9648
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- embedBlockID := arg["embedBlockID"].(string)
- includeIDsArg := arg["includeIDs"].([]interface{})
- var includeIDs []string
- for _, includeID := range includeIDsArg {
- includeIDs = append(includeIDs, includeID.(string))
- }
- headingMode := 0 // 0:带标题下方块
- headingModeArg := arg["headingMode"]
- if nil != headingModeArg {
- headingMode = int(headingModeArg.(float64))
- }
- breadcrumb := false
- breadcrumbArg := arg["breadcrumb"]
- if nil != breadcrumbArg {
- breadcrumb = breadcrumbArg.(bool)
- }
- blocks := model.GetEmbedBlock(embedBlockID, includeIDs, headingMode, breadcrumb)
- ret.Data = map[string]interface{}{
- "blocks": blocks,
- }
- }
- func updateEmbedBlock(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- id := arg["id"].(string)
- content := arg["content"].(string)
- err := model.UpdateEmbedBlock(id, content)
- if nil != err {
- ret.Code = -1
- ret.Msg = err.Error()
- return
- }
- }
- func searchEmbedBlock(c *gin.Context) {
- ret := gulu.Ret.NewResult()
- defer c.JSON(http.StatusOK, ret)
- arg, ok := util.JsonArg(c, ret)
- if !ok {
- return
- }
- embedBlockID := arg["embedBlockID"].(string)
- 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))
- }
- breadcrumb := false
- breadcrumbArg := arg["breadcrumb"]
- if nil != breadcrumbArg {
- breadcrumb = breadcrumbArg.(bool)
- }
- blocks := model.SearchEmbedBlock(embedBlockID, stmt, excludeIDs, headingMode, breadcrumb)
- 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
- }
- reqId := arg["reqId"]
- ret.Data = map[string]interface{}{"reqId": reqId}
- if nil == arg["id"] {
- return
- }
- isSquareBrackets := false
- if isSquareBracketsArg := arg["isSquareBrackets"]; nil != isSquareBracketsArg {
- isSquareBrackets = isSquareBracketsArg.(bool)
- }
- isDatabase := false
- if isDatabaseArg := arg["isDatabase"]; nil != isDatabaseArg {
- isDatabase = isDatabaseArg.(bool)
- }
- rootID := arg["rootID"].(string)
- id := arg["id"].(string)
- keyword := arg["k"].(string)
- beforeLen := int(arg["beforeLen"].(float64))
- blocks, newDoc := model.SearchRefBlock(id, rootID, keyword, beforeLen, isSquareBrackets, isDatabase)
- ret.Data = map[string]interface{}{
- "blocks": blocks,
- "newDoc": newDoc,
- "k": util.EscapeHTML(keyword),
- "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
- }
- page, pageSize, query, paths, boxes, types, method, orderBy, groupBy := parseSearchBlockArgs(arg)
- blocks, matchedBlockCount, matchedRootCount, pageCount := model.FullTextSearchBlock(query, boxes, paths, types, method, orderBy, groupBy, page, pageSize)
- ret.Data = map[string]interface{}{
- "blocks": blocks,
- "matchedBlockCount": matchedBlockCount,
- "matchedRootCount": matchedRootCount,
- "pageCount": pageCount,
- }
- }
- func parseSearchBlockArgs(arg map[string]interface{}) (page, pageSize int, query string, paths, boxes []string, types map[string]bool, method, orderBy, groupBy int) {
- page = 1
- if nil != arg["page"] {
- page = int(arg["page"].(float64))
- }
- if 0 >= page {
- page = 1
- }
- pageSize = 32
- if nil != arg["pageSize"] {
- pageSize = int(arg["pageSize"].(float64))
- }
- if 0 >= pageSize {
- pageSize = 32
- }
- queryArg := arg["query"]
- if nil != queryArg {
- query = queryArg.(string)
- }
- pathsArg := arg["paths"]
- if nil != pathsArg {
- for _, p := range pathsArg.([]interface{}) {
- path := p.(string)
- 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)
- }
- }
- paths = gulu.Str.RemoveDuplicatedElem(paths)
- boxes = gulu.Str.RemoveDuplicatedElem(boxes)
- }
- if nil != arg["types"] {
- typesArg := arg["types"].(map[string]interface{})
- types = map[string]bool{}
- for t, b := range typesArg {
- types[t] = b.(bool)
- }
- }
- // method:0:关键字,1:查询语法,2:SQL,3:正则表达式
- methodArg := arg["method"]
- if nil != methodArg {
- method = int(methodArg.(float64))
- }
- // orderBy:0:按块类型(默认),1:按创建时间升序,2:按创建时间降序,3:按更新时间升序,4:按更新时间降序,5:按内容顺序(仅在按文档分组时),6:按相关度升序,7:按相关度降序
- orderByArg := arg["orderBy"]
- if nil != orderByArg {
- orderBy = int(orderByArg.(float64))
- }
- // groupBy: 0:不分组,1:按文档分组
- groupByArg := arg["groupBy"]
- if nil != groupByArg {
- groupBy = int(groupByArg.(float64))
- }
- return
- }
- func parseSearchAssetContentArgs(arg map[string]interface{}) (page, pageSize int, query string, types map[string]bool, method, orderBy int) {
- page = 1
- if nil != arg["page"] {
- page = int(arg["page"].(float64))
- }
- if 0 >= page {
- page = 1
- }
- pageSize = 32
- if nil != arg["pageSize"] {
- pageSize = int(arg["pageSize"].(float64))
- }
- if 0 >= pageSize {
- pageSize = 32
- }
- queryArg := arg["query"]
- if nil != queryArg {
- query = queryArg.(string)
- }
- if nil != arg["types"] {
- typesArg := arg["types"].(map[string]interface{})
- types = map[string]bool{}
- for t, b := range typesArg {
- types[t] = b.(bool)
- }
- }
- // method:0:关键字,1:查询语法,2:SQL,3:正则表达式
- methodArg := arg["method"]
- if nil != methodArg {
- method = int(methodArg.(float64))
- }
- // orderBy:0:按相关度降序,1:按相关度升序,2:按更新时间升序,3:按更新时间降序
- orderByArg := arg["orderBy"]
- if nil != orderByArg {
- orderBy = int(orderByArg.(float64))
- }
- return
- }
|