/* 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 }