You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Our team Solus is building a Mixer for Mina Protocol. For the Mixer to work we are emitting a Deposit event as a primitive approach to store the Merkle Tree outside the smart contract, also we are emitting a Nullifier event to track withdrawals and avoid double-spending. It looks something like this:
On the first request the events are fetched and all it’s fine. But, in the second request ( await zkapp.fetchEvents() ) this error starts to appear:
We needed this logic to work to finish the Mixer so we debugged to find if the mistake was on our side. After debugging for a while, we found that the error comes directly from SnarkyJS. Specifically, it happens in the fetchEvents function of the SmartContract class: https://github.com/o1-labs/snarkyjs/blob/main/src/lib/zkapp.ts#L971.
Conditions to replicate the error
The smart contract has more than one type of event.
The smartContract.fetchEvents(…) function is called more than once.
The smart contract is executed on a local blockchain.
In the Local Blockchain events are stored as an object in memory:
constevents: Record<string,any>={};
When the events of a specific Public Key are fetched from Mina (the Local Blockchain) the function fetchEvents(...) is not returning a new object, it is returning a reference to the events object that already exists. This is due to the way how JS stores compound objects in memory.
// mina.ts in the Local Mina Blockchain// Line 510:asyncfetchEvents(
publicKey: PublicKey,tokenId: Field=TokenId.default): Promise<any[]>{returnevents?.[publicKey.toBase58()]?.[TokenId.toBase58(tokenId)]??[];}
// zkapp.ts in SmartContract.fetchEvents(..)letevents=(awaitMina.fetchEvents(this.address,this.self.body.tokenId)).filter((el: any)=>{letslot=UInt32.from(el.slot);returnend===undefined
? start.lte(slot).toBoolean()
: start.lte(slot).toBoolean()&&slot.lte(end).toBoolean();}).map((el: any)=>el.events).flat();// events = [event, event]// Where each event inside events is a reference to the original event stored in memory.
Each one of the events come as an array in the following format:
events[i]=event=[eventTypeIndex, ...n]// where n are the n parameters that the event has.
When the map function iterates over each event if the smart contract has more than one type of event it executes the following code:
...
}else{// if there are multiple events we have to use the index event[0] to find the exact event typelettype=sortedEventTypes[event[0]];// all other elements of the array are values used to construct the original object, we can drop the first value since its just an indexevent.shift();return{
type,event: this.events[type].fromFields(event.map((f: string)=>Field(f))),};}
The problem happens in line 1007 (https://github.com/o1-labs/snarkyjs/blob/main/src/lib/zkapp.ts#L1007) when this.events[type] ends up being undefined and therefore this.events[type].fromFields fails. The reason why this.events[type] is undefined is because in the second request we make (zkapp.fetchEvents() )the variable type is undefined, while in the first request no error occurs because type is indeed the type of the event. In our case “deposit” or “nullifier”.
When event.shift() is executed it deletes the first element of the event array and is logical to think that it doesn’t modify anything beyond the local scope of the callback function inside events.map , because the map function by itself is non-destructive. However, the callback inside the map function could be destructive and each event inside map() is not a new array, but a reference pointing to the original event array that was fetched from the Mina local blockchain at the beginning of the function. So the original events object ends up being mutated.
That is why on the second request to smartcontract.fetchEvents(...) the type is undefined, this happens because the element at the first index of the event array is not anymore the index of the type of event (it was deleted in the last execution with the event.shift()). Instead it is the element that should be at event[1].
Quick solution
It could be solved using a non-destructive function such as .slice() that creates a new array from the original event array.
...
}else{// if there are multiple events we have to use the index event[0] to find the exact event typelettype=sortedEventTypes[event[0]];// all other elements of the array are values used to construct the original object, we can drop the first value since its just an indexleteventProps=event.slice(1);return{
type,event: this.events[type].fromFields(eventProps.map((f: string)=>Field(f))),};}
Finally
It is useful to clarify that this error is only an error that happens in local testing when the local blockchain is executed.
When the fetchEvents function is implemented in the original Mina network, a remote request to the blockchain is going to be implemented instead of a simple in-memory reference to an events object stored in local memory.
asyncfetchEvents(){throwError('fetchEvents() is not implemented yet for remote blockchains.');},
I hope the explanation was not too long.
Thank you for reading guys.
The text was updated successfully, but these errors were encountered:
Overview
Our team Solus is building a Mixer for Mina Protocol. For the Mixer to work we are emitting a Deposit event as a primitive approach to store the Merkle Tree outside the smart contract, also we are emitting a Nullifier event to track withdrawals and avoid double-spending. It looks something like this:
Until here everything is ok. The problem comes when we fetch the events from the smart contract.
On the first request the events are fetched and all it’s fine. But, in the second request (
await zkapp.fetchEvents()
) this error starts to appear:We needed this logic to work to finish the Mixer so we debugged to find if the mistake was on our side. After debugging for a while, we found that the error comes directly from SnarkyJS. Specifically, it happens in the fetchEvents function of the SmartContract class: https://github.com/o1-labs/snarkyjs/blob/main/src/lib/zkapp.ts#L971.
Conditions to replicate the error
smartContract.fetchEvents(…)
function is called more than once.The error in SmartContract.fetchEvents
The problem that is happening between the first and the second request lies on the way Mina stores the events in the LocalBlockchain and the instruction:
event.shift()
in the line 1004 (https://github.com/o1-labs/snarkyjs/blob/main/src/lib/zkapp.ts#L1004).In the Local Blockchain events are stored as an object in memory:
When the events of a specific Public Key are fetched from Mina (the Local Blockchain) the function
fetchEvents(...)
is not returning a new object, it is returning a reference to theevents
object that already exists. This is due to the way how JS stores compound objects in memory.Each one of the events come as an array in the following format:
When the map function iterates over each
event
if the smart contract has more than one type of event it executes the following code:The problem happens in line 1007 (https://github.com/o1-labs/snarkyjs/blob/main/src/lib/zkapp.ts#L1007) when
this.events[type]
ends up being undefined and thereforethis.events[type].fromFields
fails. The reason whythis.events[type]
is undefined is because in the second request we make (zkapp.fetchEvents()
)the variabletype
is undefined, while in the first request no error occurs becausetype
is indeed the type of the event. In our case “deposit” or “nullifier”.When
event.shift()
is executed it deletes the first element of theevent
array and is logical to think that it doesn’t modify anything beyond the local scope of the callback function insideevents.map
, because the map function by itself is non-destructive. However, the callback inside the map function could be destructive and each event insidemap()
is not a new array, but a reference pointing to the original event array that was fetched from theMina
local blockchain at the beginning of the function. So the original events object ends up being mutated.That is why on the second request to
smartcontract.fetchEvents(...)
the type is undefined, this happens because the element at the first index of theevent
array is not anymore the index of the type of event (it was deleted in the last execution with theevent.shift()
). Instead it is the element that should be atevent[1]
.Quick solution
It could be solved using a non-destructive function such as
.slice()
that creates a new array from the originalevent
array.Finally
It is useful to clarify that this error is only an error that happens in local testing when the local blockchain is executed.
When the
fetchEvents
function is implemented in the original Mina network, a remote request to the blockchain is going to be implemented instead of a simple in-memory reference to anevents
object stored in local memory.I hope the explanation was not too long.
Thank you for reading guys.
The text was updated successfully, but these errors were encountered: