Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: refactor StandardIndexedTree for abstract leaves and preimages and optimized it #3530

Merged
merged 22 commits into from
Dec 5, 2023
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
880b702
wip abstracting leaf and preimage in indexed
sirasistant Dec 1, 2023
ed471d4
feat: experiment with generic indexed trees
sirasistant Dec 1, 2023
38fcf05
chore: add test dep to devDependencies
sirasistant Dec 1, 2023
bc4915b
chore: remove old LeafData struct
sirasistant Dec 1, 2023
47a3aee
fix: getLatestLeafPreimage can return undefined
sirasistant Dec 1, 2023
064d7a8
feat: remove local leaf cache in indexed tree
sirasistant Dec 1, 2023
d458756
feat: optimize findLeafIndex for indexed tree
sirasistant Dec 1, 2023
ae6b6a0
docs: added jsdoc
sirasistant Dec 4, 2023
47bb7f0
refactor: low leaf getter returns option
sirasistant Dec 4, 2023
6d7fc7d
Merge branch 'master' into arv/public_data_tree_in
sirasistant Dec 4, 2023
7250939
fix: make sure low leaf in range
sirasistant Dec 4, 2023
240e8b5
fix: improve performance of getLeafIndex
sirasistant Dec 4, 2023
bcc8ae0
chore: cleanups
sirasistant Dec 4, 2023
1b7c9f8
style: fix formatting
sirasistant Dec 4, 2023
3848116
Merge branch 'master' into arv/public_data_tree_indexed
sirasistant Dec 4, 2023
08791c5
chore: reset file to master
sirasistant Dec 4, 2023
989ad09
Merge branch 'master' into arv/public_data_tree_indexed
sirasistant Dec 4, 2023
edb4548
fix: addressed some PR comments
sirasistant Dec 5, 2023
e731f97
fix: avoid findLeafIndex in sparse trees
sirasistant Dec 5, 2023
037a479
refactor: avoid createTree helper
sirasistant Dec 5, 2023
a217fe2
fix: formatting
sirasistant Dec 5, 2023
2a77b8b
Merge branch 'master' into arv/public_data_tree_indexed
sirasistant Dec 5, 2023
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ import { pedersenHash } from '@aztec/foundation/crypto';
import { EthAddress } from '@aztec/foundation/eth-address';
import { Fr, GrumpkinScalar } from '@aztec/foundation/fields';
import { DebugLogger, createDebugLogger } from '@aztec/foundation/log';
import { AppendOnlyTree, Pedersen, StandardTree, newTree } from '@aztec/merkle-tree';
import { AppendOnlyTree, Pedersen, StandardTree, newTree, treeBuilder } from '@aztec/merkle-tree';
import {
ChildContractArtifact,
ImportTestContractArtifact,
Expand Down Expand Up @@ -126,7 +126,7 @@ describe('Private Execution test suite', () => {
if (!trees[name]) {
const db = levelup(createMemDown());
const pedersen = new Pedersen();
trees[name] = await newTree(StandardTree, db, pedersen, name, treeHeights[name]);
trees[name] = await newTree(treeBuilder(StandardTree), db, pedersen, name, treeHeights[name]);
}
await trees[name].appendLeaves(leaves.map(l => l.toBuffer()));

Expand Down
26 changes: 13 additions & 13 deletions yarn-project/aztec-node/src/aztec-node/server.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import {
L1_TO_L2_MSG_TREE_HEIGHT,
NOTE_HASH_TREE_HEIGHT,
NULLIFIER_TREE_HEIGHT,
NullifierLeafPreimage,
PUBLIC_DATA_TREE_HEIGHT,
} from '@aztec/circuits.js';
import { computeGlobalsHash, computePublicDataTreeIndex } from '@aztec/circuits.js/abis';
Expand Down Expand Up @@ -429,19 +430,19 @@ export class AztecNodeService implements AztecNode {
return undefined;
}

const leafDataPromise = db.getLeafData(MerkleTreeId.NULLIFIER_TREE, Number(index));
const leafPreimagePromise = db.getLeafPreimage(MerkleTreeId.NULLIFIER_TREE, index);
const siblingPathPromise = db.getSiblingPath<typeof NULLIFIER_TREE_HEIGHT>(
MerkleTreeId.NULLIFIER_TREE,
BigInt(index),
);

const [leafData, siblingPath] = await Promise.all([leafDataPromise, siblingPathPromise]);
const [leafPreimage, siblingPath] = await Promise.all([leafPreimagePromise, siblingPathPromise]);

if (!leafData) {
if (!leafPreimage) {
return undefined;
}

return new NullifierMembershipWitness(BigInt(index), leafData, siblingPath);
return new NullifierMembershipWitness(BigInt(index), leafPreimage as NullifierLeafPreimage, siblingPath);
}

/**
Expand All @@ -463,22 +464,21 @@ export class AztecNodeService implements AztecNode {
nullifier: Fr,
): Promise<NullifierMembershipWitness | undefined> {
const committedDb = await this.#getWorldState(blockNumber);
const { index, alreadyPresent } = await committedDb.getPreviousValueIndex(
MerkleTreeId.NULLIFIER_TREE,
nullifier.toBigInt(),
);
const findResult = await committedDb.getPreviousValueIndex(MerkleTreeId.NULLIFIER_TREE, nullifier.toBigInt());
if (!findResult) {
return undefined;
}
const { index, alreadyPresent } = findResult;
if (alreadyPresent) {
this.log.warn(`Nullifier ${nullifier.toBigInt()} already exists in the tree`);
}
const leafData = await committedDb.getLeafData(MerkleTreeId.NULLIFIER_TREE, index);
if (!leafData) {
return undefined;
}
const preimageData = (await committedDb.getLeafPreimage(MerkleTreeId.NULLIFIER_TREE, index))!;

const siblingPath = await committedDb.getSiblingPath<typeof NULLIFIER_TREE_HEIGHT>(
MerkleTreeId.NULLIFIER_TREE,
BigInt(index),
);
return new NullifierMembershipWitness(BigInt(index), leafData, siblingPath);
return new NullifierMembershipWitness(BigInt(index), preimageData as NullifierLeafPreimage, siblingPath);
}

/**
Expand Down
93 changes: 85 additions & 8 deletions yarn-project/circuits.js/src/structs/rollup/base_rollup.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import { toBigIntBE, toBufferBE } from '@aztec/foundation/bigint-buffer';
import { Fr } from '@aztec/foundation/fields';
import { BufferReader, Tuple } from '@aztec/foundation/serialize';
import { IndexedTreeLeaf, IndexedTreeLeafPreimage } from '@aztec/foundation/trees';

import {
BLOCKS_TREE_HEIGHT,
Expand All @@ -25,28 +27,103 @@ import { AppendOnlyTreeSnapshot } from './append_only_tree_snapshot.js';
* Class containing the data of a preimage of a single leaf in the nullifier tree.
* Note: It's called preimage because this data gets hashed before being inserted as a node into the `IndexedTree`.
*/
export class NullifierLeafPreimage {
export class NullifierLeafPreimage implements IndexedTreeLeafPreimage {
constructor(
/**
* Leaf value inside the indexed tree's linked list.
*/
public leafValue: Fr,
public nullifier: Fr,
/**
* Next value inside the indexed tree's linked list.
*/
public nextValue: Fr,
public nextNullifier: Fr,
/**
* Index of the next leaf in the indexed tree's linked list.
*/
public nextIndex: UInt32,
public nextIndex: bigint,
) {}

toBuffer() {
return serializeToBuffer(this.leafValue, this.nextValue, this.nextIndex);
getKey(): bigint {
return this.nullifier.toBigInt();
}

getNextKey(): bigint {
return this.nextNullifier.toBigInt();
}

getNextIndex(): bigint {
return this.nextIndex;
}

asLeaf(): NullifierLeaf {
return new NullifierLeaf(this.nullifier);
}

toBuffer(): Buffer {
return Buffer.concat(this.toHashInputs());
}

toHashInputs(): Buffer[] {
return [
Buffer.from(this.nullifier.toBuffer()),
Buffer.from(toBufferBE(this.nextIndex, 32)),
Buffer.from(this.nextNullifier.toBuffer()),
];
}

clone(): NullifierLeafPreimage {
return new NullifierLeafPreimage(this.nullifier, this.nextNullifier, this.nextIndex);
}

static empty(): NullifierLeafPreimage {
return new NullifierLeafPreimage(Fr.ZERO, Fr.ZERO, 0n);
}

static fromBuffer(buf: Buffer): NullifierLeafPreimage {
const nullifier = Fr.fromBuffer(buf.subarray(0, 32));
const nextIndex = toBigIntBE(buf.subarray(32, 64));
const nextNullifier = Fr.fromBuffer(buf.subarray(64, 96));
return new NullifierLeafPreimage(nullifier, nextNullifier, nextIndex);
}

static fromLeaf(leaf: NullifierLeaf, nextKey: bigint, nextIndex: bigint): NullifierLeafPreimage {
return new NullifierLeafPreimage(leaf.nullifier, new Fr(nextKey), nextIndex);
}

static clone(preimage: NullifierLeafPreimage): NullifierLeafPreimage {
return new NullifierLeafPreimage(preimage.nullifier, preimage.nextNullifier, preimage.nextIndex);
}
}

/**
* A nullifier to be inserted in the nullifier tree.
*/
export class NullifierLeaf implements IndexedTreeLeaf {
constructor(
/**
* Nullifier value.
*/
public nullifier: Fr,
) {}

getKey(): bigint {
return this.nullifier.toBigInt();
}

toBuffer(): Buffer {
return this.nullifier.toBuffer();
}

isEmpty(): boolean {
return this.nullifier.isZero();
}

static buildDummy(key: bigint): NullifierLeaf {
return new NullifierLeaf(new Fr(key));
}

static empty() {
return new NullifierLeafPreimage(Fr.ZERO, Fr.ZERO, 0);
static fromBuffer(buf: Buffer): NullifierLeaf {
return new NullifierLeaf(Fr.fromBuffer(buf));
}
}

Expand Down
2 changes: 1 addition & 1 deletion yarn-project/circuits.js/src/tests/factories.ts
Original file line number Diff line number Diff line change
Expand Up @@ -906,7 +906,7 @@ export function makeBaseRollupInputs(seed = 0): BaseRollupInputs {

const lowNullifierLeafPreimages = makeTuple(
MAX_NEW_NULLIFIERS_PER_BASE_ROLLUP,
x => new NullifierLeafPreimage(fr(x), fr(x + 0x100), x + 0x200),
x => new NullifierLeafPreimage(fr(x), fr(x + 0x100), BigInt(x + 0x200)),
seed + 0x1000,
);

Expand Down
10 changes: 8 additions & 2 deletions yarn-project/end-to-end/src/e2e_blacklist_token_contract.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@ import {
computeAuthWitMessageHash,
computeMessageSecretHash,
} from '@aztec/aztec.js';
import { Pedersen, SparseTree, newTree } from '@aztec/merkle-tree';
import { Pedersen, SparseTree, newTree, treeBuilder } from '@aztec/merkle-tree';
import { SlowTreeContract, TokenBlacklistContract, TokenContract } from '@aztec/noir-contracts/types';

import { jest } from '@jest/globals';
Expand Down Expand Up @@ -107,7 +107,13 @@ describe('e2e_blacklist_token_contract', () => {
slowTree = await SlowTreeContract.deploy(wallets[0]).send().deployed();

const depth = 254;
slowUpdateTreeSimulator = await newTree(SparseTree, levelup(createMemDown()), new Pedersen(), 'test', depth);
slowUpdateTreeSimulator = await newTree(
treeBuilder(SparseTree),
levelup(createMemDown()),
new Pedersen(),
'test',
depth,
);

const deployTx = TokenBlacklistContract.deploy(wallets[0], accounts[0], slowTree.address).send({});
const receipt = await deployTx.wait();
Expand Down
10 changes: 8 additions & 2 deletions yarn-project/end-to-end/src/e2e_slow_tree.test.ts
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
/* eslint-disable camelcase */
import { CheatCodes, DebugLogger, Fr, Wallet } from '@aztec/aztec.js';
import { Pedersen, SparseTree, newTree } from '@aztec/merkle-tree';
import { Pedersen, SparseTree, newTree, treeBuilder } from '@aztec/merkle-tree';
import { SlowTreeContract } from '@aztec/noir-contracts/types';

import { default as levelup } from 'levelup';
Expand All @@ -27,7 +27,13 @@ describe('e2e_slow_tree', () => {

it('Messing around with noir slow tree', async () => {
const depth = 254;
const slowUpdateTreeSimulator = await newTree(SparseTree, levelup(createMemDown()), new Pedersen(), 'test', depth);
const slowUpdateTreeSimulator = await newTree(
treeBuilder(SparseTree),
levelup(createMemDown()),
new Pedersen(),
'test',
depth,
);
const getMembershipProof = async (index: bigint, includeUncommitted: boolean) => {
return {
index,
Expand Down
4 changes: 1 addition & 3 deletions yarn-project/end-to-end/tsconfig.json
Original file line number Diff line number Diff line change
Expand Up @@ -55,7 +55,5 @@
"path": "../world-state"
}
],
"include": [
"src"
]
"include": ["src"]
}
1 change: 1 addition & 0 deletions yarn-project/foundation/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@
"./sleep": "./dest/sleep/index.js",
"./timer": "./dest/timer/index.js",
"./transport": "./dest/transport/index.js",
"./trees": "./dest/trees/index.js",
"./wasm": "./dest/wasm/index.js",
"./worker": "./dest/worker/index.js",
"./bigint-buffer": "./dest/bigint-buffer/index.js",
Expand Down
1 change: 1 addition & 0 deletions yarn-project/foundation/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ export * as serialize from './serialize/index.js';
export * as sleep from './sleep/index.js';
export * as timer from './timer/index.js';
export * as transport from './transport/index.js';
export * as trees from './trees/index.js';
export * as types from './types/index.js';
export * as url from './url/index.js';
export * as wasm from './wasm/index.js';
Expand Down
48 changes: 48 additions & 0 deletions yarn-project/foundation/src/trees/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
/**
* A leaf of an indexed merkle tree.
*/
export interface IndexedTreeLeaf {
/**
* Returns key of the leaf. It's used for indexing.
*/
getKey(): bigint;
/**
* Serializes the leaf into a buffer.
*/
toBuffer(): Buffer;
/**
* Returns true if the leaf is empty.
*/
isEmpty(): boolean;
}

/**
* Preimage of an indexed merkle tree leaf.
*/
export interface IndexedTreeLeafPreimage {
/**
* Returns key of the leaf corresponding to this preimage.
*/
getKey(): bigint;
/**
* Returns the key of the next leaf.
*/
getNextKey(): bigint;
/**
* Returns the index of the next leaf.
*/
getNextIndex(): bigint;

/**
* Returns the preimage as a leaf.
*/
asLeaf(): IndexedTreeLeaf;
/**
* Serializes the preimage into a buffer.
*/
toBuffer(): Buffer;
/**
* Serializes the preimage to an array of buffers for hashing.
*/
toHashInputs(): Buffer[];
}
1 change: 1 addition & 0 deletions yarn-project/merkle-tree/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@
"tslib": "^2.4.0"
},
"devDependencies": {
"@aztec/circuits.js": "workspace:^",
"@jest/globals": "^29.5.0",
"@types/jest": "^29.5.0",
"@types/levelup": "^5.1.2",
Expand Down
4 changes: 2 additions & 2 deletions yarn-project/merkle-tree/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,10 @@ export * from './interfaces/merkle_tree.js';
export * from './interfaces/update_only_tree.js';
export * from './pedersen.js';
export * from './sparse_tree/sparse_tree.js';
export { LowLeafWitnessData, StandardIndexedTree } from './standard_indexed_tree/standard_indexed_tree.js';
export { StandardIndexedTree } from './standard_indexed_tree/standard_indexed_tree.js';
export * from './standard_tree/standard_tree.js';
export { INITIAL_LEAF } from './tree_base.js';
export { newTree } from './new_tree.js';
export { newTree, builder as treeBuilder } from './new_tree.js';
export { loadTree } from './load_tree.js';
export * from './snapshots/snapshot_builder.js';
export * from './snapshots/full_snapshot.js';
Expand Down
Loading