mirror of
https://github.com/arnaucube/arbo.git
synced 2026-01-08 15:01:29 +01:00
AddBatch: commit tx at end,allow batch w/ len!=2^n
This commit is contained in:
71
addbatch.go
71
addbatch.go
@@ -189,6 +189,13 @@ func (t *Tree) AddBatchOpt(keys, values [][]byte) ([]int, error) {
|
|||||||
invalids = append(invalids, kvsNonP2[i].pos)
|
invalids = append(invalids, kvsNonP2[i].pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// store root to db
|
||||||
|
if err := t.tx.Put(dbKeyRoot, t.root); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err = t.tx.Commit(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
return invalids, nil
|
return invalids, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -209,6 +216,13 @@ func (t *Tree) AddBatchOpt(keys, values [][]byte) ([]int, error) {
|
|||||||
invalids = append(invalids, excedents[i].pos)
|
invalids = append(invalids, excedents[i].pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// store root to db
|
||||||
|
if err := t.tx.Put(dbKeyRoot, t.root); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if err = t.tx.Commit(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
return invalids, nil
|
return invalids, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -247,6 +261,7 @@ func (t *Tree) AddBatchOpt(keys, values [][]byte) ([]int, error) {
|
|||||||
}
|
}
|
||||||
t.root = newRoot
|
t.root = newRoot
|
||||||
|
|
||||||
|
// add the key-values that have not been used yet
|
||||||
var invalids []int
|
var invalids []int
|
||||||
for i := 0; i < len(excedents); i++ {
|
for i := 0; i < len(excedents); i++ {
|
||||||
// Add until the level L
|
// Add until the level L
|
||||||
@@ -255,14 +270,39 @@ func (t *Tree) AddBatchOpt(keys, values [][]byte) ([]int, error) {
|
|||||||
invalids = append(invalids, excedents[i].pos) // TODO WIP
|
invalids = append(invalids, excedents[i].pos) // TODO WIP
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// store root to db
|
||||||
|
if err := t.tx.Put(dbKeyRoot, t.root); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = t.tx.Commit(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
return invalids, nil
|
return invalids, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// CASE D
|
keysAtL, err := t.getKeysAtLevel(l + 1)
|
||||||
if true { // TODO enter in CASE D if len(keysAtL)=nCPU, if not, CASE E
|
if err != nil {
|
||||||
return t.caseD(nCPU, l, kvs)
|
return nil, err
|
||||||
}
|
}
|
||||||
|
// CASE D
|
||||||
|
if len(keysAtL) == nCPU { // enter in CASE D if len(keysAtL)=nCPU, if not, CASE E
|
||||||
|
invalids, err := t.caseD(nCPU, l, keysAtL, kvs)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// store root to db
|
||||||
|
if err := t.tx.Put(dbKeyRoot, t.root); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err = t.tx.Commit(); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return invalids, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// CASE E: add one key of each bucket, and then do CASE D
|
||||||
|
|
||||||
// TODO store t.root into DB
|
// TODO store t.root into DB
|
||||||
// TODO update NLeafs from DB
|
// TODO update NLeafs from DB
|
||||||
@@ -296,11 +336,7 @@ func (t *Tree) caseB(l int, kvs []kv) ([]int, []kv, error) {
|
|||||||
return invalids, kvsNonP2, nil
|
return invalids, kvsNonP2, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *Tree) caseD(nCPU, l int, kvs []kv) ([]int, error) {
|
func (t *Tree) caseD(nCPU, l int, keysAtL [][]byte, kvs []kv) ([]int, error) {
|
||||||
keysAtL, err := t.getKeysAtLevel(l + 1)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
buckets := splitInBuckets(kvs, nCPU)
|
buckets := splitInBuckets(kvs, nCPU)
|
||||||
|
|
||||||
subRoots := make([][]byte, nCPU)
|
subRoots := make([][]byte, nCPU)
|
||||||
@@ -314,14 +350,13 @@ func (t *Tree) caseD(nCPU, l int, kvs []kv) ([]int, error) {
|
|||||||
var err error
|
var err error
|
||||||
txs[cpu], err = t.db.NewTx()
|
txs[cpu], err = t.db.NewTx()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err) // TODO
|
panic(err) // TODO WIP
|
||||||
}
|
}
|
||||||
bucketTree := Tree{tx: txs[cpu], db: t.db, maxLevels: t.maxLevels, // maxLevels-l
|
bucketTree := Tree{tx: txs[cpu], db: t.db, maxLevels: t.maxLevels - l, // maxLevels-l
|
||||||
hashFunction: t.hashFunction, root: keysAtL[cpu]}
|
hashFunction: t.hashFunction, root: keysAtL[cpu]}
|
||||||
|
|
||||||
for j := 0; j < len(buckets[cpu]); j++ {
|
for j := 0; j < len(buckets[cpu]); j++ {
|
||||||
if err = bucketTree.add(l, buckets[cpu][j].k, buckets[cpu][j].v); err != nil {
|
if err = bucketTree.add(l, buckets[cpu][j].k, buckets[cpu][j].v); err != nil {
|
||||||
fmt.Println("failed", buckets[cpu][j].k[:4])
|
|
||||||
invalidsInBucket[cpu] = append(invalidsInBucket[cpu], buckets[cpu][j].pos)
|
invalidsInBucket[cpu] = append(invalidsInBucket[cpu], buckets[cpu][j].pos)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -331,6 +366,13 @@ func (t *Tree) caseD(nCPU, l int, kvs []kv) ([]int, error) {
|
|||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
|
||||||
|
// merge buckets txs into Tree.tx
|
||||||
|
for i := 0; i < len(txs); i++ {
|
||||||
|
if err := t.tx.Add(txs[i]); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
newRoot, err := t.upFromKeys(subRoots)
|
newRoot, err := t.upFromKeys(subRoots)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
@@ -477,6 +519,13 @@ func (t *Tree) buildTreeBottomUp(nCPU int, kvs []kv) ([]int, error) {
|
|||||||
}
|
}
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
|
||||||
|
// merge buckets txs into Tree.tx
|
||||||
|
for i := 0; i < len(txs); i++ {
|
||||||
|
if err := t.tx.Add(txs[i]); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
newRoot, err := t.upFromKeys(subRoots)
|
newRoot, err := t.upFromKeys(subRoots)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ func TestAddBatchCaseA(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed without CASE A: ", time.Since(start))
|
||||||
|
|
||||||
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
@@ -94,7 +94,43 @@ func TestAddBatchCaseA(t *testing.T) {
|
|||||||
start = time.Now()
|
start = time.Now()
|
||||||
indexes, err := tree2.AddBatchOpt(keys, values)
|
indexes, err := tree2.AddBatchOpt(keys, values)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed with CASE A: ", time.Since(start))
|
||||||
|
c.Check(len(indexes), qt.Equals, 0)
|
||||||
|
|
||||||
|
// check that both trees roots are equal
|
||||||
|
c.Check(tree2.Root(), qt.DeepEquals, tree.Root())
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAddBatchCaseANotPowerOf2(t *testing.T) {
|
||||||
|
c := qt.New(t)
|
||||||
|
|
||||||
|
nLeafs := 1027
|
||||||
|
|
||||||
|
tree, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
|
c.Assert(err, qt.IsNil)
|
||||||
|
defer tree.db.Close()
|
||||||
|
|
||||||
|
for i := 0; i < nLeafs; i++ {
|
||||||
|
k := BigIntToBytes(big.NewInt(int64(i)))
|
||||||
|
v := BigIntToBytes(big.NewInt(int64(i * 2)))
|
||||||
|
if err := tree.Add(k, v); err != nil {
|
||||||
|
t.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
|
c.Assert(err, qt.IsNil)
|
||||||
|
defer tree2.db.Close()
|
||||||
|
|
||||||
|
var keys, values [][]byte
|
||||||
|
for i := 0; i < nLeafs; i++ {
|
||||||
|
k := BigIntToBytes(big.NewInt(int64(i)))
|
||||||
|
v := BigIntToBytes(big.NewInt(int64(i * 2)))
|
||||||
|
keys = append(keys, k)
|
||||||
|
values = append(values, v)
|
||||||
|
}
|
||||||
|
indexes, err := tree2.AddBatchOpt(keys, values)
|
||||||
|
c.Assert(err, qt.IsNil)
|
||||||
c.Check(len(indexes), qt.Equals, 0)
|
c.Check(len(indexes), qt.Equals, 0)
|
||||||
|
|
||||||
// check that both trees roots are equal
|
// check that both trees roots are equal
|
||||||
@@ -118,7 +154,7 @@ func TestAddBatchCaseB(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed without CASE B: ", time.Since(start))
|
||||||
|
|
||||||
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
@@ -144,7 +180,7 @@ func TestAddBatchCaseB(t *testing.T) {
|
|||||||
start = time.Now()
|
start = time.Now()
|
||||||
indexes, err := tree2.AddBatchOpt(keys, values)
|
indexes, err := tree2.AddBatchOpt(keys, values)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed with CASE B: ", time.Since(start))
|
||||||
c.Check(len(indexes), qt.Equals, 0)
|
c.Check(len(indexes), qt.Equals, 0)
|
||||||
|
|
||||||
// check that both trees roots are equal
|
// check that both trees roots are equal
|
||||||
@@ -296,7 +332,7 @@ func TestAddBatchCaseC(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed without CASE C: ", time.Since(start))
|
||||||
|
|
||||||
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
@@ -311,7 +347,6 @@ func TestAddBatchCaseC(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// tree2.PrintGraphviz(nil)
|
|
||||||
|
|
||||||
var keys, values [][]byte
|
var keys, values [][]byte
|
||||||
for i := 101; i < nLeafs; i++ {
|
for i := 101; i < nLeafs; i++ {
|
||||||
@@ -323,20 +358,11 @@ func TestAddBatchCaseC(t *testing.T) {
|
|||||||
start = time.Now()
|
start = time.Now()
|
||||||
indexes, err := tree2.AddBatchOpt(keys, values)
|
indexes, err := tree2.AddBatchOpt(keys, values)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed with CASE C: ", time.Since(start))
|
||||||
c.Check(len(indexes), qt.Equals, 0)
|
c.Check(len(indexes), qt.Equals, 0)
|
||||||
|
|
||||||
// check that both trees roots are equal
|
// check that both trees roots are equal
|
||||||
c.Check(tree2.Root(), qt.DeepEquals, tree.Root())
|
c.Check(tree2.Root(), qt.DeepEquals, tree.Root())
|
||||||
|
|
||||||
// tree.PrintGraphviz(nil)
|
|
||||||
// tree2.PrintGraphviz(nil)
|
|
||||||
// // tree.PrintGraphvizFirstNLevels(nil, 4)
|
|
||||||
// // tree2.PrintGraphvizFirstNLevels(nil, 4)
|
|
||||||
// fmt.Println("TREE")
|
|
||||||
// printLeafs("t1.txt", tree)
|
|
||||||
// fmt.Println("TREE2")
|
|
||||||
// printLeafs("t2.txt", tree2)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestAddBatchCaseD(t *testing.T) {
|
func TestAddBatchCaseD(t *testing.T) {
|
||||||
@@ -356,7 +382,7 @@ func TestAddBatchCaseD(t *testing.T) {
|
|||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed without CASE D: ", time.Since(start))
|
||||||
|
|
||||||
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
tree2, err := NewTree(memory.NewMemoryStorage(), 100, HashFunctionPoseidon)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
@@ -382,7 +408,7 @@ func TestAddBatchCaseD(t *testing.T) {
|
|||||||
start = time.Now()
|
start = time.Now()
|
||||||
indexes, err := tree2.AddBatchOpt(keys, values)
|
indexes, err := tree2.AddBatchOpt(keys, values)
|
||||||
c.Assert(err, qt.IsNil)
|
c.Assert(err, qt.IsNil)
|
||||||
fmt.Println(time.Since(start))
|
fmt.Println("time elapsed with CASE D: ", time.Since(start))
|
||||||
c.Check(len(indexes), qt.Equals, 0)
|
c.Check(len(indexes), qt.Equals, 0)
|
||||||
|
|
||||||
// check that both trees roots are equal
|
// check that both trees roots are equal
|
||||||
|
|||||||
Reference in New Issue
Block a user