Developing with Cartesi and Avail: A Step-by-Step Guide
This guide will walk you through setting up a Cartesi dApp using Avail on your local machine. You will learn how to send transactions either directly (through Cartesi Rollups Smart Contracts deployed on an L1) or through Avail DA using EIP-712 signed messages. Also how to inspect the dApp state and outputs through the APIs provided by the Cartesi Rollups Framework.
- Nonodo: A node responsible for communication with Avail and Cartesi.
- Cartesi Machine: Backend logic engine for the dApp.
You can install nonodo in multiple ways. Either through an NPM Package or building from source
- Install brunodo, the experimental version of nonodo through npm using the command:
npm i -g brunodo
- Download the Cartesi machine for your OS from this link.
For Linux and macOS:
- Extract the tar.xz file:
tar -xf <filename>.tar.xz
Replace \<filename\>
with the actual name of the file you downloaded.
- Navigate to the extracted directory, rename the extracted folder to
cartesi-machine
:
cd <cartesi-machine>
- Set up environment variables for the Cartesi Machine. You'll need to add the
bin
directory to your system’s PATH so that you can run the Cartesi Machine binaries from anywhere. For Linux or macOS, you can do this by adding the following line to your\~/.bashrc
,\~/.bash_profile
, or\~/.zshrc
file, depending on your shell:
export PATH=$PATH:/path/to/cartesi-machine/bin
Replace /path/to/cartesi-machine/
with the actual path to the bin
folder inside the extracted directory, you can get this by running the command in your terminal while inside the cartesi machine folder: pwd
. This should print out the path to location of the cartesi-machine folder.
- After adding the line, refresh your terminal configuration by running:
source ~/.bashrc
Or, if you're using zsh:
source ~/.zshrc
- Verify the installation by checking if the Cartesi Machine binary is available. You can do this by running:
cartesi-machine --help
This should display the available options for the Cartesi Machine, indicating that it’s correctly set up.
:::warning
For Mac users, running this command might trigger a prompt from Mac built in security feauture. Check this part of the troubleshooting section for more guidelines on how to resolve this.
:::
You can Install the Cartesi Cli using the command;
npm i -g @cartesi/cli
To test and develop your app locally it is higly recommended that you use Brunodo
since it contains all the experiemental features from Nonodo
such as simulating Avail inputs.
With it you can skip a lot of the setup and send EIP-712 messages directly to the node using the nonce and submit endpoints that will be running on localhost:8080/nonce
and localhost:8080/submit
- Start brunodo using the command;
brunodo
- In another terminal, create and build a new Cartesi dApp using the following command:
- Python
cartesi create my-dapp --template python
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 \
--volume=.:/mnt --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 --workdir=/mnt -- python dapp.py
- Rust
cartesi create my-dapp --template rust
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 -- /opt/cartesi/dapp/dapp
- Golang
cartesi create my-dapp --template go
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 -- /opt/cartesi/dapp/dapp
- Javascript
cartesi create my-dapp --template javascript
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 \
--volume=.:/mnt --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 --workdir=/opt/cartesi/dapp -- node index
:::warning
This version of the integration with Avail supports only Sepolia (chainId: 11155111) as the testnet.
:::
- Register your dApp Address
To run on the testnet environment you will need a dApp address on the network. For this we prepared a web page where you can resgister an address for your dApp https://address.mugen.builders In the above link you can connect with your wallet and using you public key generate a dApp address that will be used in the command that follows.
- Start brunodo using the command with the flag with the flag that enables integration with Avail;
brunodo --avail-enabled -d --contracts-input-box-block 6850934 --rpc-url https://sepolia.drpc.org --epoch-blocks 0 --contracts-application-address <dapp-address>
- In another terminal, create and build a new Cartesi dApp using the following command:
- Python
cartesi create my-dapp --template python
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 \
--volume=.:/mnt --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 --workdir=/mnt -- python dapp.py
- Rust
cartesi create my-dapp --template rust
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 -- /opt/cartesi/dapp/dapp
- Golang
cartesi create my-dapp --template go
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 -- /opt/cartesi/dapp/dapp
- Javascript
cartesi create my-dapp --template javascript
cd my-dapp
cartesi build
- Run the Cartesi Machine Locally on bare metal using the command;
cartesi-machine --network --flash-drive=label:root,filename:.cartesi/image.ext2 \
--volume=.:/mnt --env=ROLLUP_HTTP_SERVER_URL=http://10.0.2.2:5004 --workdir=/opt/cartesi/dapp -- node index
-
Sending transactions such as deposits or generic messages through the layer 1 is done in the same ways as Cartesi Rollups standalone. You can use
cast
, thecartesi cli
or other approaches. You can follow them here in the docs -
To send a dummy anvil execution that should go through avail and the piao sequencer before being picked up by your dapp running in the Cartesi Machine you can run the following command:
- Install The mugen-builders cli using npm:
npm install -g @mugen-builders/cli
- Run the below command to start the process of sending the transaction:
mugen-cli send
Important
Check out this Video for a demo on using the cli
We have a demo example available which you can clone, and integrate into the dapp running on your local machine very easily. You can choose to modify this dApp to fit and match your ideal implementation and design. It contains a ways to send many different types of input. Including interacting with your Cartesi dApp via avail, which utilises EIP-712 to sign typed data which is relayed on the users behalf to the avail testnet.
- Clone the frontend repo integrated with EIP 712 by using this command:
git clone https://github.com/Calindra/frontend-web-cartesi
- Install all the necessary dependencies by running this commands:
cd frontend-web-cartesi
git checkout feature/refactor-simplification
yarn install
- Generate the necessary rollup data’s by running this command;
yarn codegen
- Start the frontend application by running:
yarn dev --port 3000
- Finally open your browser and navigate to the URL where your frontend dapp is running, you can now interact with your dapp running on local by signing and sending data to your dapp via Avail.
- To send data via avail use the “Send L2 EIP-712 Input” form in the Input section.
- If you are running with the testnet remember to point to Sepolia
Interacting with your Cartesi dApp using the @mugen-builders/client
npm package allows you to send data via EIP-712 or directly using signed inputs. This package simplifies the process of relaying data to Cartesi dApps, providing flexibility to work with both EIP-712 formatted data and standard inputs.
You can check the description of the function in the package's page
- Install the npm package by running the following command:
npm install @mugen-builders/[email protected]
To integrate the package into the front-end of your dApp, use the advanceEIP712
and advanceInput
methods to handle both EIP-712 typed data and simple input data(which goes through the L1). Below is an example implementation:
import { advanceInput, advanceEIP712 } from "@mugen-builders/client";
const addInput = async (_input) => {
const provider = new ethers.providers.Web3Provider(wallet.provider);
const signer = provider.getSigner();
// For EIP-712 input
let availInput = await advanceEIP712(signer, provider, dappAddress, _input, {
cartesiNodeUrl: "http://localhost:8080",
});
// For simple input
let l1Input = await advanceInput(signer, dappAddress, _input, {
inputBoxAddress: "0x58Df21fE097d4bE5dCf61e01d9ea3f6B81c2E1dB",
});
};
The return of advanceEIP712
will be the same as advanceInput
. Both methods will return lists with reports, notices and vouchers generated from that input, allowing you to interact with your dApp using the provided data.
This simplifies interaction with your dApp, providing an easy way to handle both types of inputs.
Inspecting the state of your dApp though handle_inspect
function is done in the same way as using Cartesi Rollups standalone. You can refer to the docs
Querying outputs directly is the exact same as using Cartesi Rollups standalone. You can refer to the docs
To query outputs from a specific the process is very similar to using Cartesi Rollups Standalone. You can refer to the docs to read more.
The big difference is the output format. Instead of querying inputs through the index
field, you query them through an id
field.
This id field can come in two ways:
- It is a hex value returned from
/submit
endpoint when the input comes from and EIP-712 signed message - It is string containing a scalar integer value that can be found inside the events emitted by the
inputBox
contract when sending the transaction through the layer 1.
query {
inputs(first: 30) {
edges {
node {
id
index
status
blockTimestamp
msgSender
payload
}
}
}
}
query {
input(id: "<input-id>") {
id
index
status
blockTimestamp
msgSender
payload
notices {
edges {
node {
payload
}
}
}
reports {
edges {
node {
payload
}
}
}
vouchers {
edges {
node {
payload
}
}
}
}
}