research/sparse_merkle_tree/new_bintrie_optimized.py

127 lines
4.2 KiB
Python

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)
def multi_update(db, root, keys, values):
for k, v in zip(keys, values):
root = update(db, root, k, v)
return root