673 lines
16 KiB
Go
673 lines
16 KiB
Go
// 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 sql
|
|
|
|
import (
|
|
"bytes"
|
|
"database/sql"
|
|
"sort"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/88250/lute/ast"
|
|
"github.com/88250/vitess-sqlparser/sqlparser"
|
|
"github.com/emirpasic/gods/sets/hashset"
|
|
"github.com/siyuan-note/logging"
|
|
"github.com/siyuan-note/siyuan/kernel/treenode"
|
|
"github.com/siyuan-note/siyuan/kernel/util"
|
|
)
|
|
|
|
func queryBlockHashes(rootID string) (ret map[string]string) {
|
|
stmt := "SELECT id, hash FROM blocks WHERE root_id = ?"
|
|
rows, err := query(stmt, rootID)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
ret = map[string]string{}
|
|
for rows.Next() {
|
|
var id, hash string
|
|
if err = rows.Scan(&id, &hash); nil != err {
|
|
logging.LogErrorf("query scan field failed: %s", err)
|
|
return
|
|
}
|
|
ret[id] = hash
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryRootBlockByCondition(condition string) (ret []*Block) {
|
|
sqlStmt := "SELECT *, length(hpath) - length(replace(hpath, '/', '')) AS lv FROM blocks WHERE type = 'd' AND " + condition + " ORDER BY box DESC,lv ASC LIMIT 128"
|
|
rows, err := query(sqlStmt)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
var block Block
|
|
var sepCount int
|
|
if err = rows.Scan(&block.ID, &block.ParentID, &block.RootID, &block.Hash, &block.Box, &block.Path, &block.HPath, &block.Name, &block.Alias, &block.Memo, &block.Tag, &block.Content, &block.FContent, &block.Markdown, &block.Length, &block.Type, &block.SubType, &block.IAL, &block.Sort, &block.Created, &block.Updated, &sepCount); nil != err {
|
|
logging.LogErrorf("query scan field failed: %s", err)
|
|
return
|
|
}
|
|
ret = append(ret, &block)
|
|
}
|
|
return
|
|
}
|
|
|
|
func (block *Block) IsContainerBlock() bool {
|
|
switch block.Type {
|
|
case "d", "b", "l", "i", "s":
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func IsBlockFolded(id string) (ret bool) {
|
|
sqlStmt := "SELECT parent_id, ial FROM blocks WHERE id = ? AND type != 'd'"
|
|
for i := 0; i < 64; i++ {
|
|
row := queryRow(sqlStmt, id)
|
|
var pid, ial string
|
|
if err := row.Scan(&pid, &ial); nil != err {
|
|
if sql.ErrNoRows != err {
|
|
logging.LogErrorf("query scan field failed: %s", err)
|
|
}
|
|
return
|
|
}
|
|
id = pid
|
|
if strings.Contains(ial, "fold=\"1\"") {
|
|
return true
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryBlockChildrenIDs(id string) (ret []string) {
|
|
ret = append(ret, id)
|
|
childIDs := queryBlockIDByParentID(id)
|
|
for _, childID := range childIDs {
|
|
ret = append(ret, queryBlockChildrenIDs(childID)...)
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryBlockIDByParentID(parentID string) (ret []string) {
|
|
sqlStmt := "SELECT id FROM blocks WHERE parent_id = ?"
|
|
rows, err := query(sqlStmt, parentID)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
var id string
|
|
rows.Scan(&id)
|
|
ret = append(ret, id)
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryRecentUpdatedBlocks() (ret []*Block) {
|
|
sqlStmt := "SELECT * FROM blocks WHERE type = 'p' AND length > 1 ORDER BY updated DESC LIMIT 16"
|
|
if util.ContainerIOS == util.Container || util.ContainerAndroid == util.Container {
|
|
sqlStmt = "SELECT * FROM blocks WHERE type = 'd' ORDER BY updated DESC LIMIT 16"
|
|
}
|
|
rows, err := query(sqlStmt)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryBlockByNameOrAlias(rootID, text string) (ret *Block) {
|
|
sqlStmt := "SELECT * FROM blocks WHERE root_id = ? AND (alias LIKE ? OR name = ?)"
|
|
row := queryRow(sqlStmt, rootID, "%"+text+"%", text)
|
|
ret = scanBlockRow(row)
|
|
return
|
|
}
|
|
|
|
func QueryBlockAliases(rootID string) (ret []string) {
|
|
sqlStmt := "SELECT alias FROM blocks WHERE root_id = ? AND alias != ''"
|
|
rows, err := query(sqlStmt, rootID)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
var aliasesRows []string
|
|
for rows.Next() {
|
|
var name string
|
|
rows.Scan(&name)
|
|
aliasesRows = append(aliasesRows, name)
|
|
}
|
|
|
|
for _, aliasStr := range aliasesRows {
|
|
aliases := strings.Split(aliasStr, ",")
|
|
for _, alias := range aliases {
|
|
var exist bool
|
|
for _, retAlias := range ret {
|
|
if retAlias == alias {
|
|
exist = true
|
|
}
|
|
}
|
|
if !exist {
|
|
ret = append(ret, alias)
|
|
}
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryNames() (ret []string) {
|
|
ret = []string{}
|
|
sqlStmt := "SELECT name FROM blocks WHERE name != '' LIMIT ?"
|
|
rows, err := query(sqlStmt, 10240)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
var namesRows []string
|
|
for rows.Next() {
|
|
var name string
|
|
rows.Scan(&name)
|
|
namesRows = append(namesRows, name)
|
|
}
|
|
|
|
set := hashset.New()
|
|
for _, namesStr := range namesRows {
|
|
names := strings.Split(namesStr, ",")
|
|
for _, name := range names {
|
|
if "" == strings.TrimSpace(name) {
|
|
continue
|
|
}
|
|
set.Add(name)
|
|
}
|
|
}
|
|
for _, v := range set.Values() {
|
|
ret = append(ret, v.(string))
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryAliases() (ret []string) {
|
|
ret = []string{}
|
|
sqlStmt := "SELECT alias FROM blocks WHERE alias != '' LIMIT ?"
|
|
rows, err := query(sqlStmt, 10240)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
var aliasesRows []string
|
|
for rows.Next() {
|
|
var alias string
|
|
rows.Scan(&alias)
|
|
aliasesRows = append(aliasesRows, alias)
|
|
}
|
|
|
|
set := hashset.New()
|
|
for _, aliasStr := range aliasesRows {
|
|
aliases := strings.Split(aliasStr, ",")
|
|
for _, alias := range aliases {
|
|
if "" == strings.TrimSpace(alias) {
|
|
continue
|
|
}
|
|
set.Add(alias)
|
|
}
|
|
}
|
|
for _, v := range set.Values() {
|
|
ret = append(ret, v.(string))
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryDocIDsByTitle(title string, excludeIDs []string) (ret []string) {
|
|
ret = []string{}
|
|
notIn := "('" + strings.Join(excludeIDs, "','") + "')"
|
|
sqlStmt := "SELECT id FROM blocks WHERE type = 'd' AND content = ? AND id NOT IN " + notIn + " LIMIT ?"
|
|
rows, err := query(sqlStmt, title, 32)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
set := hashset.New()
|
|
for rows.Next() {
|
|
var id string
|
|
rows.Scan(&id)
|
|
set.Add(id)
|
|
}
|
|
|
|
for _, v := range set.Values() {
|
|
ret = append(ret, v.(string))
|
|
}
|
|
return
|
|
}
|
|
|
|
func queryDocTitles() (ret []string) {
|
|
ret = []string{}
|
|
sqlStmt := "SELECT content FROM blocks WHERE type = 'd' LIMIT ?"
|
|
rows, err := query(sqlStmt, 10240)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
var docNamesRows []string
|
|
for rows.Next() {
|
|
var name string
|
|
rows.Scan(&name)
|
|
docNamesRows = append(docNamesRows, name)
|
|
}
|
|
|
|
set := hashset.New()
|
|
for _, nameStr := range docNamesRows {
|
|
names := strings.Split(nameStr, ",")
|
|
for _, name := range names {
|
|
if "" == strings.TrimSpace(name) {
|
|
continue
|
|
}
|
|
set.Add(name)
|
|
}
|
|
}
|
|
for _, v := range set.Values() {
|
|
ret = append(ret, v.(string))
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryBlockNamesByRootID(rootID string) (ret []string) {
|
|
sqlStmt := "SELECT DISTINCT name FROM blocks WHERE root_id = ? AND name != ''"
|
|
rows, err := query(sqlStmt, rootID)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
var name string
|
|
rows.Scan(&name)
|
|
ret = append(ret, name)
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryBookmarkBlocksByKeyword(bookmark string) (ret []*Block) {
|
|
sqlStmt := "SELECT * FROM blocks WHERE ial LIKE ?"
|
|
rows, err := query(sqlStmt, "%bookmark=%")
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryBookmarkBlocks() (ret []*Block) {
|
|
sqlStmt := "SELECT * FROM blocks WHERE ial LIKE ?"
|
|
rows, err := query(sqlStmt, "%bookmark=%")
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func QueryBookmarkLabels() (ret []string) {
|
|
ret = []string{}
|
|
sqlStmt := "SELECT * FROM blocks WHERE ial LIKE ?"
|
|
rows, err := query(sqlStmt, "%bookmark=%")
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
labels := map[string]bool{}
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
if v := ialAttr(block.IAL, "bookmark"); "" != v {
|
|
labels[v] = true
|
|
}
|
|
}
|
|
}
|
|
|
|
for label := range labels {
|
|
ret = append(ret, label)
|
|
}
|
|
sort.Strings(ret)
|
|
return
|
|
}
|
|
|
|
func Query(stmt string) (ret []map[string]interface{}, err error) {
|
|
ret = []map[string]interface{}{}
|
|
rows, err := query(stmt)
|
|
if nil != err {
|
|
logging.LogWarnf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
cols, _ := rows.Columns()
|
|
if nil == cols {
|
|
return
|
|
}
|
|
|
|
for rows.Next() {
|
|
columns := make([]interface{}, len(cols))
|
|
columnPointers := make([]interface{}, len(cols))
|
|
for i := range columns {
|
|
columnPointers[i] = &columns[i]
|
|
}
|
|
|
|
if err = rows.Scan(columnPointers...); nil != err {
|
|
return
|
|
}
|
|
|
|
m := make(map[string]interface{})
|
|
for i, colName := range cols {
|
|
val := columnPointers[i].(*interface{})
|
|
m[colName] = *val
|
|
}
|
|
ret = append(ret, m)
|
|
}
|
|
return
|
|
}
|
|
|
|
func SelectBlocksRawStmtNoParse(stmt string, limit int) (ret []*Block) {
|
|
return selectBlocksRawStmt(stmt, limit)
|
|
}
|
|
|
|
func SelectBlocksRawStmt(stmt string, limit int) (ret []*Block) {
|
|
parsedStmt, err := sqlparser.Parse(stmt)
|
|
if nil != err {
|
|
return selectBlocksRawStmt(stmt, limit)
|
|
}
|
|
|
|
switch parsedStmt.(type) {
|
|
case *sqlparser.Select:
|
|
slct := parsedStmt.(*sqlparser.Select)
|
|
if nil == slct.Limit {
|
|
slct.Limit = &sqlparser.Limit{
|
|
Rowcount: &sqlparser.SQLVal{
|
|
Type: sqlparser.IntVal,
|
|
Val: []byte(strconv.Itoa(limit)),
|
|
},
|
|
}
|
|
}
|
|
|
|
stmt = sqlparser.String(slct)
|
|
default:
|
|
return
|
|
}
|
|
|
|
stmt = strings.ReplaceAll(stmt, "\\'", "''")
|
|
stmt = strings.ReplaceAll(stmt, "\\\"", "\"")
|
|
stmt = strings.ReplaceAll(stmt, "\\\\*", "\\*")
|
|
stmt = strings.ReplaceAll(stmt, "from dual", "")
|
|
rows, err := query(stmt)
|
|
if nil != err {
|
|
if strings.Contains(err.Error(), "syntax error") {
|
|
return
|
|
}
|
|
logging.LogWarnf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func selectBlocksRawStmt(stmt string, limit int) (ret []*Block) {
|
|
rows, err := query(stmt)
|
|
if nil != err {
|
|
if strings.Contains(err.Error(), "syntax error") {
|
|
return
|
|
}
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
|
|
confLimit := !strings.Contains(strings.ToLower(stmt), " limit ")
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
if confLimit && limit < len(ret) {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func scanBlockRows(rows *sql.Rows) (ret *Block) {
|
|
var block Block
|
|
if err := rows.Scan(&block.ID, &block.ParentID, &block.RootID, &block.Hash, &block.Box, &block.Path, &block.HPath, &block.Name, &block.Alias, &block.Memo, &block.Tag, &block.Content, &block.FContent, &block.Markdown, &block.Length, &block.Type, &block.SubType, &block.IAL, &block.Sort, &block.Created, &block.Updated); nil != err {
|
|
logging.LogErrorf("query scan field failed: %s\n%s", err, logging.ShortStack())
|
|
return
|
|
}
|
|
ret = &block
|
|
return
|
|
}
|
|
|
|
func scanBlockRow(row *sql.Row) (ret *Block) {
|
|
var block Block
|
|
if err := row.Scan(&block.ID, &block.ParentID, &block.RootID, &block.Hash, &block.Box, &block.Path, &block.HPath, &block.Name, &block.Alias, &block.Memo, &block.Tag, &block.Content, &block.FContent, &block.Markdown, &block.Length, &block.Type, &block.SubType, &block.IAL, &block.Sort, &block.Created, &block.Updated); nil != err {
|
|
if sql.ErrNoRows != err {
|
|
logging.LogErrorf("query scan field failed: %s\n%s", err, logging.ShortStack())
|
|
}
|
|
return
|
|
}
|
|
ret = &block
|
|
return
|
|
}
|
|
|
|
func GetChildBlocks(parentID, condition string) (ret []*Block) {
|
|
blockIDs := queryBlockChildrenIDs(parentID)
|
|
var params []string
|
|
for _, id := range blockIDs {
|
|
params = append(params, "\""+id+"\"")
|
|
}
|
|
|
|
ret = []*Block{}
|
|
sqlStmt := "SELECT * FROM blocks AS ref WHERE ref.id IN (" + strings.Join(params, ",") + ")"
|
|
if "" != condition {
|
|
sqlStmt += " AND " + condition
|
|
}
|
|
rows, err := query(sqlStmt)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetAllChildBlocks(rootID, condition string) (ret []*Block) {
|
|
ret = []*Block{}
|
|
sqlStmt := "SELECT * FROM blocks AS ref WHERE ref.root_id = ?"
|
|
if "" != condition {
|
|
sqlStmt += " AND " + condition
|
|
}
|
|
rows, err := query(sqlStmt, rootID)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetRefUnresolvedBlocks() (ret []*Block) {
|
|
stmt := "SELECT * FROM blocks WHERE content LIKE ?"
|
|
rows, err := query(stmt, "%ref resolve failed%")
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetRefExistedBlocks() (ret []*Block) {
|
|
stmt := "SELECT * FROM blocks WHERE markdown LIKE ? OR markdown LIKE ?"
|
|
rows, err := query(stmt, "%((20%", "%<<20%")
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetBlock(id string) (ret *Block) {
|
|
ret = getBlockCache(id)
|
|
if nil != ret {
|
|
return
|
|
}
|
|
row := queryRow("SELECT * FROM blocks WHERE id = ?", id)
|
|
ret = scanBlockRow(row)
|
|
if nil != ret {
|
|
putBlockCache(ret)
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetAllRootBlocks() (ret []*Block) {
|
|
stmt := "SELECT * FROM blocks WHERE type = 'd'"
|
|
rows, err := query(stmt)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", stmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetBlocks(ids []string) (ret []*Block) {
|
|
length := len(ids)
|
|
stmtBuilder := bytes.Buffer{}
|
|
stmtBuilder.WriteString("SELECT * FROM blocks WHERE id IN (")
|
|
var args []interface{}
|
|
for i, id := range ids {
|
|
args = append(args, id)
|
|
stmtBuilder.WriteByte('?')
|
|
if i < length-1 {
|
|
stmtBuilder.WriteByte(',')
|
|
}
|
|
}
|
|
stmtBuilder.WriteString(")")
|
|
sqlStmt := stmtBuilder.String()
|
|
rows, err := query(sqlStmt, args...)
|
|
if nil != err {
|
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
|
return
|
|
}
|
|
defer rows.Close()
|
|
for rows.Next() {
|
|
if block := scanBlockRows(rows); nil != block {
|
|
ret = append(ret, block)
|
|
putBlockCache(block)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func GetContainerText(container *ast.Node) string {
|
|
buf := &bytes.Buffer{}
|
|
buf.Grow(4096)
|
|
leaf := treenode.FirstLeafBlock(container)
|
|
ast.Walk(leaf, func(n *ast.Node, entering bool) ast.WalkStatus {
|
|
if !entering {
|
|
return ast.WalkContinue
|
|
}
|
|
switch n.Type {
|
|
case ast.NodeTagOpenMarker, ast.NodeTagCloseMarker:
|
|
buf.WriteByte('#')
|
|
case ast.NodeText, ast.NodeLinkText, ast.NodeFileAnnotationRefText, ast.NodeFootnotesRef,
|
|
ast.NodeCodeSpanContent, ast.NodeInlineMathContent, ast.NodeCodeBlockCode, ast.NodeMathBlockContent:
|
|
buf.Write(n.Tokens)
|
|
case ast.NodeTextMark:
|
|
buf.WriteString(n.Content())
|
|
case ast.NodeBlockRef:
|
|
if anchor := n.ChildByType(ast.NodeBlockRefText); nil != anchor {
|
|
buf.WriteString(anchor.Text())
|
|
} else if anchor = n.ChildByType(ast.NodeBlockRefDynamicText); nil != anchor {
|
|
buf.WriteString(anchor.Text())
|
|
} else {
|
|
text := GetRefText(n.TokensStr())
|
|
buf.WriteString(text)
|
|
}
|
|
return ast.WalkSkipChildren
|
|
}
|
|
return ast.WalkContinue
|
|
})
|
|
return buf.String()
|
|
}
|