mirror of
https://github.com/status-im/research.git
synced 2025-01-12 08:04:12 +00:00
Added optimized sparse merkle tree
This commit is contained in:
parent
c5edaabaff
commit
6ab4a5da40
29
trie_research/bintrie2/bintrie_test.py
Normal file
29
trie_research/bintrie2/bintrie_test.py
Normal file
@ -0,0 +1,29 @@
|
||||
import new_bintrie as t1
|
||||
import new_bintrie_optimized as t2
|
||||
import time
|
||||
import binascii
|
||||
|
||||
keys = [t1.sha3(bytes([i // 256, i % 256])) for i in range(10000)]
|
||||
|
||||
d = t1.EphemDB()
|
||||
r = t1.new_tree(d)
|
||||
a = time.time()
|
||||
for k in keys[:1000]:
|
||||
r = t1.update(d, r, k, k)
|
||||
print("Naive bintree time to update: %.4f" % (time.time() - a))
|
||||
print("Root: %s" % binascii.hexlify(r))
|
||||
|
||||
d = t2.EphemDB()
|
||||
r = t2.new_tree(d)
|
||||
a = time.time()
|
||||
for k in keys[:1000]:
|
||||
r = t2.update(d, r, k, k)
|
||||
print("DB-optimized bintree time to update: %.4f" % (time.time() - a))
|
||||
print("Root: %s" % binascii.hexlify(r))
|
||||
print("Writes: %d, reads: %d" % (d.writes, d.reads))
|
||||
d.reads = 0
|
||||
for k in keys[:500]:
|
||||
assert t2.get(d, r, k) == k
|
||||
for k in keys[-500:]:
|
||||
assert t2.get(d, r, k) == b'\x00' * 32
|
||||
print("Reads: %d" % d.reads)
|
@ -14,7 +14,7 @@ class EphemDB():
|
||||
del self.kv[k]
|
||||
|
||||
zerohashes = [b'\x00' * 32]
|
||||
for i in range(255):
|
||||
for i in range(256):
|
||||
zerohashes.insert(0, sha3(zerohashes[0] + zerohashes[0]))
|
||||
|
||||
def new_tree(db):
|
||||
@ -106,7 +106,7 @@ def compress_proof(proof):
|
||||
bits = bytearray(32)
|
||||
oproof = b''
|
||||
for i, p in enumerate(proof):
|
||||
if p == zerohashes[i]:
|
||||
if p == zerohashes[i+1]:
|
||||
bits[i // 8] ^= 1 << i % 8
|
||||
else:
|
||||
oproof += p
|
||||
@ -118,7 +118,7 @@ def decompress_proof(oproof):
|
||||
pos = 32
|
||||
for i in range(256):
|
||||
if bits[i // 8] & (1 << (i % 8)):
|
||||
proof.append(zerohashes[i])
|
||||
proof.append(zerohashes[i+1])
|
||||
else:
|
||||
proof.append(oproof[pos: pos + 32])
|
||||
pos += 32
|
||||
|
121
trie_research/bintrie2/new_bintrie_optimized.py
Normal file
121
trie_research/bintrie2/new_bintrie_optimized.py
Normal file
@ -0,0 +1,121 @@
|
||||
from ethereum.utils import sha3, encode_hex
|
||||
|
||||
class EphemDB():
|
||||
def __init__(self, kv=None):
|
||||
self.reads = 0
|
||||
self.writes = 0
|
||||
self.kv = kv or {}
|
||||
|
||||
def get(self, k):
|
||||
self.reads += 1
|
||||
return self.kv.get(k, None)
|
||||
|
||||
def put(self, k, v):
|
||||
self.writes += 1
|
||||
self.kv[k] = v
|
||||
|
||||
def delete(self, k):
|
||||
del self.kv[k]
|
||||
|
||||
# Hashes of empty subtrees
|
||||
zerohashes = [b'\x00' * 32]
|
||||
for i in range(256):
|
||||
zerohashes.insert(0, sha3(zerohashes[0] + zerohashes[0]))
|
||||
|
||||
# Create a new empty tree
|
||||
def new_tree(db):
|
||||
return zerohashes[0]
|
||||
|
||||
# Convert a binary key into an integer path value
|
||||
def key_to_path(k):
|
||||
return int.from_bytes(k, 'big')
|
||||
|
||||
tt256m1 = 2**256 - 1
|
||||
|
||||
# And convert back
|
||||
def path_to_key(k):
|
||||
return (k & tt256m1).to_bytes(32, 'big')
|
||||
|
||||
# Read a key from a given tree
|
||||
def get(db, root, key):
|
||||
v = root
|
||||
path = key_to_path(key)
|
||||
for i in range(256):
|
||||
if v == zerohashes[i]:
|
||||
return b'\x00' * 32
|
||||
child = db.get(v)
|
||||
if len(child) == 65:
|
||||
if (path % 2**256) == key_to_path(child[1:33]):
|
||||
return child[33:]
|
||||
else:
|
||||
return b'\x00' * 32
|
||||
else:
|
||||
if (path >> 255) & 1:
|
||||
v = child[32:]
|
||||
else:
|
||||
v = child[:32]
|
||||
path <<= 1
|
||||
return v
|
||||
|
||||
# Make a root hash of a (sub)tree with a single key/value pair
|
||||
def make_single_key_hash(path, depth, value):
|
||||
if depth == 256:
|
||||
return value
|
||||
elif (path >> 255) & 1:
|
||||
return sha3(zerohashes[depth+1] + make_single_key_hash(path << 1, depth + 1, value))
|
||||
else:
|
||||
return sha3(make_single_key_hash(path << 1, depth + 1, value) + zerohashes[depth+1])
|
||||
|
||||
# Make a root hash of a (sub)tree with two key/value pairs, and save intermediate nodes in the DB
|
||||
def make_double_key_hash(db, path1, path2, depth, value1, value2):
|
||||
if depth == 256:
|
||||
raise Exception("Cannot fit two values into one slot!")
|
||||
if (path1 >> 255) & 1:
|
||||
if (path2 >> 255) & 1:
|
||||
child = zerohashes[depth+1] + make_double_key_hash(db, path1 << 1, path2 << 1, depth + 1, value1, value2)
|
||||
db.put(sha3(child), child)
|
||||
return sha3(child)
|
||||
else:
|
||||
L = make_single_key_hash(path2 << 1, depth + 1, value2)
|
||||
R = make_single_key_hash(path1 << 1, depth + 1, value1)
|
||||
db.put(L, b'\x01' + path_to_key(path2 << 1) + value2)
|
||||
db.put(R, b'\x01' + path_to_key(path1 << 1) + value1)
|
||||
child = L + R
|
||||
else:
|
||||
if (path2 >> 255) & 1:
|
||||
L = make_single_key_hash(path1 << 1, depth + 1, value1)
|
||||
R = make_single_key_hash(path2 << 1, depth + 1, value2)
|
||||
db.put(L, b'\x01' + path_to_key(path1 << 1) + value1)
|
||||
db.put(R, b'\x01' + path_to_key(path2 << 1) + value2)
|
||||
child = L + R
|
||||
else:
|
||||
child = make_double_key_hash(db, path1 << 1, path2 << 1, depth + 1, value1, value2) + zerohashes[depth+1]
|
||||
db.put(sha3(child), child)
|
||||
return sha3(child)
|
||||
|
||||
# Update a tree with a given key/value pair
|
||||
def update(db, root, key, value):
|
||||
return _update(db, root, key_to_path(key), 0, value)
|
||||
|
||||
def _update(db, root, path, depth, value):
|
||||
if depth == 256:
|
||||
return value
|
||||
# Update an empty subtree: make a single-key subtree
|
||||
if root == zerohashes[depth]:
|
||||
k = make_single_key_hash(path, depth, value)
|
||||
db.put(k, b'\x01' + path_to_key(path) + value)
|
||||
return k
|
||||
child = db.get(root)
|
||||
# Update a single-key subtree: make a double-key subtree
|
||||
if len(child) == 65:
|
||||
origpath, origvalue = key_to_path(child[1:33]), child[33:]
|
||||
return make_double_key_hash(db, path, origpath, depth, value, origvalue)
|
||||
# Update a multi-key subtree: recurse down
|
||||
elif (path >> 255) & 1:
|
||||
new_child = child[:32] + _update(db, child[32:], path << 1, depth + 1, value)
|
||||
db.put(sha3(new_child), new_child)
|
||||
return sha3(new_child)
|
||||
else:
|
||||
new_child = _update(db, child[:32], path << 1, depth + 1, value) + child[32:]
|
||||
db.put(sha3(new_child), new_child)
|
||||
return sha3(new_child)
|
Loading…
x
Reference in New Issue
Block a user