canepat e58d0fa537
Add B+ tree prototype implementation (#258)
* Add B+ tree prototype implementation

* Add Apache license header to all files
Fix linter errors
2022-01-21 07:06:09 +00:00

814 lines
26 KiB
Go

/*
Copyright 2022 Erigon contributors
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package bptree
import (
"fmt"
"sort"
)
func upsert(n *Node23, kvItems KeyValues, stats *Stats) (nodes []*Node23, newFirstKey *Felt, intermediateKeys []*Felt) {
ensure(sort.IsSorted(kvItems), "kvItems are not sorted by key")
if kvItems.Len() == 0 && n == nil {
return []*Node23{n}, nil, []*Felt{}
}
if n == nil {
n = makeEmptyLeafNode()
}
if n.isLeaf {
return upsertLeaf(n, kvItems, stats)
} else {
return upsertInternal(n, kvItems, stats)
}
}
func upsertLeaf(n *Node23, kvItems KeyValues, stats *Stats) (nodes []*Node23, newFirstKey *Felt, intermediateKeys []*Felt) {
ensure(n.isLeaf, "node is not leaf")
if kvItems.Len() == 0 {
if n.nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.nextKey())
}
return []*Node23{n}, nil, intermediateKeys
}
if !n.exposed {
n.exposed = true
stats.ExposedCount++
stats.OpeningHashes += n.howManyHashes()
}
currentFirstKey := n.firstKey()
addOrReplaceLeaf(n, kvItems, stats)
if n.firstKey() != currentFirstKey {
newFirstKey = n.firstKey()
} else {
newFirstKey = nil
}
if n.keyCount() > 3 {
for n.keyCount() > 3 {
newLeaf := makeLeafNode(n.keys[:3], n.values[:3], stats)
intermediateKeys = append(intermediateKeys, n.keys[2])
nodes = append(nodes, newLeaf)
n.keys, n.values = n.keys[2:], n.values[2:]
}
newLeaf := makeLeafNode(n.keys, n.values, stats)
if n.nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.nextKey())
}
nodes = append(nodes, newLeaf)
return nodes, newFirstKey, intermediateKeys
} else {
if n.nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.nextKey())
}
return []*Node23{n}, newFirstKey, intermediateKeys
}
}
func upsertInternal(n *Node23, kvItems KeyValues, stats *Stats) (nodes []*Node23, newFirstKey *Felt, intermediateKeys []*Felt) {
ensure(!n.isLeaf, "node is not internal")
if kvItems.Len() == 0 {
if n.lastLeaf().nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.lastLeaf().nextKey())
}
return []*Node23{n}, nil, intermediateKeys
}
if !n.exposed {
n.exposed = true
stats.ExposedCount++
stats.OpeningHashes += n.howManyHashes()
}
itemSubsets := splitItems(n, kvItems)
newChildren := make([]*Node23, 0)
newKeys := make([]*Felt, 0)
for i := len(n.children) - 1; i >= 0; i-- {
child := n.children[i]
childNodes, childNewFirstKey, childIntermediateKeys := upsert(child, itemSubsets[i], stats)
newChildren = append(childNodes, newChildren...)
newKeys = append(childIntermediateKeys, newKeys...)
if childNewFirstKey != nil {
if i > 0 {
// Handle newFirstKey here
previousChild := n.children[i-1]
if previousChild.isLeaf {
ensure(len(previousChild.keys) > 0, "upsertInternal: previousChild has no keys")
if previousChild.nextKey() != childNewFirstKey {
previousChild.setNextKey(childNewFirstKey, stats)
}
} else {
ensure(len(previousChild.children) > 0, "upsertInternal: previousChild has no children")
lastLeaf := previousChild.lastLeaf()
if lastLeaf.nextKey() != childNewFirstKey {
lastLeaf.setNextKey(childNewFirstKey, stats)
}
}
// TODO(canepat): previousChild/previousLastLeaf changed instead of making new node
} else {
// Propagate newFirstKey up
newFirstKey = childNewFirstKey
}
}
}
n.children = newChildren
if n.childrenCount() > 3 {
ensure(len(newKeys) >= n.childrenCount()-1 || n.childrenCount()%2 == 0 && n.childrenCount()%len(newKeys) == 0, "upsertInternal: inconsistent #children vs #newKeys")
var hasIntermediateKeys bool
if len(newKeys) == n.childrenCount()-1 || len(newKeys) == n.childrenCount() {
/* Groups are: 2,2...2 or 3 */
hasIntermediateKeys = true
} else {
/* Groups are: 2,2...2 */
hasIntermediateKeys = false
}
for n.childrenCount() > 3 {
nodes = append(nodes, makeInternalNode(n.children[:2], newKeys[:1], stats))
n.children = n.children[2:]
if hasIntermediateKeys {
intermediateKeys = append(intermediateKeys, newKeys[1])
newKeys = newKeys[2:]
} else {
newKeys = newKeys[1:]
}
}
ensure(n.childrenCount() > 0 && len(newKeys) > 0, "upsertInternal: inconsistent #children vs #newKeys")
if n.childrenCount() == 2 {
ensure(len(newKeys) > 0, "upsertInternal: inconsistent #newKeys")
nodes = append(nodes, makeInternalNode(n.children, newKeys[:1], stats))
intermediateKeys = append(intermediateKeys, newKeys[1:]...)
} else if n.childrenCount() == 3 {
ensure(len(newKeys) > 1, "upsertInternal: inconsistent #newKeys")
nodes = append(nodes, makeInternalNode(n.children, newKeys[:2], stats))
intermediateKeys = append(intermediateKeys, newKeys[2:]...)
} else {
ensure(false, fmt.Sprintf("upsertInternal: inconsistent #children=%d #newKeys=%d\n", n.childrenCount(), len(newKeys)))
}
return nodes, newFirstKey, intermediateKeys
} else { // n.childrenCount() is 2 or 3
ensure(len(newKeys) > 0, "upsertInternal: newKeys count is zero")
if len(newKeys) == len(n.children) {
n.keys = newKeys[:len(newKeys)-1]
intermediateKeys = append(intermediateKeys, newKeys[len(newKeys)-1])
} else {
n.keys = newKeys
}
// TODO(canepat): n.keys changed instead of making new node
n.updated = true
stats.UpdatedCount++
return []*Node23{n}, newFirstKey, intermediateKeys
}
}
func addOrReplaceLeaf(n *Node23, kvItems KeyValues, stats *Stats) {
ensure(n.isLeaf, "addOrReplaceLeaf: node is not leaf")
ensure(len(n.keys) > 0 && len(n.values) > 0, "addOrReplaceLeaf: node keys/values are empty")
ensure(len(kvItems.keys) > 0 && len(kvItems.keys) == len(kvItems.values), "addOrReplaceLeaf: invalid kvItems")
// Temporarily remove next key/value
nextKey, nextValue := n.nextKey(), n.nextValue()
n.keys = n.keys[:len(n.keys)-1]
n.values = n.values[:len(n.values)-1]
// kvItems are ordered by key: search there using n.keys that here are 1 or 2 by design (0 just for empty tree)
switch n.keyCount() {
case 0:
n.keys = append(n.keys, kvItems.keys...)
n.values = append(n.values, kvItems.values...)
case 1:
addOrReplaceLeaf1(n, kvItems, stats)
case 2:
addOrReplaceLeaf2(n, kvItems, stats)
default:
ensure(false, fmt.Sprintf("addOrReplaceLeaf: invalid key count %d", n.keyCount()))
}
// Restore next key/value
n.keys = append(n.keys, nextKey)
n.values = append(n.values, nextValue)
}
func addOrReplaceLeaf1(n *Node23, kvItems KeyValues, stats *Stats) {
ensure(n.isLeaf, "addOrReplaceLeaf1: node is not leaf")
ensure(n.keyCount() == 1, "addOrReplaceLeaf1: leaf has not 1 *canonical* key")
key0, value0 := n.keys[0], n.values[0]
index0 := sort.Search(kvItems.Len(), func(i int) bool { return *kvItems.keys[i] >= *key0 })
if index0 < kvItems.Len() {
// Insert keys/values concatenating new ones around key0
n.keys = append(make([]*Felt, 0), kvItems.keys[:index0]...)
n.values = append(make([]*Felt, 0), kvItems.values[:index0]...)
n.keys = append(n.keys, key0)
n.values = append(n.values, value0)
if *kvItems.keys[index0] == *key0 {
// Incoming key matches an existing key: update
n.keys = append(n.keys, kvItems.keys[index0+1:]...)
n.values = append(n.values, kvItems.values[index0+1:]...)
n.updated = true
stats.UpdatedCount++
} else {
n.keys = append(n.keys, kvItems.keys[index0:]...)
n.values = append(n.values, kvItems.values[index0:]...)
}
} else {
// key0 greater than any input key
n.keys = append(kvItems.keys, key0)
n.values = append(kvItems.values, value0)
}
}
func addOrReplaceLeaf2(n *Node23, kvItems KeyValues, stats *Stats) {
ensure(n.isLeaf, "addOrReplaceLeaf2: node is not leaf")
ensure(n.keyCount() == 2, "addOrReplaceLeaf2: leaf has not 2 *canonical* keys")
key0, value0, key1, value1 := n.keys[0], n.values[0], n.keys[1], n.values[1]
index0 := sort.Search(kvItems.Len(), func(i int) bool { return *kvItems.keys[i] >= *key0 })
index1 := sort.Search(kvItems.Len(), func(i int) bool { return *kvItems.keys[i] >= *key1 })
ensure(index1 >= index0, "addOrReplaceLeaf2: keys not ordered")
if index0 < kvItems.Len() {
if index1 < kvItems.Len() {
// Insert keys/values concatenating new ones around key0 and key1
n.keys = append(make([]*Felt, 0), kvItems.keys[:index0]...)
n.values = append(make([]*Felt, 0), kvItems.values[:index0]...)
n.keys = append(n.keys, key0)
n.values = append(n.values, value0)
if *kvItems.keys[index0] == *key0 {
// Incoming key matches an existing key: update
n.keys = append(n.keys, kvItems.keys[index0+1:index1]...)
n.values = append(n.values, kvItems.values[index0+1:index1]...)
n.updated = true
stats.UpdatedCount++
} else {
n.keys = append(n.keys, kvItems.keys[index0:index1]...)
n.values = append(n.values, kvItems.values[index0:index1]...)
}
n.keys = append(n.keys, key1)
n.values = append(n.values, value1)
if *kvItems.keys[index1] == *key1 {
// Incoming key matches an existing key: update
n.keys = append(n.keys, kvItems.keys[index1+1:]...)
n.values = append(n.values, kvItems.values[index1+1:]...)
if !n.updated {
n.updated = true
stats.UpdatedCount++
}
} else {
n.keys = append(n.keys, kvItems.keys[index1:]...)
n.values = append(n.values, kvItems.values[index1:]...)
}
} else {
// Insert keys/values concatenating new ones around key0, then add key1
n.keys = append(make([]*Felt, 0), kvItems.keys[:index0]...)
n.values = append(make([]*Felt, 0), kvItems.values[:index0]...)
n.keys = append(n.keys, key0)
n.values = append(n.values, value0)
if *kvItems.keys[index0] == *key0 {
// Incoming key matches an existing key: update
n.keys = append(n.keys, kvItems.keys[index0+1:]...)
n.values = append(n.values, kvItems.values[index0+1:]...)
n.updated = true
stats.UpdatedCount++
} else {
n.keys = append(n.keys, kvItems.keys[index0:]...)
n.values = append(n.values, kvItems.values[index0:]...)
}
n.keys = append(n.keys, key1)
n.values = append(n.values, value1)
}
} else {
ensure(index1 == index0, "addOrReplaceLeaf2: keys not ordered")
// Both key0 and key1 greater than any input key
n.keys = append(kvItems.keys, key0, key1)
n.values = append(kvItems.values, value0, value1)
}
}
func splitItems(n *Node23, kvItems KeyValues) []KeyValues {
ensure(!n.isLeaf, "splitItems: node is not internal")
ensure(len(n.keys) > 0, "splitItems: internal node has no keys")
itemSubsets := make([]KeyValues, 0)
for i, key := range n.keys {
splitIndex := sort.Search(kvItems.Len(), func(i int) bool { return *kvItems.keys[i] >= *key })
itemSubsets = append(itemSubsets, KeyValues{kvItems.keys[:splitIndex], kvItems.values[:splitIndex]})
kvItems = KeyValues{kvItems.keys[splitIndex:], kvItems.values[splitIndex:]}
if i == len(n.keys)-1 {
itemSubsets = append(itemSubsets, kvItems)
}
}
ensure(len(itemSubsets) == len(n.children), "item subsets and children have different cardinality")
return itemSubsets
}
func delete(n *Node23, keysToDelete []Felt, stats *Stats) (deleted *Node23, nextKey *Felt, intermediateKeys []*Felt) {
ensure(sort.IsSorted(Keys(keysToDelete)), "keysToDelete are not sorted")
if n == nil {
return n, nil, intermediateKeys
}
if n.isLeaf {
return deleteLeaf(n, keysToDelete, stats)
} else {
return deleteInternal(n, keysToDelete, stats)
}
}
func deleteLeaf(n *Node23, keysToDelete []Felt, stats *Stats) (deleted *Node23, nextKey *Felt, intermediateKeys []*Felt) {
ensure(n.isLeaf, fmt.Sprintf("node %s is not leaf", n))
if len(keysToDelete) == 0 {
if n.nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.nextKey())
}
return n, nil, intermediateKeys
}
if !n.exposed {
n.exposed = true
stats.ExposedCount++
stats.OpeningHashes += n.howManyHashes()
}
currentFirstKey := n.firstKey()
deleteLeafKeys(n, keysToDelete, stats)
if n.keyCount() == 1 {
return nil, n.nextKey(), intermediateKeys
} else {
if n.nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.nextKey())
}
if n.firstKey() != currentFirstKey {
return n, n.firstKey(), intermediateKeys
} else {
return n, nil, intermediateKeys
}
}
}
func deleteLeafKeys(n *Node23, keysToDelete []Felt, stats *Stats) (deleted KeyValues) {
ensure(n.isLeaf, "deleteLeafKeys: node is not leaf")
switch n.keyCount() {
case 2:
if Keys(keysToDelete).Contains(*n.keys[0]) {
deleted.keys = n.keys[:1]
deleted.values = n.values[:1]
n.keys = n.keys[1:]
n.values = n.values[1:]
stats.DeletedCount++
}
case 3:
if Keys(keysToDelete).Contains(*n.keys[0]) {
if Keys(keysToDelete).Contains(*n.keys[1]) {
deleted.keys = n.keys[:2]
deleted.values = n.values[:2]
n.keys = n.keys[2:]
n.values = n.values[2:]
stats.DeletedCount++
} else {
deleted.keys = n.keys[:1]
deleted.values = n.values[:1]
n.keys = n.keys[1:]
n.values = n.values[1:]
n.updated = true
stats.UpdatedCount++
}
} else {
if Keys(keysToDelete).Contains(*n.keys[1]) {
deleted.keys = n.keys[1:2]
deleted.values = n.values[1:2]
n.keys = append(n.keys[:1], n.keys[2])
n.values = append(n.values[:1], n.values[2])
n.updated = true
stats.UpdatedCount++
}
}
default:
ensure(false, fmt.Sprintf("unexpected number of keys in %s", n))
}
return deleted
}
func deleteInternal(n *Node23, keysToDelete []Felt, stats *Stats) (deleted *Node23, nextKey *Felt, intermediateKeys []*Felt) {
ensure(!n.isLeaf, fmt.Sprintf("node %s is not internal", n))
if len(keysToDelete) == 0 {
if n.lastLeaf().nextKey() != nil {
intermediateKeys = append(intermediateKeys, n.lastLeaf().nextKey())
}
return n, nil, intermediateKeys
}
if !n.exposed {
n.exposed = true
stats.ExposedCount++
stats.OpeningHashes += n.howManyHashes()
}
keySubsets := splitKeys(n, keysToDelete)
newKeys := make([]*Felt, 0)
for i := len(n.children) - 1; i >= 0; i-- {
child, childNextKey, childIntermediateKeys := delete(n.children[i], keySubsets[i], stats)
newKeys = append(childIntermediateKeys, newKeys...)
if i > 0 {
previousIndex := i - 1
previousChild := n.children[previousIndex]
for previousChild.isEmpty() && previousIndex-1 >= 0 {
previousChild = n.children[previousIndex-1]
previousIndex = previousIndex - 1
}
if child == nil || childNextKey != nil {
if previousChild.isLeaf {
ensure(len(previousChild.keys) > 0, "delete: previousChild has no keys")
if previousChild.nextKey() != childNextKey {
previousChild.setNextKey(childNextKey, stats)
}
} else {
ensure(len(previousChild.children) > 0, "delete: previousChild has no children")
lastLeaf := previousChild.lastLeaf()
if lastLeaf.nextKey() != childNextKey {
lastLeaf.setNextKey(childNextKey, stats)
}
}
}
if !previousChild.isEmpty() && child != nil && child.childrenCount() == 1 {
child.keys = child.keys[:0]
newLeft, newRight := mergeRight2Left(previousChild, child, stats)
n.children = append(n.children[:previousIndex], append([]*Node23{newLeft, newRight}, n.children[i+1:]...)...)
}
} else {
nextIndex := i + 1
nextChild := n.children[nextIndex]
for nextChild.isEmpty() && nextIndex+1 < n.childrenCount() {
nextChild = n.children[nextIndex+1]
nextIndex = nextIndex + 1
}
if !nextChild.isEmpty() && child != nil && child.childrenCount() == 1 {
child.keys = child.keys[:0]
newLeft, newRight := mergeLeft2Right(child, nextChild, stats)
n.children = append([]*Node23{newLeft, newRight}, n.children[nextIndex+1:]...)
}
if childNextKey != nil {
nextKey = childNextKey
}
}
}
switch len(n.children) {
case 2:
nextKey, intermediateKeys = update2Node(n, newKeys, nextKey, intermediateKeys, stats)
case 3:
nextKey, intermediateKeys = update3Node(n, newKeys, nextKey, intermediateKeys, stats)
default:
ensure(false, fmt.Sprintf("unexpected number of children in %s", n))
}
for _, child := range n.children {
if child.updated {
n.updated = true
stats.UpdatedCount++
break
}
}
if n.keyCount() == 0 {
return nil, nextKey, intermediateKeys
} else {
return n, nextKey, intermediateKeys
}
}
func mergeLeft2Right(left, right *Node23, stats *Stats) (newLeft, newRight *Node23) {
ensure(!left.isLeaf, "mergeLeft2Right: left is leaf")
ensure(left.childrenCount() > 0, "mergeLeft2Right: left has no children")
if left.firstChild().childrenCount() == 1 {
newLeftFirstChild, newRightFirstChild := mergeLeft2Right(left.firstChild(), right.firstChild(), stats)
left = makeInternalNode(
[]*Node23{newLeftFirstChild},
left.keys,
stats,
)
right = makeInternalNode(
append([]*Node23{newRightFirstChild}, right.children[1:]...),
right.keys,
stats,
)
}
if right.childrenCount() < 3 {
if !left.firstChild().isEmpty() {
if right.childrenCount() == 1 {
if right.firstChild().isEmpty() {
newLeft = left
newRight = makeInternalNode([]*Node23{}, []*Felt{}, stats)
} else {
newRight = makeInternalNode(
append([]*Node23{left.firstChild()}, right.children...),
[]*Felt{left.lastLeaf().nextKey()},
stats,
)
if left.keyCount() > 1 {
newLeft = makeInternalNode(left.children[1:], left.keys[1:], stats)
} else {
newLeft = makeInternalNode(left.children[1:], left.keys, stats)
}
}
} else {
newRight = makeInternalNode(
append([]*Node23{left.firstChild()}, right.children...),
append([]*Felt{left.lastLeaf().nextKey()}, right.keys...),
stats,
)
if left.keyCount() > 1 {
newLeft = makeInternalNode(left.children[1:], left.keys[1:], stats)
} else {
newLeft = makeInternalNode(left.children[1:], left.keys, stats)
}
}
} else {
newRight = right
newLeft = makeInternalNode([]*Node23{}, []*Felt{}, stats)
}
} else {
newLeft, newRight = mergeRight2Left(left, right, stats)
}
return newLeft, newRight
}
func mergeRight2Left(left, right *Node23, stats *Stats) (newLeft, newRight *Node23) {
ensure(!right.isLeaf, "mergeRight2Left: right is leaf")
ensure(right.childrenCount() > 0, "mergeRight2Left: right has no children")
if right.firstChild().childrenCount() == 1 {
newLeftLastChild, newRightFirstChild := mergeRight2Left(left.lastChild(), right.firstChild(), stats)
left = makeInternalNode(
append(left.children[:len(left.children)-1], newLeftLastChild),
left.keys,
stats,
)
right = makeInternalNode(
[]*Node23{newRightFirstChild},
right.keys,
stats,
)
}
if left.childrenCount() < 3 {
if !right.firstChild().isEmpty() {
if left.childrenCount() == 1 {
if left.firstChild().isEmpty() {
newLeft = makeInternalNode([]*Node23{}, []*Felt{}, stats)
newRight = right
} else {
newLeft = makeInternalNode(
append(left.children, right.firstChild()),
[]*Felt{right.firstLeaf().firstKey()},
stats,
)
if right.keyCount() > 1 {
newRight = makeInternalNode(right.children[1:], right.keys[1:], stats)
} else {
newRight = makeInternalNode(right.children[1:], right.keys, stats)
}
}
} else {
newLeft = makeInternalNode(
append(left.children, right.firstChild()),
append(left.keys, right.firstLeaf().firstKey()),
stats,
)
if right.keyCount() > 1 {
newRight = makeInternalNode(right.children[1:], right.keys[1:], stats)
} else {
newRight = makeInternalNode(right.children[1:], right.keys, stats)
}
}
} else {
newLeft = left
newRight = makeInternalNode([]*Node23{}, []*Felt{}, stats)
}
} else {
newLeft, newRight = mergeLeft2Right(left, right, stats)
}
return newLeft, newRight
}
func splitKeys(n *Node23, keysToDelete []Felt) [][]Felt {
ensure(!n.isLeaf, "splitKeys: node is not internal")
ensure(len(n.keys) > 0, fmt.Sprintf("splitKeys: internal node %s has no keys", n))
keySubsets := make([][]Felt, 0)
for i, key := range n.keys {
splitIndex := sort.Search(len(keysToDelete), func(i int) bool { return keysToDelete[i] >= *key })
keySubsets = append(keySubsets, keysToDelete[:splitIndex])
keysToDelete = keysToDelete[splitIndex:]
if i == len(n.keys)-1 {
keySubsets = append(keySubsets, keysToDelete)
}
}
ensure(len(keySubsets) == len(n.children), "key subsets and children have different cardinality")
return keySubsets
}
func update2Node(n *Node23, newKeys []*Felt, nextKey *Felt, intermediateKeys []*Felt, stats *Stats) (*Felt, []*Felt) {
ensure(len(n.children) == 2, "update2Node: wrong number of children")
switch len(newKeys) {
case 0:
break
case 1:
n.keys = newKeys
case 2:
n.keys = newKeys[:1]
intermediateKeys = append(intermediateKeys, newKeys[1])
default:
ensure(false, fmt.Sprintf("update2Node: wrong number of newKeys=%d", len(newKeys)))
}
nodeA, nodeC := n.children[0], n.children[1]
if nodeA.isEmpty() {
if nodeC.isEmpty() {
/* A is empty, a_next is the "next key"; C is empty, c_next is the "next key" */
n.children = n.children[:0]
n.keys = n.keys[:0]
if nodeC.isLeaf {
return nodeC.nextKey(), intermediateKeys
}
return nextKey, intermediateKeys
} else {
/* A is empty, a_next is the "next key"; C is not empty */
n.children = n.children[1:]
/// n.keys = []*Felt{nodeC.lastLeaf().nextKey()}
if nodeA.isLeaf {
return nodeA.nextKey(), intermediateKeys
}
return nextKey, intermediateKeys
}
} else {
if nodeC.isEmpty() {
/* A is not empty; C is empty, c_next is the "next key" */
n.children = n.children[:1]
/// n.keys = []*Felt{nodeA.lastLeaf().nextKey()}
if nodeC.isLeaf {
nodeA.setNextKey(nodeC.nextKey(), stats)
}
return nextKey, intermediateKeys
} else {
/* A is not empty; C is not empty */
n.keys = []*Felt{nodeA.lastLeaf().nextKey()}
return nextKey, intermediateKeys
}
}
}
func update3Node(n *Node23, newKeys []*Felt, nextKey *Felt, intermediateKeys []*Felt, stats *Stats) (*Felt, []*Felt) {
ensure(len(n.children) == 3, "update3Node: wrong number of children")
switch len(newKeys) {
case 0:
break
case 1:
n.keys = newKeys
case 2:
n.keys = newKeys
case 3:
n.keys = newKeys[:2]
intermediateKeys = append(intermediateKeys, newKeys[2])
default:
ensure(false, fmt.Sprintf("update3Node: wrong number of newKeys=%d", len(newKeys)))
}
nodeA, nodeB, nodeC := n.children[0], n.children[1], n.children[2]
if nodeA.isEmpty() {
if nodeB.isEmpty() {
if nodeC.isEmpty() {
/* A is empty, a_next is the "next key"; B is empty, b_next is the "next key"; C is empty, c_next is the "next key" */
n.children = n.children[:0]
n.keys = n.keys[:0]
if nodeA.isLeaf {
return nodeC.nextKey(), intermediateKeys
}
return nextKey, intermediateKeys
} else {
/* A is empty, a_next is the "next key"; B is empty, b_next is the "next key"; C is not empty */
n.children = n.children[2:]
/// n.keys = []*Felt{nodeC.lastLeaf().nextKey()}
if nodeA.isLeaf {
return nodeB.nextKey(), intermediateKeys
}
return nextKey, intermediateKeys
}
} else {
if nodeC.isEmpty() {
/* A is empty, a_next is the "next key"; B is not empty; C is empty, c_next is the "next key" */
n.children = n.children[1:2]
/// n.keys = []*Felt{nodeB.lastLeaf().nextKey()}
if nodeA.isLeaf {
nodeB.setNextKey(nodeC.nextKey(), stats)
return nodeA.nextKey(), intermediateKeys
}
return nextKey, intermediateKeys
} else {
/* A is empty, a_next is the "next key"; B is not empty; C is not empty */
n.children = n.children[1:]
if nodeA.isLeaf {
n.keys = []*Felt{nodeB.nextKey()}
return nodeA.nextKey(), intermediateKeys
}
n.keys = []*Felt{nodeB.lastLeaf().nextKey()}
return nextKey, intermediateKeys
}
}
} else {
if nodeB.isEmpty() {
if nodeC.isEmpty() {
/* A is not empty; B is empty, b_next is the "next key"; C is empty, c_next is the "next key" */
n.children = n.children[:1]
if nodeA.isLeaf {
nodeA.setNextKey(nodeC.nextKey(), stats)
}
/// n.keys = []*Felt{nodeA.lastLeaf().nextKey()}
return nextKey, intermediateKeys
} else {
/* A is not empty; B is empty, b_next is the "next key"; C is not empty */
n.children = append(n.children[:1], n.children[2])
if nodeA.isLeaf {
n.keys = []*Felt{nodeB.nextKey()}
nodeA.setNextKey(nodeB.nextKey(), stats)
} else {
n.keys = []*Felt{nodeA.lastLeaf().nextKey()}
}
return nextKey, intermediateKeys
}
} else {
if nodeC.isEmpty() {
/* A is not empty; B is not empty; C is empty, c_next is the "next key" */
n.children = n.children[:2]
if nodeA.isLeaf {
n.keys = []*Felt{nodeA.nextKey()}
nodeB.setNextKey(nodeC.nextKey(), stats)
} else {
n.keys = []*Felt{nodeA.lastLeaf().nextKey()}
}
return nextKey, intermediateKeys
} else {
/* A is not empty; B is not empty; C is not empty */
///n.keys = []*Felt{nodeA.lastLeaf().nextKey(), nodeB.lastLeaf().nextKey()}
return nextKey, intermediateKeys
}
}
}
}
func demote(node *Node23, nextKey *Felt, intermediateKeys []*Felt, stats *Stats) (*Node23, *Felt) {
if node == nil {
return nil, nextKey
} else if len(node.children) == 0 {
if len(node.keys) == 0 {
return nil, nextKey
} else {
return node, nextKey
}
} else if len(node.children) == 1 {
return demote(node.children[0], nextKey, intermediateKeys, stats)
} else if len(node.children) == 2 {
firstChild, secondChild := node.children[0], node.children[1]
if firstChild.keyCount() == 0 && secondChild.keyCount() == 0 {
return nil, nextKey
}
if firstChild.keyCount() == 0 && secondChild.keyCount() > 0 {
return secondChild, nextKey
}
if firstChild.keyCount() > 0 && secondChild.keyCount() == 0 {
return firstChild, nextKey
}
if firstChild.keyCount() == 2 && secondChild.keyCount() == 2 {
if firstChild.isLeaf {
keys := []*Felt{firstChild.firstKey(), secondChild.firstKey(), secondChild.nextKey()}
values := []*Felt{firstChild.firstValue(), secondChild.firstValue(), secondChild.nextValue()}
return makeLeafNode(keys, values, stats), nextKey
}
}
}
return node, nextKey
}