-
Notifications
You must be signed in to change notification settings - Fork 145
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
Add documentation for the Indexed Merkle Map API #1088
base: main
Are you sure you want to change the base?
Conversation
The latest updates on your projects. Learn more about Vercel for Git ↗︎
1 Skipped Deployment
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think it's good to have this article alongside the typedoc, but I'd personally prefer to highlight ZkProgram rather than smart contract.
I'm also open to just promoting the API out of experimental, especially with this article written. Let's see what @Trivo25 thinks 🤔 , so if we do that, we can also remove some of the Experimental
references here.
This is possible because the `IndexedMerkleMap` can be used within provable code. | ||
|
||
```ts | ||
class MyContract extends SmartContract { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I would rather write new examples using zkProgram since we have more confidence that it will not be changed soon. SmartContract
is more likely to be modified by the new API.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Regardless of documentation maintainability, I think that demonstrating a SmartContract
example for the Indexed Merkle Map
API provides greater benefits for developers and serves as more effective documentation.
When someone wants to use the Indexed Merkle Map API, they will likely integrate it directly within a smart contract, as it works seamlessly in provable code.
In contrast, a ZkProgram
example might lead readers to mistakenly believe that the API should be used with a ZkProgram
. This misunderstanding could imply that verifying a zkProgram proof for Indexed Merkle Map updates is straightforward, when in fact it is only practical in advanced scenarios such as recursive reducers.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This misunderstanding could imply that verifying a zkProgram proof for Indexed Merkle Map updates is straightforward, when in fact it is only practical in advanced scenarios such as recursive reducers.
Maybe I don't understand this myself. What's wrong with this example:
import { Experimental, Field, SelfProof, ZkProgram } from 'o1js';
const height = 5;
class MyMerkleMap extends Experimental.IndexedMerkleMap(height) {}
const RecursiveProgram = ZkProgram({
name: 'RecursiveProgram',
publicOutput: MyMerkleMap,
methods: {
init: {
privateInputs: [],
method: async () => {
const map = new MyMerkleMap();
return { publicOutput: map };
},
},
insert: {
privateInputs: [SelfProof, Field, Field],
method: async (
proof: SelfProof<undefined, MyMerkleMap>,
key: Field,
value: Field
) => {
proof.verify();
const newMap = proof.publicOutput.clone();
newMap.insert(key, value);
return { publicOutput: newMap };
},
},
update: {
privateInputs: [SelfProof, Field, Field],
method: async (
proof: SelfProof<undefined, MyMerkleMap>,
key: Field,
value: Field
) => {
proof.verify();
const newMap = proof.publicOutput.clone();
newMap.update(key, value);
return { publicOutput: newMap };
},
},
},
});
await RecursiveProgram.compile();
class RecursiveProof extends ZkProgram.Proof(RecursiveProgram) {}
const p = await RecursiveProgram.init();
const p2 = await RecursiveProgram.insert(p.proof, Field(1), Field(2));
const p3 = await RecursiveProgram.update(p2.proof, Field(1), Field(3));
console.log(p3.proof.publicOutput.get(Field(1))); // 3
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There's nothing wrong with the example but I think whoever wants to use the Indexed MM
would use it directly within a zkApp as it's more practical and for that, I think a SmartContract
example would be a better guide for the reader and less confusing.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think whoever wants to use the Indexed MM would use it directly within a zkApp as it's more practical
I'm not sold on this. Do you actually believe indexed merkle map is a usable solution for zkapps today? To me, your example is misleading because it teases the existence of a map
API that works on Mina, but doesn't address the same old concurrent update or state storage issues that exist for every zkapp. In production, this example would not work with multiple users accessing it at once, and I want to stop creating new examples that break down at scale.
I know that this is a touchy issue in the community right now as protokit devs are looking for L1 alternatives to the StateMap
so that they can deploy on mainnet: (discord example). IMO there is absolutely no alternative that currently works, and I don't want to beat around the bush with that fact and pretend this is a viable alternative. Do you disagree with this?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fair enough, I don't disagree!
I see your point here! But that's also applicable to many primitives in the documentation. I was trying to keep it consistent with the current o1js docs as I was documenting the Indexed Merkle Map
API.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My goal with example code in the docs is for developers to grasp the concept and then take it to build one step further. My problem with the pattern we have in most of our existing documentation is that it only works to a certain point, and that's what we show. When someone tries to take the next step, they hit a wall.
What I like about ZkProgram
is that the final product can be a Mina ZkApp, or it could live on protokit or zeko. And before deploying it to any network, a developer can change different variables, combine it with other ZkProgram
s, and extend it in all kinds of ways locally.
The Indexed Merkle Map is an improved version of the [MerkleMap](/zkapps/tutorials/common-types-and-functions#merkle-map), offering enhanced efficiency and usability: | ||
|
||
- **Reduced Constraints:** Uses 4-8x fewer constraints than `MerkleMap`. | ||
- **Support for Dummy Updates:** Handles updates that don't affect the root for keys like `0` and `-1`. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My understanding was that a dummy update was just (0,0) 🤔 how about something like:
- Support for -1 and 0 as keys: Useful for creating dummy updates (with key and value both set to zero) which do not alter the map root
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the point of a dummy update? I think that is the main thing to elaborate on. Saying we can create an update that doesn't affect the root doesn't intuitively mean anything to me. Is the useful part that we can, for instance, use it with Provable.if
to sometimes set a value, and sometimes not? Are there other useful cases? Maybe we could find an intuitive case to highlight and focus on the value of a dummy update, rather than the definition?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No description provided.