database.go 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425
  1. // SiYuan - Refactor your thinking
  2. // Copyright (c) 2020-present, b3log.org
  3. //
  4. // This program is free software: you can redistribute it and/or modify
  5. // it under the terms of the GNU Affero General Public License as published by
  6. // the Free Software Foundation, either version 3 of the License, or
  7. // (at your option) any later version.
  8. //
  9. // This program is distributed in the hope that it will be useful,
  10. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. // GNU Affero General Public License for more details.
  13. //
  14. // You should have received a copy of the GNU Affero General Public License
  15. // along with this program. If not, see <https://www.gnu.org/licenses/>.
  16. package sql
  17. import (
  18. "bytes"
  19. "crypto/sha256"
  20. "database/sql"
  21. "errors"
  22. "fmt"
  23. "os"
  24. "path/filepath"
  25. "regexp"
  26. "runtime"
  27. "runtime/debug"
  28. "strings"
  29. "sync"
  30. "time"
  31. "unicode/utf8"
  32. "github.com/88250/gulu"
  33. "github.com/88250/lute/ast"
  34. "github.com/88250/lute/html"
  35. "github.com/88250/lute/parse"
  36. "github.com/mattn/go-sqlite3"
  37. _ "github.com/mattn/go-sqlite3"
  38. "github.com/siyuan-note/eventbus"
  39. "github.com/siyuan-note/logging"
  40. "github.com/siyuan-note/siyuan/kernel/treenode"
  41. "github.com/siyuan-note/siyuan/kernel/util"
  42. )
  43. var (
  44. db *sql.DB
  45. historyDB *sql.DB
  46. assetContentDB *sql.DB
  47. )
  48. func init() {
  49. regex := func(re, s string) (bool, error) {
  50. re = strings.ReplaceAll(re, "\\\\", "\\")
  51. return regexp.MatchString(re, s)
  52. }
  53. sql.Register("sqlite3_extended", &sqlite3.SQLiteDriver{
  54. ConnectHook: func(conn *sqlite3.SQLiteConn) error {
  55. return conn.RegisterFunc("regexp", regex, true)
  56. },
  57. })
  58. }
  59. var initDatabaseLock = sync.Mutex{}
  60. func InitDatabase(forceRebuild bool) (err error) {
  61. initDatabaseLock.Lock()
  62. defer initDatabaseLock.Unlock()
  63. ClearCache()
  64. disableCache()
  65. defer enableCache()
  66. util.IncBootProgress(2, "Initializing database...")
  67. if forceRebuild {
  68. ClearQueue()
  69. }
  70. initDBConnection()
  71. if !forceRebuild {
  72. // 检查数据库结构版本,如果版本不一致的话说明改过表结构,需要重建
  73. if util.DatabaseVer == getDatabaseVer() {
  74. return
  75. }
  76. logging.LogInfof("the database structure is changed, rebuilding database...")
  77. }
  78. // 不存在库或者版本不一致都会走到这里
  79. closeDatabase()
  80. if gulu.File.IsExist(util.DBPath) {
  81. if err = removeDatabaseFile(); nil != err {
  82. logging.LogErrorf("remove database file [%s] failed: %s", util.DBPath, err)
  83. util.PushClearProgress()
  84. err = nil
  85. }
  86. }
  87. if gulu.File.IsExist(util.BlockTreePath) {
  88. treenode.InitBlockTree(true)
  89. }
  90. initDBConnection()
  91. initDBTables()
  92. logging.LogInfof("reinitialized database [%s]", util.DBPath)
  93. return
  94. }
  95. func initDBTables() {
  96. _, err := db.Exec("DROP TABLE IF EXISTS stat")
  97. if nil != err {
  98. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [stat] failed: %s", err)
  99. }
  100. _, err = db.Exec("CREATE TABLE stat (key, value)")
  101. if nil != err {
  102. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [stat] failed: %s", err)
  103. }
  104. setDatabaseVer()
  105. _, err = db.Exec("DROP TABLE IF EXISTS blocks")
  106. if nil != err {
  107. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [blocks] failed: %s", err)
  108. }
  109. _, err = db.Exec("CREATE TABLE blocks (id, parent_id, root_id, hash, box, path, hpath, name, alias, memo, tag, content, fcontent, markdown, length, type, subtype, ial, sort, created, updated)")
  110. if nil != err {
  111. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [blocks] failed: %s", err)
  112. }
  113. _, err = db.Exec("DROP TABLE IF EXISTS blocks_fts")
  114. if nil != err {
  115. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [blocks_fts] failed: %s", err)
  116. }
  117. _, err = db.Exec("CREATE VIRTUAL TABLE blocks_fts USING fts5(id UNINDEXED, parent_id UNINDEXED, root_id UNINDEXED, hash UNINDEXED, box UNINDEXED, path UNINDEXED, hpath, name, alias, memo, tag, content, fcontent, markdown UNINDEXED, length UNINDEXED, type UNINDEXED, subtype UNINDEXED, ial, sort UNINDEXED, created UNINDEXED, updated UNINDEXED, tokenize=\"siyuan\")")
  118. if nil != err {
  119. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [blocks_fts] failed: %s", err)
  120. }
  121. _, err = db.Exec("DROP TABLE IF EXISTS blocks_fts_case_insensitive")
  122. if nil != err {
  123. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [blocks_fts_case_insensitive] failed: %s", err)
  124. }
  125. _, err = db.Exec("CREATE VIRTUAL TABLE blocks_fts_case_insensitive USING fts5(id UNINDEXED, parent_id UNINDEXED, root_id UNINDEXED, hash UNINDEXED, box UNINDEXED, path UNINDEXED, hpath, name, alias, memo, tag, content, fcontent, markdown UNINDEXED, length UNINDEXED, type UNINDEXED, subtype UNINDEXED, ial, sort UNINDEXED, created UNINDEXED, updated UNINDEXED, tokenize=\"siyuan case_insensitive\")")
  126. if nil != err {
  127. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [blocks_fts_case_insensitive] failed: %s", err)
  128. }
  129. _, err = db.Exec("DROP TABLE IF EXISTS spans")
  130. if nil != err {
  131. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [spans] failed: %s", err)
  132. }
  133. _, err = db.Exec("CREATE TABLE spans (id, block_id, root_id, box, path, content, markdown, type, ial)")
  134. if nil != err {
  135. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [spans] failed: %s", err)
  136. }
  137. _, err = db.Exec("DROP TABLE IF EXISTS assets")
  138. if nil != err {
  139. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [assets] failed: %s", err)
  140. }
  141. _, err = db.Exec("CREATE TABLE assets (id, block_id, root_id, box, docpath, path, name, title, hash)")
  142. if nil != err {
  143. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [assets] failed: %s", err)
  144. }
  145. _, err = db.Exec("DROP TABLE IF EXISTS attributes")
  146. if nil != err {
  147. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [attributes] failed: %s", err)
  148. }
  149. _, err = db.Exec("CREATE TABLE attributes (id, name, value, type, block_id, root_id, box, path)")
  150. if nil != err {
  151. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [attributes] failed: %s", err)
  152. }
  153. _, err = db.Exec("DROP TABLE IF EXISTS refs")
  154. if nil != err {
  155. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [refs] failed: %s", err)
  156. }
  157. _, err = db.Exec("CREATE TABLE refs (id, def_block_id, def_block_parent_id, def_block_root_id, def_block_path, block_id, root_id, box, path, content, markdown, type)")
  158. if nil != err {
  159. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [refs] failed: %s", err)
  160. }
  161. _, err = db.Exec("DROP TABLE IF EXISTS file_annotation_refs")
  162. if nil != err {
  163. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "drop table [refs] failed: %s", err)
  164. }
  165. _, err = db.Exec("CREATE TABLE file_annotation_refs (id, file_path, annotation_id, block_id, root_id, box, path, content, type)")
  166. if nil != err {
  167. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [refs] failed: %s", err)
  168. }
  169. }
  170. func initDBConnection() {
  171. if nil != db {
  172. closeDatabase()
  173. }
  174. dsn := util.DBPath + "?_journal_mode=WAL" +
  175. "&_synchronous=OFF" +
  176. "&_mmap_size=2684354560" +
  177. "&_secure_delete=OFF" +
  178. "&_cache_size=-20480" +
  179. "&_page_size=32768" +
  180. "&_busy_timeout=7000" +
  181. "&_ignore_check_constraints=ON" +
  182. "&_temp_store=MEMORY" +
  183. "&_case_sensitive_like=OFF"
  184. var err error
  185. db, err = sql.Open("sqlite3_extended", dsn)
  186. if nil != err {
  187. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create database failed: %s", err)
  188. }
  189. db.SetMaxIdleConns(20)
  190. db.SetMaxOpenConns(20)
  191. db.SetConnMaxLifetime(365 * 24 * time.Hour)
  192. }
  193. var initHistoryDatabaseLock = sync.Mutex{}
  194. func InitHistoryDatabase(forceRebuild bool) {
  195. initHistoryDatabaseLock.Lock()
  196. defer initHistoryDatabaseLock.Unlock()
  197. initHistoryDBConnection()
  198. if !forceRebuild && gulu.File.IsExist(util.HistoryDBPath) {
  199. return
  200. }
  201. historyDB.Close()
  202. if err := os.RemoveAll(util.HistoryDBPath); nil != err {
  203. logging.LogErrorf("remove history database file [%s] failed: %s", util.HistoryDBPath, err)
  204. return
  205. }
  206. initHistoryDBConnection()
  207. initHistoryDBTables()
  208. }
  209. func initHistoryDBConnection() {
  210. if nil != historyDB {
  211. historyDB.Close()
  212. }
  213. dsn := util.HistoryDBPath + "?_journal_mode=WAL" +
  214. "&_synchronous=OFF" +
  215. "&_mmap_size=2684354560" +
  216. "&_secure_delete=OFF" +
  217. "&_cache_size=-20480" +
  218. "&_page_size=32768" +
  219. "&_busy_timeout=7000" +
  220. "&_ignore_check_constraints=ON" +
  221. "&_temp_store=MEMORY" +
  222. "&_case_sensitive_like=OFF"
  223. var err error
  224. historyDB, err = sql.Open("sqlite3_extended", dsn)
  225. if nil != err {
  226. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create history database failed: %s", err)
  227. }
  228. historyDB.SetMaxIdleConns(3)
  229. historyDB.SetMaxOpenConns(3)
  230. historyDB.SetConnMaxLifetime(365 * 24 * time.Hour)
  231. }
  232. func initHistoryDBTables() {
  233. historyDB.Exec("DROP TABLE histories_fts_case_insensitive")
  234. _, err := historyDB.Exec("CREATE VIRTUAL TABLE histories_fts_case_insensitive USING fts5(id UNINDEXED, type UNINDEXED, op UNINDEXED, title, content, path UNINDEXED, created UNINDEXED, tokenize=\"siyuan case_insensitive\")")
  235. if nil != err {
  236. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [histories_fts_case_insensitive] failed: %s", err)
  237. }
  238. }
  239. var initAssetContentDatabaseLock = sync.Mutex{}
  240. func InitAssetContentDatabase(forceRebuild bool) {
  241. initAssetContentDatabaseLock.Lock()
  242. defer initAssetContentDatabaseLock.Unlock()
  243. initAssetContentDBConnection()
  244. if !forceRebuild && gulu.File.IsExist(util.AssetContentDBPath) {
  245. return
  246. }
  247. assetContentDB.Close()
  248. if err := os.RemoveAll(util.AssetContentDBPath); nil != err {
  249. logging.LogErrorf("remove assets database file [%s] failed: %s", util.AssetContentDBPath, err)
  250. return
  251. }
  252. initAssetContentDBConnection()
  253. initAssetContentDBTables()
  254. }
  255. func initAssetContentDBConnection() {
  256. if nil != assetContentDB {
  257. assetContentDB.Close()
  258. }
  259. dsn := util.AssetContentDBPath + "?_journal_mode=WAL" +
  260. "&_synchronous=OFF" +
  261. "&_mmap_size=2684354560" +
  262. "&_secure_delete=OFF" +
  263. "&_cache_size=-20480" +
  264. "&_page_size=32768" +
  265. "&_busy_timeout=7000" +
  266. "&_ignore_check_constraints=ON" +
  267. "&_temp_store=MEMORY" +
  268. "&_case_sensitive_like=OFF"
  269. var err error
  270. assetContentDB, err = sql.Open("sqlite3_extended", dsn)
  271. if nil != err {
  272. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create assets database failed: %s", err)
  273. }
  274. assetContentDB.SetMaxIdleConns(3)
  275. assetContentDB.SetMaxOpenConns(3)
  276. assetContentDB.SetConnMaxLifetime(365 * 24 * time.Hour)
  277. }
  278. func initAssetContentDBTables() {
  279. assetContentDB.Exec("DROP TABLE asset_contents_fts_case_insensitive")
  280. _, err := assetContentDB.Exec("CREATE VIRTUAL TABLE asset_contents_fts_case_insensitive USING fts5(id UNINDEXED, name, ext, path, size UNINDEXED, updated UNINDEXED, content, tokenize=\"siyuan case_insensitive\")")
  281. if nil != err {
  282. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "create table [asset_contents_fts_case_insensitive] failed: %s", err)
  283. }
  284. }
  285. var (
  286. caseSensitive bool
  287. indexAssetPath bool
  288. )
  289. func SetCaseSensitive(b bool) {
  290. caseSensitive = b
  291. if b {
  292. db.Exec("PRAGMA case_sensitive_like = ON;")
  293. } else {
  294. db.Exec("PRAGMA case_sensitive_like = OFF;")
  295. }
  296. }
  297. func SetIndexAssetPath(b bool) {
  298. indexAssetPath = b
  299. }
  300. func refsFromTree(tree *parse.Tree) (refs []*Ref, fileAnnotationRefs []*FileAnnotationRef) {
  301. ast.Walk(tree.Root, func(n *ast.Node, entering bool) ast.WalkStatus {
  302. if entering {
  303. return ast.WalkContinue
  304. }
  305. if treenode.IsBlockRef(n) {
  306. ref := buildRef(tree, n)
  307. if !isRepeatedRef(refs, ref) {
  308. refs = append(refs, ref)
  309. }
  310. } else if treenode.IsFileAnnotationRef(n) {
  311. pathID := n.TextMarkFileAnnotationRefID
  312. idx := strings.LastIndex(pathID, "/")
  313. if -1 == idx {
  314. return ast.WalkContinue
  315. }
  316. filePath := pathID[:idx]
  317. annotationID := pathID[idx+1:]
  318. anchor := n.TextMarkTextContent
  319. text := filePath
  320. if "" != anchor {
  321. text = anchor
  322. }
  323. parentBlock := treenode.ParentBlock(n)
  324. ref := &FileAnnotationRef{
  325. ID: ast.NewNodeID(),
  326. FilePath: filePath,
  327. AnnotationID: annotationID,
  328. BlockID: parentBlock.ID,
  329. RootID: tree.ID,
  330. Box: tree.Box,
  331. Path: tree.Path,
  332. Content: text,
  333. Type: treenode.TypeAbbr(n.Type.String()),
  334. }
  335. fileAnnotationRefs = append(fileAnnotationRefs, ref)
  336. } else if treenode.IsEmbedBlockRef(n) {
  337. ref := buildEmbedRef(tree, n)
  338. if !isRepeatedRef(refs, ref) {
  339. refs = append(refs, ref)
  340. }
  341. }
  342. return ast.WalkContinue
  343. })
  344. return
  345. }
  346. func isRepeatedRef(refs []*Ref, ref *Ref) bool {
  347. // Repeated references to the same block within a block only count as one reference https://github.com/siyuan-note/siyuan/issues/9670
  348. for _, r := range refs {
  349. if r.DefBlockID == ref.DefBlockID && r.BlockID == ref.BlockID {
  350. return true
  351. }
  352. }
  353. return false
  354. }
  355. func buildRef(tree *parse.Tree, refNode *ast.Node) *Ref {
  356. // 多个类型可能会导致渲染的 Markdown 不正确,所以这里只保留 block-ref 类型
  357. tmpTyp := refNode.TextMarkType
  358. refNode.TextMarkType = "block-ref"
  359. markdown := treenode.ExportNodeStdMd(refNode, luteEngine)
  360. refNode.TextMarkType = tmpTyp
  361. defBlockID, text, _ := treenode.GetBlockRef(refNode)
  362. var defBlockParentID, defBlockRootID, defBlockPath string
  363. defBlock := treenode.GetBlockTree(defBlockID)
  364. if nil != defBlock {
  365. defBlockParentID = defBlock.ParentID
  366. defBlockRootID = defBlock.RootID
  367. defBlockPath = defBlock.Path
  368. }
  369. parentBlock := treenode.ParentBlock(refNode)
  370. return &Ref{
  371. ID: ast.NewNodeID(),
  372. DefBlockID: defBlockID,
  373. DefBlockParentID: defBlockParentID,
  374. DefBlockRootID: defBlockRootID,
  375. DefBlockPath: defBlockPath,
  376. BlockID: parentBlock.ID,
  377. RootID: tree.ID,
  378. Box: tree.Box,
  379. Path: tree.Path,
  380. Content: text,
  381. Markdown: markdown,
  382. Type: treenode.TypeAbbr(refNode.Type.String()),
  383. }
  384. }
  385. func buildEmbedRef(tree *parse.Tree, embedNode *ast.Node) *Ref {
  386. defBlockID := getEmbedRef(embedNode)
  387. var defBlockParentID, defBlockRootID, defBlockPath string
  388. defBlock := treenode.GetBlockTree(defBlockID)
  389. if nil != defBlock {
  390. defBlockParentID = defBlock.ParentID
  391. defBlockRootID = defBlock.RootID
  392. defBlockPath = defBlock.Path
  393. }
  394. return &Ref{
  395. ID: ast.NewNodeID(),
  396. DefBlockID: defBlockID,
  397. DefBlockParentID: defBlockParentID,
  398. DefBlockRootID: defBlockRootID,
  399. DefBlockPath: defBlockPath,
  400. BlockID: embedNode.ID,
  401. RootID: tree.ID,
  402. Box: tree.Box,
  403. Path: tree.Path,
  404. Content: "", // 通过嵌入块构建引用时定义块可能还没有入库,所以这里统一不填充内容
  405. Markdown: "",
  406. Type: treenode.TypeAbbr(embedNode.Type.String()),
  407. }
  408. }
  409. func getEmbedRef(embedNode *ast.Node) (queryBlockID string) {
  410. queryBlockID = treenode.GetEmbedBlockRef(embedNode)
  411. return
  412. }
  413. func fromTree(node *ast.Node, tree *parse.Tree) (blocks []*Block, spans []*Span, assets []*Asset, attributes []*Attribute) {
  414. rootID := tree.Root.ID
  415. boxID := tree.Box
  416. p := tree.Path
  417. ast.Walk(node, func(n *ast.Node, entering bool) ast.WalkStatus {
  418. if !entering {
  419. return ast.WalkContinue
  420. }
  421. // 构造行级元素
  422. spanBlocks, spanSpans, spanAssets, spanAttrs, walkStatus := buildSpanFromNode(n, tree, rootID, boxID, p)
  423. if 0 < len(spanBlocks) {
  424. blocks = append(blocks, spanBlocks...)
  425. }
  426. if 0 < len(spanSpans) {
  427. spans = append(spans, spanSpans...)
  428. }
  429. if 0 < len(spanAssets) {
  430. assets = append(assets, spanAssets...)
  431. }
  432. if 0 < len(spanAttrs) {
  433. attributes = append(attributes, spanAttrs...)
  434. }
  435. // 构造属性
  436. attrs := buildAttributeFromNode(n, rootID, boxID, p)
  437. if 0 < len(attrs) {
  438. attributes = append(attributes, attrs...)
  439. }
  440. if -1 != walkStatus {
  441. return walkStatus
  442. }
  443. // 构造块级元素
  444. if "" == n.ID || !n.IsBlock() {
  445. return ast.WalkContinue
  446. }
  447. b, attrs := buildBlockFromNode(n, tree)
  448. blocks = append(blocks, b)
  449. if 0 < len(attrs) {
  450. attributes = append(attributes, attrs...)
  451. }
  452. return ast.WalkContinue
  453. })
  454. return
  455. }
  456. func buildAttributeFromNode(n *ast.Node, rootID, boxID, p string) (attributes []*Attribute) {
  457. switch n.Type {
  458. case ast.NodeKramdownSpanIAL:
  459. parentBlock := treenode.ParentBlock(n)
  460. attrs := parse.IALValMap(n)
  461. for name, val := range attrs {
  462. if !isAttr(name) {
  463. continue
  464. }
  465. attr := &Attribute{
  466. ID: ast.NewNodeID(),
  467. Name: name,
  468. Value: val,
  469. Type: "s",
  470. BlockID: parentBlock.ID,
  471. RootID: rootID,
  472. Box: boxID,
  473. Path: p,
  474. }
  475. attributes = append(attributes, attr)
  476. }
  477. case ast.NodeKramdownBlockIAL:
  478. attrs := parse.IALValMap(n)
  479. for name, val := range attrs {
  480. if !isAttr(name) {
  481. continue
  482. }
  483. attr := &Attribute{
  484. ID: ast.NewNodeID(),
  485. Name: name,
  486. Value: val,
  487. Type: "b",
  488. BlockID: n.ID,
  489. RootID: rootID,
  490. Box: boxID,
  491. Path: p,
  492. }
  493. attributes = append(attributes, attr)
  494. }
  495. }
  496. return
  497. }
  498. func isAttr(name string) bool {
  499. return strings.HasPrefix(name, "custom-") || "name" == name || "alias" == name || "memo" == name || "bookmark" == name || "fold" == name || "heading-fold" == name || "style" == name
  500. }
  501. func buildSpanFromNode(n *ast.Node, tree *parse.Tree, rootID, boxID, p string) (blocks []*Block, spans []*Span, assets []*Asset, attributes []*Attribute, walkStatus ast.WalkStatus) {
  502. boxLocalPath := filepath.Join(util.DataDir, boxID)
  503. docDirLocalPath := filepath.Join(boxLocalPath, p)
  504. switch n.Type {
  505. case ast.NodeImage:
  506. text := n.Text()
  507. markdown := treenode.ExportNodeStdMd(n, luteEngine)
  508. parentBlock := treenode.ParentBlock(n)
  509. span := &Span{
  510. ID: ast.NewNodeID(),
  511. BlockID: parentBlock.ID,
  512. RootID: rootID,
  513. Box: boxID,
  514. Path: p,
  515. Content: text,
  516. Markdown: markdown,
  517. Type: treenode.TypeAbbr(n.Type.String()),
  518. IAL: treenode.IALStr(n),
  519. }
  520. spans = append(spans, span)
  521. walkStatus = ast.WalkSkipChildren
  522. destNode := n.ChildByType(ast.NodeLinkDest)
  523. if nil == destNode {
  524. return
  525. }
  526. // assetsLinkDestsInTree
  527. if !util.IsAssetLinkDest(destNode.Tokens) {
  528. return
  529. }
  530. dest := gulu.Str.FromBytes(destNode.Tokens)
  531. var title string
  532. if titleNode := n.ChildByType(ast.NodeLinkTitle); nil != titleNode {
  533. title = gulu.Str.FromBytes(titleNode.Tokens)
  534. }
  535. var hash string
  536. var hashErr error
  537. if lp := assetLocalPath(dest, boxLocalPath, docDirLocalPath); "" != lp {
  538. if !gulu.File.IsDir(lp) {
  539. hash, hashErr = util.GetEtag(lp)
  540. if nil != hashErr {
  541. logging.LogErrorf("calc asset [%s] hash failed: %s", lp, hashErr)
  542. }
  543. }
  544. }
  545. name, _ := util.LastID(dest)
  546. asset := &Asset{
  547. ID: ast.NewNodeID(),
  548. BlockID: parentBlock.ID,
  549. RootID: rootID,
  550. Box: boxID,
  551. DocPath: p,
  552. Path: dest,
  553. Name: name,
  554. Title: title,
  555. Hash: hash,
  556. }
  557. assets = append(assets, asset)
  558. return
  559. case ast.NodeTextMark:
  560. typ := treenode.TypeAbbr(n.Type.String()) + " " + n.TextMarkType
  561. text := n.Content()
  562. markdown := treenode.ExportNodeStdMd(n, luteEngine)
  563. parentBlock := treenode.ParentBlock(n)
  564. span := &Span{
  565. ID: ast.NewNodeID(),
  566. BlockID: parentBlock.ID,
  567. RootID: rootID,
  568. Box: boxID,
  569. Path: p,
  570. Content: text,
  571. Markdown: markdown,
  572. Type: typ,
  573. IAL: treenode.IALStr(n),
  574. }
  575. spans = append(spans, span)
  576. if n.IsTextMarkType("a") {
  577. dest := n.TextMarkAHref
  578. if util.IsAssetLinkDest([]byte(dest)) {
  579. var title string
  580. if titleNode := n.ChildByType(ast.NodeLinkTitle); nil != titleNode {
  581. title = gulu.Str.FromBytes(titleNode.Tokens)
  582. }
  583. var hash string
  584. var hashErr error
  585. if lp := assetLocalPath(dest, boxLocalPath, docDirLocalPath); "" != lp {
  586. if !gulu.File.IsDir(lp) {
  587. hash, hashErr = util.GetEtag(lp)
  588. if nil != hashErr {
  589. logging.LogErrorf("calc asset [%s] hash failed: %s", lp, hashErr)
  590. }
  591. }
  592. }
  593. name, _ := util.LastID(dest)
  594. asset := &Asset{
  595. ID: ast.NewNodeID(),
  596. BlockID: parentBlock.ID,
  597. RootID: rootID,
  598. Box: boxID,
  599. DocPath: p,
  600. Path: dest,
  601. Name: name,
  602. Title: title,
  603. Hash: hash,
  604. }
  605. assets = append(assets, asset)
  606. }
  607. }
  608. walkStatus = ast.WalkSkipChildren
  609. return
  610. case ast.NodeDocument:
  611. if asset := docTitleImgAsset(n); nil != asset {
  612. assets = append(assets, asset)
  613. }
  614. if tags := docTagSpans(n); 0 < len(tags) {
  615. spans = append(spans, tags...)
  616. }
  617. case ast.NodeInlineHTML, ast.NodeHTMLBlock, ast.NodeIFrame, ast.NodeWidget, ast.NodeAudio, ast.NodeVideo:
  618. nodes, err := html.ParseFragment(bytes.NewReader(n.Tokens), &html.Node{Type: html.ElementNode})
  619. if nil != err {
  620. logging.LogErrorf("parse HTML failed: %s", err)
  621. walkStatus = ast.WalkContinue
  622. return
  623. }
  624. if 1 > len(nodes) &&
  625. ast.NodeHTMLBlock != n.Type { // HTML 块若内容为空时无法在数据库中查询到 https://github.com/siyuan-note/siyuan/issues/4691
  626. walkStatus = ast.WalkContinue
  627. return
  628. }
  629. if ast.NodeHTMLBlock == n.Type || ast.NodeIFrame == n.Type || ast.NodeWidget == n.Type || ast.NodeAudio == n.Type || ast.NodeVideo == n.Type {
  630. b, attrs := buildBlockFromNode(n, tree)
  631. blocks = append(blocks, b)
  632. attributes = append(attributes, attrs...)
  633. }
  634. if ast.NodeInlineHTML == n.Type {
  635. // 没有行级 HTML,只有块级 HTML,这里转换为块
  636. b, attrs := buildBlockFromNode(n, tree)
  637. b.Type = ast.NodeHTMLBlock.String()
  638. blocks = append(blocks, b)
  639. attributes = append(attributes, attrs...)
  640. walkStatus = ast.WalkContinue
  641. return
  642. }
  643. if 1 > len(nodes) {
  644. walkStatus = ast.WalkContinue
  645. return
  646. }
  647. var src []byte
  648. for _, attr := range nodes[0].Attr {
  649. if "src" == attr.Key || "data-assets" == attr.Key || "custom-data-assets" == attr.Key {
  650. src = gulu.Str.ToBytes(attr.Val)
  651. break
  652. }
  653. }
  654. if 1 > len(src) {
  655. walkStatus = ast.WalkContinue
  656. return
  657. }
  658. if !util.IsAssetLinkDest(src) {
  659. walkStatus = ast.WalkContinue
  660. return
  661. }
  662. dest := string(src)
  663. var hash string
  664. var hashErr error
  665. if lp := assetLocalPath(dest, boxLocalPath, docDirLocalPath); "" != lp {
  666. hash, hashErr = util.GetEtag(lp)
  667. if nil != hashErr {
  668. logging.LogErrorf("calc asset [%s] hash failed: %s", lp, hashErr)
  669. }
  670. }
  671. parentBlock := treenode.ParentBlock(n)
  672. if ast.NodeInlineHTML != n.Type {
  673. parentBlock = n
  674. }
  675. name, _ := util.LastID(dest)
  676. asset := &Asset{
  677. ID: ast.NewNodeID(),
  678. BlockID: parentBlock.ID,
  679. RootID: rootID,
  680. Box: boxID,
  681. DocPath: p,
  682. Path: dest,
  683. Name: name,
  684. Title: "",
  685. Hash: hash,
  686. }
  687. assets = append(assets, asset)
  688. walkStatus = ast.WalkSkipChildren
  689. return
  690. }
  691. walkStatus = -1
  692. return
  693. }
  694. func BuildBlockFromNode(n *ast.Node, tree *parse.Tree) (block *Block) {
  695. block, _ = buildBlockFromNode(n, tree)
  696. return
  697. }
  698. func buildBlockFromNode(n *ast.Node, tree *parse.Tree) (block *Block, attributes []*Attribute) {
  699. boxID := tree.Box
  700. p := tree.Path
  701. rootID := tree.Root.ID
  702. name := html.UnescapeString(n.IALAttr("name"))
  703. alias := html.UnescapeString(n.IALAttr("alias"))
  704. memo := html.UnescapeString(n.IALAttr("memo"))
  705. tag := tagFromNode(n)
  706. var content, fcontent, markdown, parentID string
  707. ialContent := treenode.IALStr(n)
  708. hash := treenode.NodeHash(n, tree, luteEngine)
  709. var length int
  710. if ast.NodeDocument == n.Type {
  711. content = n.IALAttr("title")
  712. fcontent = content
  713. length = utf8.RuneCountInString(fcontent)
  714. } else if n.IsContainerBlock() {
  715. markdown = treenode.ExportNodeStdMd(n, luteEngine)
  716. if !treenode.IsNodeOCRed(n) {
  717. util.PushNodeOCRQueue(n)
  718. }
  719. content = treenode.NodeStaticContent(n, nil, true, indexAssetPath)
  720. fc := treenode.FirstLeafBlock(n)
  721. if !treenode.IsNodeOCRed(fc) {
  722. util.PushNodeOCRQueue(fc)
  723. }
  724. fcontent = treenode.NodeStaticContent(fc, nil, true, false)
  725. parentID = n.Parent.ID
  726. // 将标题块作为父节点
  727. if h := heading(n); nil != h {
  728. parentID = h.ID
  729. }
  730. length = utf8.RuneCountInString(fcontent)
  731. } else {
  732. markdown = treenode.ExportNodeStdMd(n, luteEngine)
  733. if !treenode.IsNodeOCRed(n) {
  734. util.PushNodeOCRQueue(n)
  735. }
  736. content = treenode.NodeStaticContent(n, nil, true, indexAssetPath)
  737. parentID = n.Parent.ID
  738. // 将标题块作为父节点
  739. if h := heading(n); nil != h {
  740. parentID = h.ID
  741. }
  742. length = utf8.RuneCountInString(content)
  743. }
  744. block = &Block{
  745. ID: n.ID,
  746. ParentID: parentID,
  747. RootID: rootID,
  748. Hash: hash,
  749. Box: boxID,
  750. Path: p,
  751. HPath: tree.HPath,
  752. Name: name,
  753. Alias: alias,
  754. Memo: memo,
  755. Tag: tag,
  756. Content: content,
  757. FContent: fcontent,
  758. Markdown: markdown,
  759. Length: length,
  760. Type: treenode.TypeAbbr(n.Type.String()),
  761. SubType: treenode.SubTypeAbbr(n),
  762. IAL: ialContent,
  763. Sort: nSort(n),
  764. Created: util.TimeFromID(n.ID),
  765. Updated: n.IALAttr("updated"),
  766. }
  767. attrs := parse.IAL2Map(n.KramdownIAL)
  768. for attrName, attrVal := range attrs {
  769. if !isAttr(attrName) {
  770. continue
  771. }
  772. attr := &Attribute{
  773. ID: ast.NewNodeID(),
  774. Name: attrName,
  775. Value: attrVal,
  776. Type: "b",
  777. BlockID: n.ID,
  778. RootID: rootID,
  779. Box: boxID,
  780. Path: p,
  781. }
  782. attributes = append(attributes, attr)
  783. }
  784. return
  785. }
  786. func tagFromNode(node *ast.Node) (ret string) {
  787. tagBuilder := bytes.Buffer{}
  788. if ast.NodeDocument == node.Type {
  789. tagIAL := html.UnescapeString(node.IALAttr("tags"))
  790. tags := strings.Split(tagIAL, ",")
  791. for _, t := range tags {
  792. t = strings.TrimSpace(t)
  793. if "" == t {
  794. continue
  795. }
  796. tagBuilder.WriteString("#")
  797. tagBuilder.WriteString(t)
  798. tagBuilder.WriteString("# ")
  799. }
  800. return strings.TrimSpace(tagBuilder.String())
  801. }
  802. ast.Walk(node, func(n *ast.Node, entering bool) ast.WalkStatus {
  803. if !entering {
  804. return ast.WalkContinue
  805. }
  806. if n.IsTextMarkType("tag") {
  807. tagBuilder.WriteString("#")
  808. tagBuilder.WriteString(n.Text())
  809. tagBuilder.WriteString("# ")
  810. }
  811. return ast.WalkContinue
  812. })
  813. return strings.TrimSpace(tagBuilder.String())
  814. }
  815. func heading(node *ast.Node) *ast.Node {
  816. currentLevel := 16
  817. if ast.NodeHeading == node.Type {
  818. currentLevel = node.HeadingLevel
  819. }
  820. for prev := node.Previous; nil != prev; prev = prev.Previous {
  821. if ast.NodeHeading == prev.Type {
  822. if prev.HeadingLevel < currentLevel {
  823. return prev
  824. }
  825. }
  826. }
  827. return nil
  828. }
  829. func deleteByBoxTx(tx *sql.Tx, box string) (err error) {
  830. if err = deleteBlocksByBoxTx(tx, box); nil != err {
  831. return
  832. }
  833. if err = deleteSpansByBoxTx(tx, box); nil != err {
  834. return
  835. }
  836. if err = deleteAssetsByBoxTx(tx, box); nil != err {
  837. return
  838. }
  839. if err = deleteAttributesByBoxTx(tx, box); nil != err {
  840. return
  841. }
  842. if err = deleteBlockRefsByBoxTx(tx, box); nil != err {
  843. return
  844. }
  845. if err = deleteFileAnnotationRefsByBoxTx(tx, box); nil != err {
  846. return
  847. }
  848. return
  849. }
  850. func deleteBlocksByIDs(tx *sql.Tx, ids []string) (err error) {
  851. in := bytes.Buffer{}
  852. in.Grow(4096)
  853. in.WriteString("(")
  854. for i, id := range ids {
  855. in.WriteString("'")
  856. in.WriteString(id)
  857. in.WriteString("'")
  858. if i < len(ids)-1 {
  859. in.WriteString(",")
  860. }
  861. removeBlockCache(id)
  862. }
  863. in.WriteString(")")
  864. stmt := "DELETE FROM blocks WHERE id IN " + in.String()
  865. if err = execStmtTx(tx, stmt); nil != err {
  866. return
  867. }
  868. stmt = "DELETE FROM blocks_fts WHERE id IN " + in.String()
  869. if err = execStmtTx(tx, stmt); nil != err {
  870. return
  871. }
  872. if !caseSensitive {
  873. stmt = "DELETE FROM blocks_fts_case_insensitive WHERE id IN " + in.String()
  874. if err = execStmtTx(tx, stmt); nil != err {
  875. return
  876. }
  877. }
  878. return
  879. }
  880. func deleteBlocksByBoxTx(tx *sql.Tx, box string) (err error) {
  881. stmt := "DELETE FROM blocks WHERE box = ?"
  882. if err = execStmtTx(tx, stmt, box); nil != err {
  883. return
  884. }
  885. stmt = "DELETE FROM blocks_fts WHERE box = ?"
  886. if err = execStmtTx(tx, stmt, box); nil != err {
  887. return
  888. }
  889. if !caseSensitive {
  890. stmt = "DELETE FROM blocks_fts_case_insensitive WHERE box = ?"
  891. if err = execStmtTx(tx, stmt, box); nil != err {
  892. return
  893. }
  894. }
  895. ClearCache()
  896. return
  897. }
  898. func deleteSpansByPathTx(tx *sql.Tx, box, path string) (err error) {
  899. stmt := "DELETE FROM spans WHERE box = ? AND path = ?"
  900. err = execStmtTx(tx, stmt, box, path)
  901. return
  902. }
  903. func deleteSpansByRootID(tx *sql.Tx, rootID string) (err error) {
  904. stmt := "DELETE FROM spans WHERE root_id =?"
  905. err = execStmtTx(tx, stmt, rootID)
  906. return
  907. }
  908. func deleteSpansByBoxTx(tx *sql.Tx, box string) (err error) {
  909. stmt := "DELETE FROM spans WHERE box = ?"
  910. err = execStmtTx(tx, stmt, box)
  911. return
  912. }
  913. func deleteAssetsByPathTx(tx *sql.Tx, box, path string) (err error) {
  914. stmt := "DELETE FROM assets WHERE box = ? AND docpath = ?"
  915. err = execStmtTx(tx, stmt, box, path)
  916. return
  917. }
  918. func deleteAttributeByBlockID(tx *sql.Tx, blockID string) (err error) {
  919. stmt := "DELETE FROM attributes WHERE block_id = ?"
  920. err = execStmtTx(tx, stmt, blockID)
  921. return
  922. }
  923. func deleteAttributesByPathTx(tx *sql.Tx, box, path string) (err error) {
  924. stmt := "DELETE FROM attributes WHERE box = ? AND path = ?"
  925. err = execStmtTx(tx, stmt, box, path)
  926. return
  927. }
  928. func deleteAssetsByBoxTx(tx *sql.Tx, box string) (err error) {
  929. stmt := "DELETE FROM assets WHERE box = ?"
  930. err = execStmtTx(tx, stmt, box)
  931. return
  932. }
  933. func deleteAttributesByBoxTx(tx *sql.Tx, box string) (err error) {
  934. stmt := "DELETE FROM attributes WHERE box = ?"
  935. err = execStmtTx(tx, stmt, box)
  936. return
  937. }
  938. func deleteRefsByPath(tx *sql.Tx, box, path string) (err error) {
  939. stmt := "DELETE FROM refs WHERE box = ? AND path = ?"
  940. err = execStmtTx(tx, stmt, box, path)
  941. return
  942. }
  943. func deleteRefsByPathTx(tx *sql.Tx, box, path string) (err error) {
  944. stmt := "DELETE FROM refs WHERE box = ? AND path = ?"
  945. err = execStmtTx(tx, stmt, box, path)
  946. return
  947. }
  948. func deleteRefsByBoxTx(tx *sql.Tx, box string) (err error) {
  949. if err = deleteFileAnnotationRefsByBoxTx(tx, box); nil != err {
  950. return
  951. }
  952. return deleteBlockRefsByBoxTx(tx, box)
  953. }
  954. func deleteBlockRefsByBoxTx(tx *sql.Tx, box string) (err error) {
  955. stmt := "DELETE FROM refs WHERE box = ?"
  956. err = execStmtTx(tx, stmt, box)
  957. return
  958. }
  959. func deleteFileAnnotationRefsByPath(tx *sql.Tx, box, path string) (err error) {
  960. stmt := "DELETE FROM file_annotation_refs WHERE box = ? AND path = ?"
  961. err = execStmtTx(tx, stmt, box, path)
  962. return
  963. }
  964. func deleteFileAnnotationRefsByPathTx(tx *sql.Tx, box, path string) (err error) {
  965. stmt := "DELETE FROM file_annotation_refs WHERE box = ? AND path = ?"
  966. err = execStmtTx(tx, stmt, box, path)
  967. return
  968. }
  969. func deleteFileAnnotationRefsByBoxTx(tx *sql.Tx, box string) (err error) {
  970. stmt := "DELETE FROM file_annotation_refs WHERE box = ?"
  971. err = execStmtTx(tx, stmt, box)
  972. return
  973. }
  974. func deleteByRootID(tx *sql.Tx, rootID string, context map[string]interface{}) (err error) {
  975. stmt := "DELETE FROM blocks WHERE root_id = ?"
  976. if err = execStmtTx(tx, stmt, rootID); nil != err {
  977. return
  978. }
  979. stmt = "DELETE FROM blocks_fts WHERE root_id = ?"
  980. if err = execStmtTx(tx, stmt, rootID); nil != err {
  981. return
  982. }
  983. if !caseSensitive {
  984. stmt = "DELETE FROM blocks_fts_case_insensitive WHERE root_id = ?"
  985. if err = execStmtTx(tx, stmt, rootID); nil != err {
  986. return
  987. }
  988. }
  989. stmt = "DELETE FROM spans WHERE root_id = ?"
  990. if err = execStmtTx(tx, stmt, rootID); nil != err {
  991. return
  992. }
  993. stmt = "DELETE FROM assets WHERE root_id = ?"
  994. if err = execStmtTx(tx, stmt, rootID); nil != err {
  995. return
  996. }
  997. stmt = "DELETE FROM refs WHERE root_id = ?"
  998. if err = execStmtTx(tx, stmt, rootID); nil != err {
  999. return
  1000. }
  1001. stmt = "DELETE FROM file_annotation_refs WHERE root_id = ?"
  1002. if err = execStmtTx(tx, stmt, rootID); nil != err {
  1003. return
  1004. }
  1005. stmt = "DELETE FROM attributes WHERE root_id = ?"
  1006. if err = execStmtTx(tx, stmt, rootID); nil != err {
  1007. return
  1008. }
  1009. ClearCache()
  1010. eventbus.Publish(eventbus.EvtSQLDeleteBlocks, context, rootID)
  1011. return
  1012. }
  1013. func batchDeleteByRootIDs(tx *sql.Tx, rootIDs []string, context map[string]interface{}) (err error) {
  1014. ids := strings.Join(rootIDs, "','")
  1015. ids = "('" + ids + "')"
  1016. stmt := "DELETE FROM blocks WHERE root_id IN " + ids
  1017. if err = execStmtTx(tx, stmt); nil != err {
  1018. return
  1019. }
  1020. stmt = "DELETE FROM blocks_fts WHERE root_id IN " + ids
  1021. if err = execStmtTx(tx, stmt); nil != err {
  1022. return
  1023. }
  1024. if !caseSensitive {
  1025. stmt = "DELETE FROM blocks_fts_case_insensitive WHERE root_id IN " + ids
  1026. if err = execStmtTx(tx, stmt); nil != err {
  1027. return
  1028. }
  1029. }
  1030. stmt = "DELETE FROM spans WHERE root_id IN " + ids
  1031. if err = execStmtTx(tx, stmt); nil != err {
  1032. return
  1033. }
  1034. stmt = "DELETE FROM assets WHERE root_id IN " + ids
  1035. if err = execStmtTx(tx, stmt); nil != err {
  1036. return
  1037. }
  1038. stmt = "DELETE FROM refs WHERE root_id IN " + ids
  1039. if err = execStmtTx(tx, stmt); nil != err {
  1040. return
  1041. }
  1042. stmt = "DELETE FROM file_annotation_refs WHERE root_id IN " + ids
  1043. if err = execStmtTx(tx, stmt); nil != err {
  1044. return
  1045. }
  1046. stmt = "DELETE FROM attributes WHERE root_id IN " + ids
  1047. if err = execStmtTx(tx, stmt); nil != err {
  1048. return
  1049. }
  1050. ClearCache()
  1051. eventbus.Publish(eventbus.EvtSQLDeleteBlocks, context, fmt.Sprintf("%d", len(rootIDs)))
  1052. return
  1053. }
  1054. func batchDeleteByPathPrefix(tx *sql.Tx, boxID, pathPrefix string) (err error) {
  1055. stmt := "DELETE FROM blocks WHERE box = ? AND path LIKE ?"
  1056. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1057. return
  1058. }
  1059. stmt = "DELETE FROM blocks_fts WHERE box = ? AND path LIKE ?"
  1060. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1061. return
  1062. }
  1063. if !caseSensitive {
  1064. stmt = "DELETE FROM blocks_fts_case_insensitive WHERE box = ? AND path LIKE ?"
  1065. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1066. return
  1067. }
  1068. }
  1069. stmt = "DELETE FROM spans WHERE box = ? AND path LIKE ?"
  1070. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1071. return
  1072. }
  1073. stmt = "DELETE FROM assets WHERE box = ? AND docpath LIKE ?"
  1074. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1075. return
  1076. }
  1077. stmt = "DELETE FROM refs WHERE box = ? AND path LIKE ?"
  1078. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1079. return
  1080. }
  1081. stmt = "DELETE FROM file_annotation_refs WHERE box = ? AND path LIKE ?"
  1082. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1083. return
  1084. }
  1085. stmt = "DELETE FROM attributes WHERE box = ? AND path LIKE ?"
  1086. if err = execStmtTx(tx, stmt, boxID, pathPrefix+"%"); nil != err {
  1087. return
  1088. }
  1089. ClearCache()
  1090. return
  1091. }
  1092. func batchUpdateHPath(tx *sql.Tx, boxID, rootID, newHPath string, context map[string]interface{}) (err error) {
  1093. stmt := "UPDATE blocks SET hpath = ? WHERE box = ? AND root_id = ?"
  1094. if err = execStmtTx(tx, stmt, newHPath, boxID, rootID); nil != err {
  1095. return
  1096. }
  1097. stmt = "UPDATE blocks_fts SET hpath = ? WHERE box = ? AND root_id = ?"
  1098. if err = execStmtTx(tx, stmt, newHPath, boxID, rootID); nil != err {
  1099. return
  1100. }
  1101. if !caseSensitive {
  1102. stmt = "UPDATE blocks_fts_case_insensitive SET hpath = ? WHERE box = ? AND root_id = ?"
  1103. if err = execStmtTx(tx, stmt, newHPath, boxID, rootID); nil != err {
  1104. return
  1105. }
  1106. }
  1107. ClearCache()
  1108. evtHash := fmt.Sprintf("%x", sha256.Sum256([]byte(rootID)))[:7]
  1109. eventbus.Publish(eventbus.EvtSQLInsertBlocksFTS, context, 1, evtHash)
  1110. return
  1111. }
  1112. func CloseDatabase() {
  1113. if err := closeDatabase(); nil != err {
  1114. logging.LogErrorf("close database failed: %s", err)
  1115. return
  1116. }
  1117. if err := historyDB.Close(); nil != err {
  1118. logging.LogErrorf("close history database failed: %s", err)
  1119. return
  1120. }
  1121. logging.LogInfof("closed database")
  1122. }
  1123. func queryRow(query string, args ...interface{}) *sql.Row {
  1124. query = strings.TrimSpace(query)
  1125. if "" == query {
  1126. logging.LogErrorf("statement is empty")
  1127. return nil
  1128. }
  1129. return db.QueryRow(query, args...)
  1130. }
  1131. func query(query string, args ...interface{}) (*sql.Rows, error) {
  1132. query = strings.TrimSpace(query)
  1133. if "" == query {
  1134. return nil, errors.New("statement is empty")
  1135. }
  1136. return db.Query(query, args...)
  1137. }
  1138. func beginTx() (tx *sql.Tx, err error) {
  1139. if tx, err = db.Begin(); nil != err {
  1140. logging.LogErrorf("begin tx failed: %s\n %s", err, logging.ShortStack())
  1141. if strings.Contains(err.Error(), "database is locked") {
  1142. os.Exit(logging.ExitCodeReadOnlyDatabase)
  1143. }
  1144. }
  1145. return
  1146. }
  1147. func commitTx(tx *sql.Tx) (err error) {
  1148. if nil == tx {
  1149. logging.LogErrorf("tx is nil")
  1150. return errors.New("tx is nil")
  1151. }
  1152. if err = tx.Commit(); nil != err {
  1153. logging.LogErrorf("commit tx failed: %s\n %s", err, logging.ShortStack())
  1154. }
  1155. return
  1156. }
  1157. func beginHistoryTx() (tx *sql.Tx, err error) {
  1158. if tx, err = historyDB.Begin(); nil != err {
  1159. logging.LogErrorf("begin history tx failed: %s\n %s", err, logging.ShortStack())
  1160. if strings.Contains(err.Error(), "database is locked") {
  1161. os.Exit(logging.ExitCodeReadOnlyDatabase)
  1162. }
  1163. }
  1164. return
  1165. }
  1166. func commitHistoryTx(tx *sql.Tx) (err error) {
  1167. if nil == tx {
  1168. logging.LogErrorf("tx is nil")
  1169. return errors.New("tx is nil")
  1170. }
  1171. if err = tx.Commit(); nil != err {
  1172. logging.LogErrorf("commit tx failed: %s\n %s", err, logging.ShortStack())
  1173. }
  1174. return
  1175. }
  1176. func beginAssetContentTx() (tx *sql.Tx, err error) {
  1177. if tx, err = assetContentDB.Begin(); nil != err {
  1178. logging.LogErrorf("begin asset content tx failed: %s\n %s", err, logging.ShortStack())
  1179. if strings.Contains(err.Error(), "database is locked") {
  1180. os.Exit(logging.ExitCodeReadOnlyDatabase)
  1181. }
  1182. }
  1183. return
  1184. }
  1185. func commitAssetContentTx(tx *sql.Tx) (err error) {
  1186. if nil == tx {
  1187. logging.LogErrorf("tx is nil")
  1188. return errors.New("tx is nil")
  1189. }
  1190. if err = tx.Commit(); nil != err {
  1191. logging.LogErrorf("commit tx failed: %s\n %s", err, logging.ShortStack())
  1192. }
  1193. return
  1194. }
  1195. func prepareExecInsertTx(tx *sql.Tx, stmtSQL string, args []interface{}) (err error) {
  1196. stmt, err := tx.Prepare(stmtSQL)
  1197. if nil != err {
  1198. return
  1199. }
  1200. if _, err = stmt.Exec(args...); nil != err {
  1201. logging.LogErrorf("exec database stmt [%s] failed: %s", stmtSQL, err)
  1202. return
  1203. }
  1204. return
  1205. }
  1206. func execStmtTx(tx *sql.Tx, stmt string, args ...interface{}) (err error) {
  1207. if _, err = tx.Exec(stmt, args...); nil != err {
  1208. if strings.Contains(err.Error(), "database disk image is malformed") {
  1209. tx.Rollback()
  1210. closeDatabase()
  1211. removeDatabaseFile()
  1212. logging.LogFatalf(logging.ExitCodeReadOnlyDatabase, "database disk image [%s] is malformed, please restart SiYuan kernel to rebuild it", util.DBPath)
  1213. }
  1214. logging.LogErrorf("exec database stmt [%s] failed: %s\n %s", stmt, err, logging.ShortStack())
  1215. return
  1216. }
  1217. return
  1218. }
  1219. func nSort(n *ast.Node) int {
  1220. switch n.Type {
  1221. // 以下为块级元素
  1222. case ast.NodeDocument:
  1223. return 0
  1224. case ast.NodeHeading:
  1225. return 5
  1226. case ast.NodeParagraph:
  1227. return 10
  1228. case ast.NodeCodeBlock:
  1229. return 10
  1230. case ast.NodeMathBlock:
  1231. return 10
  1232. case ast.NodeTable:
  1233. return 10
  1234. case ast.NodeHTMLBlock:
  1235. return 10
  1236. case ast.NodeList:
  1237. return 20
  1238. case ast.NodeListItem:
  1239. return 20
  1240. case ast.NodeBlockquote:
  1241. return 20
  1242. case ast.NodeSuperBlock:
  1243. return 30
  1244. case ast.NodeAttributeView:
  1245. return 30
  1246. case ast.NodeText, ast.NodeTextMark:
  1247. if n.IsTextMarkType("tag") {
  1248. return 205
  1249. }
  1250. return 200
  1251. }
  1252. return 100
  1253. }
  1254. func ialAttr(ial, name string) (ret string) {
  1255. idx := strings.Index(ial, name)
  1256. if 0 > idx {
  1257. return ""
  1258. }
  1259. ret = ial[idx+len(name)+2:]
  1260. ret = ret[:strings.Index(ret, "\"")]
  1261. return
  1262. }
  1263. func removeDatabaseFile() (err error) {
  1264. err = os.RemoveAll(util.DBPath)
  1265. if nil != err {
  1266. return
  1267. }
  1268. err = os.RemoveAll(util.DBPath + "-shm")
  1269. if nil != err {
  1270. return
  1271. }
  1272. err = os.RemoveAll(util.DBPath + "-wal")
  1273. if nil != err {
  1274. return
  1275. }
  1276. return
  1277. }
  1278. func closeDatabase() (err error) {
  1279. if nil == db {
  1280. return
  1281. }
  1282. err = db.Close()
  1283. debug.FreeOSMemory()
  1284. runtime.GC() // 没有这句的话文件句柄不会释放,后面就无法删除文件
  1285. return
  1286. }