const bigInt = require("snarkjs").bigInt; const SMTMemDB = require("./smt_memdb"); const mimc7 = require("./mimc7"); function smtHash(arr) { let r = mimc7.getIV(); for (let i=0; i=0; level--) { let oldNode, newNode; const sibling = resFind.siblings[level]; if (keyBits[level]) { oldNode = [sibling, rtOld]; newNode = [sibling, rtNew]; } else { oldNode = [rtOld, sibling, ]; newNode = [rtNew, sibling, ]; } rtOld = smtHash(oldNode); rtNew = smtHash(newNode); dels.push(rtOld); ins.push([rtNew, newNode]); } res.newRoot = rtNew; await this.db.multiIns(ins); await this.db.setRoot(rtNew); this.root = rtNew; await this.db.multiDel(dels); return res; } async delete(_key) { const key = bigInt(_key); const resFind = await this.find(key); if (!resFind.found) throw new Error("Key does not exists"); const res = { siblings: [], delKey: key, delValue: resFind.foundValue }; const dels = []; const ins = []; let rtOld = smtHash([1, key, resFind.foundValue]); let rtNew; dels.push(rtOld); let mixed; if (resFind.siblings.length > 0) { const record = await this.db.get(resFind.siblings[resFind.siblings.length - 1]); if ((record.length == 3)&&(record[0].equals(bigInt.one))) { mixed = false; res.oldKey = record[1]; res.oldValue = record[2]; res.isOld0 = false; rtNew = resFind.siblings[resFind.siblings.length - 1]; } else if (record.length == 2) { mixed = true; res.oldKey = key; res.oldValue = bigInt(0); res.isOld0 = true; rtNew = bigInt.zero; } else { throw new Error("Invalid node. Database corrupted"); } } else { rtNew = bigInt.zero; res.oldKey = key; res.oldValue = bigInt(0); res.isOld0 = true; } const keyBits = this._splitBits(key); for (let level = resFind.siblings.length-1; level >=0; level--) { let newSibling = resFind.siblings[level]; if ((level == resFind.siblings.length-1)&&(!res.isOld0)) { newSibling = bigInt.zero; } const oldSibling = resFind.siblings[level]; if (keyBits[level]) { rtOld = smtHash([oldSibling, rtOld]); } else { rtOld = smtHash([rtOld, oldSibling]); } dels.push(rtOld); if (!newSibling.isZero()) { mixed = true; } if (mixed) { res.siblings.unshift(resFind.siblings[level]); let newNode; if (keyBits[level]) { newNode = [newSibling, rtNew]; } else { newNode = [rtNew, newSibling]; } rtNew = smtHash(newNode); ins.push([rtNew, newNode]); } } await this.db.multiIns(ins); await this.db.setRoot(rtNew); this.root = rtNew; await this.db.multiDel(dels); res.newRoot = rtNew; res.oldRoot = rtOld; return res; } async insert(_key, _value) { const key = bigInt(_key); const value = bigInt(_value); let addedOne = false; const res = {}; res.oldRoot = this.root; const newKeyBits = this._splitBits(key); let rtOld; const resFind = await this.find(key); if (resFind.found) throw new Error("Key already exists"); res.siblings = resFind.siblings; let mixed; if (!resFind.isOld0) { const oldKeyits = this._splitBits(resFind.notFoundKey); for (let i= res.siblings.length; oldKeyits[i] == newKeyBits[i]; i++) { res.siblings.push(bigInt.zero); } rtOld = smtHash([1, resFind.notFoundKey, resFind.notFoundValue]); res.siblings.push(rtOld); addedOne = true; mixed = false; } else if (res.siblings.length >0) { mixed = true; rtOld = bigInt.zero; } const inserts = []; const dels = []; let rt = smtHash([1, key, value]); inserts.push([rt,[1, key, value]] ); for (let i=res.siblings.length-1; i>=0; i--) { if ((i0) && (res.siblings[res.siblings.length-1].isZero())) { res.siblings.pop(); } res.oldKey = resFind.notFoundKey; res.oldValue = resFind.notFoundValue; res.newRoot = rt; res.isOld0 = resFind.isOld0; await this.db.multiIns(inserts); await this.db.setRoot(rt); this.root = rt; await this.db.multiDel(dels); return res; } async find(key) { const keyBits = this._splitBits(key); return await this._find(key, keyBits, this.root, 0); } async _find(key, keyBits, root, level) { if (typeof root === "undefined") root = this.root; let res; if (root.isZero()) { res = { found: false, siblings: [], notFoundKey: key, notFoundValue: bigInt.zero, isOld0: true }; return res; } const record = await this.db.get(root); if ((record.length==3)&&(record[0].equals(bigInt.one))) { if (record[1].equals(key)) { res = { found: true, siblings: [], foundValue: record[2], isOld0: false }; } else { res = { found: false, siblings: [], notFoundKey: record[1], notFoundValue: record[2], isOld0: false }; } } else { if (keyBits[level] == 0) { res = await this._find(key, keyBits, record[0], level+1); res.siblings.unshift(record[1]); } else { res = await this._find(key, keyBits, record[1], level+1); res.siblings.unshift(record[0]); } } return res; } } async function loadFromFile(fileName) { } async function newMemEmptyTrie() { const db = new SMTMemDB(); const rt = await db.getRoot(); const smt = new SMT(db, rt); return smt; } module.exports.loadFromFile = loadFromFile; module.exports.newMemEmptyTrie = newMemEmptyTrie;