mirror of
https://github.com/siyuan-note/siyuan.git
synced 2025-12-16 06:30:14 +01:00
⚡ Improve data indexing performance https://github.com/siyuan-note/siyuan/issues/16397
Signed-off-by: Daniel <845765@qq.com>
This commit is contained in:
parent
bcef9bdff9
commit
8429a1df0b
3 changed files with 67 additions and 2 deletions
|
|
@ -1743,7 +1743,6 @@ func upsertAvBlockRel(node *ast.Node) {
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
time.Sleep(100 * time.Millisecond)
|
time.Sleep(100 * time.Millisecond)
|
||||||
sql.FlushQueue()
|
|
||||||
|
|
||||||
affectedAvIDs = gulu.Str.RemoveDuplicatedElem(affectedAvIDs)
|
affectedAvIDs = gulu.Str.RemoveDuplicatedElem(affectedAvIDs)
|
||||||
var relatedAvIDs []string
|
var relatedAvIDs []string
|
||||||
|
|
@ -2090,7 +2089,7 @@ func updateRefTextRenameDoc(renamedTree *parse.Tree) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func FlushUpdateRefTextRenameDocJob() {
|
func FlushUpdateRefTextRenameDocJob() {
|
||||||
sql.FlushQueue()
|
sql.WaitFlushTx()
|
||||||
flushUpdateRefTextRenameDoc()
|
flushUpdateRefTextRenameDoc()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -20,6 +20,7 @@ import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math"
|
||||||
"path"
|
"path"
|
||||||
"runtime/debug"
|
"runtime/debug"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
@ -30,6 +31,7 @@ import (
|
||||||
"github.com/siyuan-note/eventbus"
|
"github.com/siyuan-note/eventbus"
|
||||||
"github.com/siyuan-note/logging"
|
"github.com/siyuan-note/logging"
|
||||||
"github.com/siyuan-note/siyuan/kernel/task"
|
"github.com/siyuan-note/siyuan/kernel/task"
|
||||||
|
"github.com/siyuan-note/siyuan/kernel/treenode"
|
||||||
"github.com/siyuan-note/siyuan/kernel/util"
|
"github.com/siyuan-note/siyuan/kernel/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -58,6 +60,32 @@ func FlushTxJob() {
|
||||||
task.AppendTask(task.DatabaseIndexCommit, FlushQueue)
|
task.AppendTask(task.DatabaseIndexCommit, FlushQueue)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func WaitFlushTx() {
|
||||||
|
var printLog bool
|
||||||
|
var lastPrintLog bool
|
||||||
|
for i := 0; isWritingDatabase(util.SQLFlushInterval + 50*time.Millisecond); i++ {
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
if 200 < i && !printLog { // 10s 后打日志
|
||||||
|
logging.LogWarnf("database is writing: \n%s", logging.ShortStack())
|
||||||
|
printLog = true
|
||||||
|
}
|
||||||
|
if 1200 < i && !lastPrintLog { // 60s 后打日志
|
||||||
|
logging.LogWarnf("database is still writing")
|
||||||
|
lastPrintLog = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func isWritingDatabase(d time.Duration) bool {
|
||||||
|
time.Sleep(d)
|
||||||
|
dbQueueLock.Lock()
|
||||||
|
defer dbQueueLock.Unlock()
|
||||||
|
if 0 < len(operationQueue) || flushingTx.Load() {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
func ClearQueue() {
|
func ClearQueue() {
|
||||||
dbQueueLock.Lock()
|
dbQueueLock.Lock()
|
||||||
defer dbQueueLock.Unlock()
|
defer dbQueueLock.Unlock()
|
||||||
|
|
@ -82,6 +110,32 @@ func FlushQueue() {
|
||||||
|
|
||||||
start := time.Now()
|
start := time.Now()
|
||||||
|
|
||||||
|
// logging.LogInfof("flushing database queue, total operations [%d]", total)
|
||||||
|
|
||||||
|
// 如果有重命名子树的操作,则统计各路径前缀的块树数量,数量较大的话阻塞整个队列,以便尽可能合并重命名子树的操作
|
||||||
|
var renameSubTreeOp *dbQueueOperation
|
||||||
|
for _, op := range ops {
|
||||||
|
if "rename_sub_tree" == op.action {
|
||||||
|
renameSubTreeOp = op
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if nil != renameSubTreeOp {
|
||||||
|
childCount := treenode.CountBlockTreesByPathPrefix(path.Dir(renameSubTreeOp.renameTree.Path))
|
||||||
|
if 512 < childCount {
|
||||||
|
scale := math.Log(float64(childCount)/512.0+1.0) / math.Log(2.0)
|
||||||
|
secs := 1.0 * scale
|
||||||
|
if secs < 1.0 {
|
||||||
|
secs = 1.0
|
||||||
|
}
|
||||||
|
if secs > 12.0 {
|
||||||
|
secs = 12.0
|
||||||
|
}
|
||||||
|
logging.LogInfof("rename sub tree [%s] with large child count [%d], sleep [%.2fs] to wait for more operations", renameSubTreeOp.renameTree.Path, childCount, secs)
|
||||||
|
time.Sleep(time.Duration(secs * float64(time.Second)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
context := map[string]interface{}{eventbus.CtxPushMsg: eventbus.CtxPushMsgToStatusBar}
|
context := map[string]interface{}{eventbus.CtxPushMsg: eventbus.CtxPushMsgToStatusBar}
|
||||||
if 512 < len(ops) {
|
if 512 < len(ops) {
|
||||||
disableCache()
|
disableCache()
|
||||||
|
|
|
||||||
|
|
@ -410,6 +410,18 @@ func RemoveBlockTreesByRootID(rootID string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func CountBlockTreesByPathPrefix(pathPrefix string) (ret int) {
|
||||||
|
sqlStmt := "SELECT COUNT(*) FROM blocktrees WHERE path LIKE ?"
|
||||||
|
err := db.QueryRow(sqlStmt, pathPrefix+"%").Scan(&ret)
|
||||||
|
if err != nil {
|
||||||
|
if errors.Is(err, sql.ErrNoRows) {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
logging.LogErrorf("sql query [%s] failed: %s", sqlStmt, err)
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
func GetBlockTreesByPathPrefix(pathPrefix string) (ret []*BlockTree) {
|
func GetBlockTreesByPathPrefix(pathPrefix string) (ret []*BlockTree) {
|
||||||
sqlStmt := "SELECT * FROM blocktrees WHERE path LIKE ?"
|
sqlStmt := "SELECT * FROM blocktrees WHERE path LIKE ?"
|
||||||
rows, err := db.Query(sqlStmt, pathPrefix+"%")
|
rows, err := db.Query(sqlStmt, pathPrefix+"%")
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue