// 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 . package model import ( "errors" "os" "path" "path/filepath" "sync" "github.com/88250/gulu" "github.com/88250/lute/parse" "github.com/siyuan-note/filelock" "github.com/siyuan-note/logging" "github.com/siyuan-note/siyuan/kernel/treenode" "github.com/siyuan-note/siyuan/kernel/util" ) type RecentDoc struct { RootID string `json:"rootID"` Icon string `json:"icon"` Title string `json:"title"` } var recentDocLock = sync.Mutex{} func RemoveRecentDoc(ids []string) { recentDocLock.Lock() defer recentDocLock.Unlock() recentDocs, err := getRecentDocs() if err != nil { return } ids = gulu.Str.RemoveDuplicatedElem(ids) for i, doc := range recentDocs { if gulu.Str.Contains(doc.RootID, ids) { recentDocs = append(recentDocs[:i], recentDocs[i+1:]...) break } } err = setRecentDocs(recentDocs) if err != nil { return } return } func setRecentDocByTree(tree *parse.Tree) { recentDoc := &RecentDoc{ RootID: tree.Root.ID, Icon: tree.Root.IALAttr("icon"), Title: tree.Root.IALAttr("title"), } recentDocLock.Lock() defer recentDocLock.Unlock() recentDocs, err := getRecentDocs() if err != nil { return } for i, c := range recentDocs { if c.RootID == recentDoc.RootID { recentDocs = append(recentDocs[:i], recentDocs[i+1:]...) break } } recentDocs = append([]*RecentDoc{recentDoc}, recentDocs...) if 32 < len(recentDocs) { recentDocs = recentDocs[:32] } err = setRecentDocs(recentDocs) return } func GetRecentDocs() (ret []*RecentDoc, err error) { recentDocLock.Lock() defer recentDocLock.Unlock() return getRecentDocs() } func setRecentDocs(recentDocs []*RecentDoc) (err error) { dirPath := filepath.Join(util.DataDir, "storage") if err = os.MkdirAll(dirPath, 0755); err != nil { logging.LogErrorf("create storage [recent-doc] dir failed: %s", err) return } data, err := gulu.JSON.MarshalIndentJSON(recentDocs, "", " ") if err != nil { logging.LogErrorf("marshal storage [recent-doc] failed: %s", err) return } lsPath := filepath.Join(dirPath, "recent-doc.json") err = filelock.WriteFile(lsPath, data) if err != nil { logging.LogErrorf("write storage [recent-doc] failed: %s", err) return } return } func getRecentDocs() (ret []*RecentDoc, err error) { tmp := []*RecentDoc{} dataPath := filepath.Join(util.DataDir, "storage/recent-doc.json") if !filelock.IsExist(dataPath) { return } data, err := filelock.ReadFile(dataPath) if err != nil { logging.LogErrorf("read storage [recent-doc] failed: %s", err) return } if err = gulu.JSON.UnmarshalJSON(data, &tmp); err != nil { logging.LogErrorf("unmarshal storage [recent-doc] failed: %s", err) return } var rootIDs []string for _, doc := range tmp { rootIDs = append(rootIDs, doc.RootID) } bts := treenode.GetBlockTrees(rootIDs) var notExists []string for _, doc := range tmp { if bt := bts[doc.RootID]; nil != bt { doc.Title = path.Base(bt.HPath) // Recent docs not updated after renaming https://github.com/siyuan-note/siyuan/issues/7827 ret = append(ret, doc) } else { notExists = append(notExists, doc.RootID) } } if 0 < len(notExists) { setRecentDocs(ret) } return } type Criterion struct { Name string `json:"name"` Sort int `json:"sort"` // 0:按块类型(默认),1:按创建时间升序,2:按创建时间降序,3:按更新时间升序,4:按更新时间降序,5:按内容顺序(仅在按文档分组时) Group int `json:"group"` // 0:不分组,1:按文档分组 HasReplace bool `json:"hasReplace"` // 是否有替换 Method int `json:"method"` // 0:文本,1:查询语法,2:SQL,3:正则表达式 HPath string `json:"hPath"` IDPath []string `json:"idPath"` K string `json:"k"` // 搜索关键字 R string `json:"r"` // 替换关键字 Types *CriterionTypes `json:"types"` // 类型过滤选项 ReplaceTypes *CriterionReplaceTypes `json:"replaceTypes"` // 替换类型过滤选项 } type CriterionTypes struct { MathBlock bool `json:"mathBlock"` Table bool `json:"table"` Blockquote bool `json:"blockquote"` SuperBlock bool `json:"superBlock"` Paragraph bool `json:"paragraph"` Document bool `json:"document"` Heading bool `json:"heading"` List bool `json:"list"` ListItem bool `json:"listItem"` CodeBlock bool `json:"codeBlock"` HtmlBlock bool `json:"htmlBlock"` EmbedBlock bool `json:"embedBlock"` DatabaseBlock bool `json:"databaseBlock"` AudioBlock bool `json:"audioBlock"` VideoBlock bool `json:"videoBlock"` IFrameBlock bool `json:"iframeBlock"` WidgetBlock bool `json:"widgetBlock"` } type CriterionReplaceTypes struct { Text bool `json:"text"` ImgText bool `json:"imgText"` ImgTitle bool `json:"imgTitle"` ImgSrc bool `json:"imgSrc"` AText bool `json:"aText"` ATitle bool `json:"aTitle"` AHref bool `json:"aHref"` Code bool `json:"code"` Em bool `json:"em"` Strong bool `json:"strong"` InlineMath bool `json:"inlineMath"` InlineMemo bool `json:"inlineMemo"` BlockRef bool `json:"blockRef"` FileAnnotationRef bool `json:"fileAnnotationRef"` Kbd bool `json:"kbd"` Mark bool `json:"mark"` S bool `json:"s"` Sub bool `json:"sub"` Sup bool `json:"sup"` Tag bool `json:"tag"` U bool `json:"u"` DocTitle bool `json:"docTitle"` CodeBlock bool `json:"codeBlock"` MathBlock bool `json:"mathBlock"` HtmlBlock bool `json:"htmlBlock"` } var criteriaLock = sync.Mutex{} func RemoveCriterion(name string) (err error) { criteriaLock.Lock() defer criteriaLock.Unlock() criteria, err := getCriteria() if err != nil { return } for i, c := range criteria { if c.Name == name { criteria = append(criteria[:i], criteria[i+1:]...) break } } err = setCriteria(criteria) return } func SetCriterion(criterion *Criterion) (err error) { if "" == criterion.Name { return errors.New(Conf.Language(142)) } criteriaLock.Lock() defer criteriaLock.Unlock() criteria, err := getCriteria() if err != nil { return } update := false for i, c := range criteria { if c.Name == criterion.Name { criteria[i] = criterion update = true break } } if !update { criteria = append(criteria, criterion) } err = setCriteria(criteria) return } func GetCriteria() (ret []*Criterion) { criteriaLock.Lock() defer criteriaLock.Unlock() ret, _ = getCriteria() return } func setCriteria(criteria []*Criterion) (err error) { dirPath := filepath.Join(util.DataDir, "storage") if err = os.MkdirAll(dirPath, 0755); err != nil { logging.LogErrorf("create storage [criteria] dir failed: %s", err) return } data, err := gulu.JSON.MarshalIndentJSON(criteria, "", " ") if err != nil { logging.LogErrorf("marshal storage [criteria] failed: %s", err) return } lsPath := filepath.Join(dirPath, "criteria.json") err = filelock.WriteFile(lsPath, data) if err != nil { logging.LogErrorf("write storage [criteria] failed: %s", err) return } return } func getCriteria() (ret []*Criterion, err error) { ret = []*Criterion{} dataPath := filepath.Join(util.DataDir, "storage/criteria.json") if !filelock.IsExist(dataPath) { return } data, err := filelock.ReadFile(dataPath) if err != nil { logging.LogErrorf("read storage [criteria] failed: %s", err) return } if err = gulu.JSON.UnmarshalJSON(data, &ret); err != nil { logging.LogErrorf("unmarshal storage [criteria] failed: %s", err) return } return } var localStorageLock = sync.Mutex{} func RemoveLocalStorageVals(keys []string) (err error) { localStorageLock.Lock() defer localStorageLock.Unlock() localStorage := getLocalStorage() for _, key := range keys { delete(localStorage, key) } return setLocalStorage(localStorage) } func SetLocalStorageVal(key string, val interface{}) (err error) { localStorageLock.Lock() defer localStorageLock.Unlock() localStorage := getLocalStorage() localStorage[key] = val return setLocalStorage(localStorage) } func SetLocalStorage(val interface{}) (err error) { localStorageLock.Lock() defer localStorageLock.Unlock() return setLocalStorage(val) } func GetLocalStorage() (ret map[string]interface{}) { localStorageLock.Lock() defer localStorageLock.Unlock() return getLocalStorage() } func setLocalStorage(val interface{}) (err error) { if util.ReadOnly { return } dirPath := filepath.Join(util.DataDir, "storage") if err = os.MkdirAll(dirPath, 0755); err != nil { logging.LogErrorf("create storage [local] dir failed: %s", err) return } data, err := gulu.JSON.MarshalIndentJSON(val, "", " ") if err != nil { logging.LogErrorf("marshal storage [local] failed: %s", err) return } lsPath := filepath.Join(dirPath, "local.json") err = filelock.WriteFile(lsPath, data) if err != nil { logging.LogErrorf("write storage [local] failed: %s", err) return } return } func getLocalStorage() (ret map[string]interface{}) { // When local.json is corrupted, clear the file to avoid being unable to enter the main interface https://github.com/siyuan-note/siyuan/issues/7911 ret = map[string]interface{}{} lsPath := filepath.Join(util.DataDir, "storage/local.json") if !filelock.IsExist(lsPath) { return } data, err := filelock.ReadFile(lsPath) if err != nil { logging.LogErrorf("read storage [local] failed: %s", err) return } if err = gulu.JSON.UnmarshalJSON(data, &ret); err != nil { logging.LogErrorf("unmarshal storage [local] failed: %s", err) return } return }