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:
parent
b469b8f57d
commit
1c1260569b
@ -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
|
||||||
|
1
main.go
1
main.go
@ -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()
|
||||||
|
Loading…
Reference in New Issue
Block a user