1
0
mirror of https://github.com/osmarks/mycorrhiza.git synced 2025-01-21 15:56:50 +00:00

Implement async backindex update

This commit is contained in:
Mikhail Chekan 2021-09-01 14:28:21 +08:00 committed by Timur Ismagilov
parent b469b8f57d
commit 1c1260569b
2 changed files with 81 additions and 37 deletions

View File

@ -2,7 +2,6 @@ package hyphae
import ( import (
"os" "os"
"sync"
"github.com/bouncepaw/mycorrhiza/util" "github.com/bouncepaw/mycorrhiza/util"
@ -34,8 +33,66 @@ func fetchText(h *Hypha) string {
return "" return ""
} }
// We'll use a quasi-union type for proper async changes
type BackIndexOperation interface {
Apply()
}
type BackIndexEditing struct {
Name string
OldLinks []string
NewLinks []string
}
func (op BackIndexEditing) Apply() {
oldLinks := toLinkSet(op.OldLinks)
newLinks := toLinkSet(op.NewLinks)
for link := range oldLinks {
if _, exists := newLinks[link]; !exists {
delete(backlinkIndex[link], op.Name)
}
}
for link := range newLinks {
if _, exists := oldLinks[link]; !exists {
if _, exists := backlinkIndex[link]; !exists {
backlinkIndex[link] = make(linkSet)
}
backlinkIndex[link][op.Name] = struct{}{}
}
}
}
type BackIndexDeletion struct {
Name string
Links []string
}
func (op BackIndexDeletion) Apply() {
for _, link := range op.Links {
if lSet, exists := backlinkIndex[link]; exists {
delete(lSet, op.Name)
}
}
}
type BackIndexRenaming struct {
OldName string
NewName string
Links []string
}
func (op BackIndexRenaming) Apply() {
for _, link := range op.Links {
if lSet, exists := backlinkIndex[link]; exists {
delete(lSet, op.OldName)
backlinkIndex[link][op.NewName] = struct{}{}
}
}
}
var backlinkIndex = make(map[string]linkSet) var backlinkIndex = make(map[string]linkSet)
var backlinkIndexMutex = sync.Mutex{} var backlinkConveyor = make(chan BackIndexOperation, 64)
// I hope, the buffer size is enough -- chekoopa
// IndexBacklinks traverses all text hyphae, extracts links from them and forms an initial index // IndexBacklinks traverses all text hyphae, extracts links from them and forms an initial index
func IndexBacklinks() { func IndexBacklinks() {
@ -52,6 +109,16 @@ func IndexBacklinks() {
} }
} }
// RunBacklinksConveyor runs an index operation processing loop
func RunBacklinksConveyor() {
// It is supposed to run as a goroutine for all the time. So, don't blame the infinite loop.
defer close(backlinkConveyor)
for {
(<-backlinkConveyor).Apply()
}
}
// BacklinksCount return an amount of backlinks for a provided hypha
func BacklinksCount(h *Hypha) int { func BacklinksCount(h *Hypha) int {
if _, exists := backlinkIndex[h.Name]; exists { if _, exists := backlinkIndex[h.Name]; exists {
return len(backlinkIndex[h.Name]) return len(backlinkIndex[h.Name])
@ -59,47 +126,23 @@ func BacklinksCount(h *Hypha) int {
return 0 return 0
} }
// BacklinksOnEdit is a creation/editing hook for backlinks index
func BacklinksOnEdit(h *Hypha, oldText string) { func BacklinksOnEdit(h *Hypha, oldText string) {
backlinkIndexMutex.Lock()
newLinks := toLinkSet(ExtractHyphaLinks(h))
oldLinks := toLinkSet(ExtractHyphaLinksFromContent(h.Name, oldText))
for link := range oldLinks {
if _, exists := newLinks[link]; !exists {
delete(backlinkIndex[link], h.Name)
}
}
for link := range newLinks {
if _, exists := oldLinks[link]; !exists {
if _, exists := backlinkIndex[link]; !exists {
backlinkIndex[link] = make(linkSet)
}
backlinkIndex[link][h.Name] = struct{}{}
}
}
backlinkIndexMutex.Unlock()
}
func BacklinksOnDelete(h *Hypha, oldText string) {
backlinkIndexMutex.Lock()
oldLinks := ExtractHyphaLinksFromContent(h.Name, oldText) oldLinks := ExtractHyphaLinksFromContent(h.Name, oldText)
for _, link := range oldLinks { newLinks := ExtractHyphaLinks(h)
if lSet, exists := backlinkIndex[link]; exists { backlinkConveyor <- BackIndexEditing{h.Name, oldLinks, newLinks}
delete(lSet, h.Name)
}
}
backlinkIndexMutex.Unlock()
} }
// BacklinksOnDelete is a deletion hook for backlinks index
func BacklinksOnDelete(h *Hypha, oldText string) {
oldLinks := ExtractHyphaLinksFromContent(h.Name, oldText)
backlinkConveyor <- BackIndexDeletion{h.Name, oldLinks}
}
// BacklinksOnRename is a renaming hook for backlinks index
func BacklinksOnRename(h *Hypha, oldName string) { func BacklinksOnRename(h *Hypha, oldName string) {
backlinkIndexMutex.Lock()
actualLinks := ExtractHyphaLinks(h) actualLinks := ExtractHyphaLinks(h)
for _, link := range actualLinks { backlinkConveyor <- BackIndexRenaming{oldName, h.Name, actualLinks}
if lSet, exists := backlinkIndex[link]; exists {
delete(lSet, oldName)
backlinkIndex[link][h.Name] = struct{}{}
}
}
backlinkIndexMutex.Unlock()
} }
// YieldHyphaBacklinks gets backlinks for a desired hypha, sorts and iterates over them // YieldHyphaBacklinks gets backlinks for a desired hypha, sorts and iterates over them

View File

@ -38,6 +38,7 @@ func main() {
// Init the subsystems: // Init the subsystems:
hyphae.Index(files.HyphaeDir()) hyphae.Index(files.HyphaeDir())
hyphae.IndexBacklinks() hyphae.IndexBacklinks()
go hyphae.RunBacklinksConveyor()
user.InitUserDatabase() user.InitUserDatabase()
history.Start() history.Start()
history.InitGitRepo() history.InitGitRepo()