mirror of
https://github.com/anyproto/any-sync.git
synced 2025-06-11 10:18:08 +09:00
416 lines
8.5 KiB
Go
416 lines
8.5 KiB
Go
package acltree
|
|
|
|
import (
|
|
"bytes"
|
|
"crypto/md5"
|
|
"fmt"
|
|
"github.com/anytypeio/go-anytype-infrastructure-experiments/util/slice"
|
|
"sort"
|
|
)
|
|
|
|
type Mode int
|
|
|
|
const (
|
|
Append Mode = iota
|
|
Rebuild
|
|
Nothing
|
|
)
|
|
|
|
// TODO: consider abstracting into separate package with iterator, remove
|
|
type Tree struct {
|
|
root *Change
|
|
headIds []string
|
|
metaHeadIds []string
|
|
attached map[string]*Change
|
|
unAttached map[string]*Change
|
|
// missed id -> list of dependency ids
|
|
waitList map[string][]string
|
|
invalidChanges map[string]struct{}
|
|
|
|
// bufs
|
|
iterCompBuf []*Change
|
|
iterQueue []*Change
|
|
|
|
duplicateEvents int
|
|
}
|
|
|
|
func (t *Tree) GetUnattachedChanges(changes ...*Change) []*Change {
|
|
return nil
|
|
}
|
|
|
|
func (t *Tree) RootId() string {
|
|
if t.root != nil {
|
|
return t.root.Id
|
|
}
|
|
return ""
|
|
}
|
|
|
|
func (t *Tree) Root() *Change {
|
|
return t.root
|
|
}
|
|
|
|
func (t *Tree) AddFast(changes ...*Change) {
|
|
for _, c := range changes {
|
|
// ignore existing
|
|
if _, ok := t.attached[c.Id]; ok {
|
|
continue
|
|
} else if _, ok := t.unAttached[c.Id]; ok {
|
|
continue
|
|
}
|
|
t.add(c)
|
|
}
|
|
t.updateHeads()
|
|
}
|
|
|
|
func (t *Tree) Add(changes ...*Change) (mode Mode) {
|
|
var beforeHeadIds = t.headIds
|
|
var attached bool
|
|
var empty = t.Len() == 0
|
|
for _, c := range changes {
|
|
// ignore existing
|
|
if _, ok := t.attached[c.Id]; ok {
|
|
continue
|
|
} else if _, ok := t.unAttached[c.Id]; ok {
|
|
continue
|
|
}
|
|
if t.add(c) {
|
|
attached = true
|
|
}
|
|
}
|
|
if !attached {
|
|
return Nothing
|
|
}
|
|
t.updateHeads()
|
|
if empty {
|
|
return Rebuild
|
|
}
|
|
for _, hid := range beforeHeadIds {
|
|
for _, newCh := range changes {
|
|
if _, ok := t.attached[newCh.Id]; ok {
|
|
if !t.after(newCh.Id, hid) {
|
|
return Rebuild
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return Append
|
|
}
|
|
|
|
func (t *Tree) RemoveInvalidChange(id string) {
|
|
stack := []string{id}
|
|
// removing all children of this id (either next or unattached)
|
|
for len(stack) > 0 {
|
|
var exists bool
|
|
top := stack[len(stack)-1]
|
|
stack = stack[:len(stack)-1]
|
|
|
|
if _, exists = t.invalidChanges[top]; exists {
|
|
continue
|
|
}
|
|
|
|
var rem *Change
|
|
t.invalidChanges[top] = struct{}{}
|
|
if rem, exists = t.unAttached[top]; exists {
|
|
delete(t.unAttached, top)
|
|
} else if rem, exists = t.attached[top]; exists {
|
|
// remove from all prev changes
|
|
for _, id := range rem.PreviousIds {
|
|
prev, exists := t.attached[id]
|
|
if !exists {
|
|
continue
|
|
}
|
|
for i, next := range prev.Next {
|
|
if next.Id == top {
|
|
prev.Next[i] = nil
|
|
prev.Next = append(prev.Next[:i], prev.Next[i+1:]...)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
delete(t.attached, top)
|
|
}
|
|
for _, el := range rem.Unattached {
|
|
stack = append(stack, el.Id)
|
|
}
|
|
for _, el := range rem.Next {
|
|
stack = append(stack, el.Id)
|
|
}
|
|
}
|
|
t.updateHeads()
|
|
}
|
|
|
|
func (t *Tree) add(c *Change) (attached bool) {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
if _, exists := t.invalidChanges[c.Id]; exists {
|
|
return false
|
|
}
|
|
|
|
if t.root == nil { // first element
|
|
t.root = c
|
|
t.attached = map[string]*Change{
|
|
c.Id: c,
|
|
}
|
|
t.unAttached = make(map[string]*Change)
|
|
t.waitList = make(map[string][]string)
|
|
t.invalidChanges = make(map[string]struct{})
|
|
return true
|
|
}
|
|
if len(c.PreviousIds) > 1 {
|
|
sort.Strings(c.PreviousIds)
|
|
}
|
|
// attaching only if all prev ids are attached
|
|
attached = true
|
|
for _, pid := range c.PreviousIds {
|
|
if prev, ok := t.attached[pid]; ok {
|
|
prev.Unattached = append(prev.Unattached, c)
|
|
continue
|
|
}
|
|
attached = false
|
|
if prev, ok := t.unAttached[pid]; ok {
|
|
prev.Unattached = append(prev.Unattached, c)
|
|
continue
|
|
}
|
|
wl := t.waitList[pid]
|
|
wl = append(wl, c.Id)
|
|
t.waitList[pid] = wl
|
|
}
|
|
if attached {
|
|
t.attach(c, true)
|
|
} else {
|
|
// clearing wait list
|
|
for _, wid := range t.waitList[c.Id] {
|
|
c.Unattached = append(c.Unattached, t.unAttached[wid])
|
|
}
|
|
delete(t.waitList, c.Id)
|
|
t.unAttached[c.Id] = c
|
|
}
|
|
return
|
|
}
|
|
|
|
func (t *Tree) canAttach(c *Change) (attach bool) {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
attach = true
|
|
for _, id := range c.PreviousIds {
|
|
if _, exists := t.attached[id]; !exists {
|
|
attach = false
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (t *Tree) attach(c *Change, newEl bool) {
|
|
t.attached[c.Id] = c
|
|
if !newEl {
|
|
delete(t.unAttached, c.Id)
|
|
}
|
|
|
|
// add next to all prev changes
|
|
for _, id := range c.PreviousIds {
|
|
// prev id must be attached if we attach this id
|
|
prev := t.attached[id]
|
|
prev.Next = append(prev.Next, c)
|
|
if len(prev.Next) > 1 {
|
|
sort.Sort(sortChanges(prev.Next))
|
|
}
|
|
for i, next := range prev.Unattached {
|
|
if next.Id == c.Id {
|
|
prev.Unattached[i] = nil
|
|
prev.Unattached = append(prev.Unattached[:i], prev.Unattached[i+1:]...)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// clearing wait list
|
|
if waitIds, ok := t.waitList[c.Id]; ok {
|
|
for _, wid := range waitIds {
|
|
next := t.unAttached[wid]
|
|
if t.canAttach(next) {
|
|
t.attach(next, false)
|
|
}
|
|
}
|
|
delete(t.waitList, c.Id)
|
|
}
|
|
|
|
for _, next := range c.Unattached {
|
|
if t.canAttach(next) {
|
|
t.attach(next, false)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (t *Tree) after(id1, id2 string) (found bool) {
|
|
t.iterate(t.attached[id2], func(c *Change) (isContinue bool) {
|
|
if c.Id == id1 {
|
|
found = true
|
|
return false
|
|
}
|
|
return true
|
|
})
|
|
return
|
|
}
|
|
|
|
func (t *Tree) dfs(startChange string) (uniqMap map[string]*Change) {
|
|
stack := make([]*Change, 0, 10)
|
|
stack = append(stack, t.attached[startChange])
|
|
uniqMap = map[string]*Change{}
|
|
|
|
for len(stack) > 0 {
|
|
ch := stack[len(stack)-1]
|
|
stack = stack[:len(stack)-1]
|
|
if _, exists := uniqMap[ch.Id]; exists {
|
|
continue
|
|
}
|
|
|
|
uniqMap[ch.Id] = ch
|
|
|
|
for _, prev := range ch.PreviousIds {
|
|
stack = append(stack, t.attached[prev])
|
|
}
|
|
}
|
|
return uniqMap
|
|
}
|
|
|
|
func (t *Tree) updateHeads() {
|
|
var newHeadIds, newMetaHeadIds []string
|
|
t.iterate(t.root, func(c *Change) (isContinue bool) {
|
|
if len(c.Next) == 0 {
|
|
newHeadIds = append(newHeadIds, c.Id)
|
|
}
|
|
return true
|
|
})
|
|
t.headIds = newHeadIds
|
|
t.metaHeadIds = newMetaHeadIds
|
|
sort.Strings(t.headIds)
|
|
sort.Strings(t.metaHeadIds)
|
|
}
|
|
|
|
func (t *Tree) ACLHeads() []string {
|
|
var aclTreeHeads []string
|
|
for _, head := range t.Heads() {
|
|
if slice.FindPos(aclTreeHeads, head) != -1 { // do not scan known heads
|
|
continue
|
|
}
|
|
precedingHeads := t.getPrecedingACLHeads(head)
|
|
|
|
for _, aclHead := range precedingHeads {
|
|
if slice.FindPos(aclTreeHeads, aclHead) != -1 {
|
|
continue
|
|
}
|
|
aclTreeHeads = append(aclTreeHeads, aclHead)
|
|
}
|
|
}
|
|
return aclTreeHeads
|
|
}
|
|
|
|
func (t *Tree) getPrecedingACLHeads(head string) []string {
|
|
headChange := t.attached[head]
|
|
|
|
if headChange.Content.GetAclData() != nil {
|
|
return []string{head}
|
|
} else {
|
|
return headChange.Content.AclHeadIds
|
|
}
|
|
}
|
|
|
|
func (t *Tree) iterate(start *Change, f func(c *Change) (isContinue bool)) {
|
|
it := newIterator()
|
|
defer freeIterator(it)
|
|
it.iterate(start, f)
|
|
}
|
|
|
|
func (t *Tree) iterateSkip(start *Change, skipBefore *Change, f func(c *Change) (isContinue bool)) {
|
|
it := newIterator()
|
|
defer freeIterator(it)
|
|
it.iterateSkip(start, skipBefore, f)
|
|
}
|
|
|
|
func (t *Tree) IterateSkip(startId string, skipBeforeId string, f func(c *Change) (isContinue bool)) {
|
|
it := newIterator()
|
|
defer freeIterator(it)
|
|
it.iterateSkip(t.attached[startId], t.attached[skipBeforeId], f)
|
|
}
|
|
|
|
func (t *Tree) Iterate(startId string, f func(c *Change) (isContinue bool)) {
|
|
t.iterate(t.attached[startId], f)
|
|
}
|
|
|
|
func (t *Tree) IterateBranching(startId string, f func(c *Change, branchLevel int) (isContinue bool)) {
|
|
// branchLevel indicates the number of parallel branches
|
|
var bc int
|
|
t.iterate(t.attached[startId], func(c *Change) (isContinue bool) {
|
|
if pl := len(c.PreviousIds); pl > 1 {
|
|
bc -= pl - 1
|
|
}
|
|
bl := bc
|
|
if nl := len(c.Next); nl > 1 {
|
|
bc += nl - 1
|
|
}
|
|
return f(c, bl)
|
|
})
|
|
}
|
|
|
|
func (t *Tree) Hash() string {
|
|
h := md5.New()
|
|
n := 0
|
|
t.iterate(t.root, func(c *Change) (isContinue bool) {
|
|
n++
|
|
fmt.Fprintf(h, "-%s", c.Id)
|
|
return true
|
|
})
|
|
return fmt.Sprintf("%d-%x", n, h.Sum(nil))
|
|
}
|
|
|
|
func (t *Tree) GetDuplicateEvents() int {
|
|
return t.duplicateEvents
|
|
}
|
|
|
|
func (t *Tree) ResetDuplicateEvents() {
|
|
t.duplicateEvents = 0
|
|
}
|
|
|
|
func (t *Tree) Len() int {
|
|
return len(t.attached)
|
|
}
|
|
|
|
func (t *Tree) Heads() []string {
|
|
return t.headIds
|
|
}
|
|
|
|
func (t *Tree) String() string {
|
|
var buf = bytes.NewBuffer(nil)
|
|
t.Iterate(t.RootId(), func(c *Change) (isContinue bool) {
|
|
buf.WriteString(c.Id)
|
|
if len(c.Next) > 1 {
|
|
buf.WriteString("-<")
|
|
} else if len(c.Next) > 0 {
|
|
buf.WriteString("->")
|
|
} else {
|
|
buf.WriteString("-|")
|
|
}
|
|
return true
|
|
})
|
|
return buf.String()
|
|
}
|
|
|
|
func (t *Tree) Get(id string) *Change {
|
|
return t.attached[id]
|
|
}
|
|
|
|
type sortChanges []*Change
|
|
|
|
func (s sortChanges) Len() int {
|
|
return len(s)
|
|
}
|
|
|
|
func (s sortChanges) Less(i, j int) bool {
|
|
return s[i].Id < s[j].Id
|
|
}
|
|
|
|
func (s sortChanges) Swap(i, j int) {
|
|
s[i], s[j] = s[j], s[i]
|
|
}
|