Skip to content

zippiehq/cartesi-coprocessor

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Cartesi Coprocessor SDK (Python Tutorial)

Tip

If you need help or want to use another language, visit the Cartesi-Discord and feel free to ask in #cartesi-coprocessor channel

Prerequisites

  • Node.js version 18 or higher

Install Tools

  1. Install nonodo Install nonodo globally via npm:
npm i -g nonodo
  1. Install cartesi machine

Download the Cartesi Machine (0.18.1-rc7) image for your OS from from this link and add the bin/ folder to your PATH:

# Replace CARTESI_MACHINE_DIR with your downloaded directory name
export PATH="$HOME/CARTESI_MACHINE_DIR/bin:$PATH"

# Example for macOS ARM64:
export PATH="$HOME/cartesi-machine-macos-arm64/bin:$PATH"

Verify the installation:

cartesi-machine --version
  1. Set up web3.storage

Install the web3.storage CLI globally:

npm install -g @web3-storage/w3cli
  1. Install foundry

Download and run the Foundry installer script:

curl -L https://foundry.paradigm.xyz | bash

After the installation, initialize Foundry:

foundryup
  1. Build the CARize utility container

Clone the CARize repository and build the Docker image:

git clone https://github.com/nyakiomaina/carize.git
cd carize
docker build -t carize:latest .

Development

  1. Start nonodo
nonodo
  1. Create a python cartesi dApp
cartesi create sample-dapp --template=python --branch "wip/coprocessor"
cd sample-dapp

Edit the dapp.py to include the notice functionality, full code below:

from os import environ
import logging
import requests

logging.basicConfig(level="INFO")
logger = logging.getLogger(__name__)

rollup_server = environ["ROLLUP_HTTP_SERVER_URL"]
logger.info(f"HTTP rollup_server url is {rollup_server}")

def emit_notice(data):
    notice_payload = {"payload": data["payload"]}
    response = requests.post(rollup_server + "/notice", json=notice_payload)
    if response.status_code == 201:
        logger.info(f"Notice emitted successfully with data: {data}")
    else:
        logger.error(f"Failed to emit notice with data: {data}. Status code: {response.status_code}")

def handle_advance(data):
    logger.info(f"Received advance request data {data}")
    emit_notice(data)
    return "accept"


def handle_inspect(data):
    logger.info(f"Received inspect request data {data}")
    emit_notice(data)
    return "accept"


handlers = {
    "advance_state": handle_advance,
    "inspect_state": handle_inspect,
}

finish = {"status": "accept"}

while True:
    logger.info("Sending finish")
    response = requests.post(rollup_server + "/finish", json=finish)
    logger.info(f"Received finish status {response.status_code}")
    if response.status_code == 202:
        logger.info("No pending rollup request, trying again")
    else:
        rollup_request = response.json()
        data = rollup_request["data"]
        handler = handlers[rollup_request["request_type"]]
        finish["status"] = handler(rollup_request["data"])
  1. Build the Cartesi dApp
cartesi build
  1. Run the cartesi machine
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
  1. Run the CARize Utility Container

After building your dApp, run the CARize container to generate necessary files:

docker run --rm \
    -v $(pwd)/.cartesi/image:/data \
    -v $(pwd):/output \
    carize:latest /carize.sh
  1. Set environment variables

Set the variables by reading the values from the output files generated after running the carize.sh script:

CID=$(cat output.cid)
SIZE=$(cat output.size)
MACHINE_HASH=$(xxd -p .cartesi/image/hash | tr -d '\n')

Uploading CAR Files to Web3.Storage

  1. Log In to web3.storage
  1. Create a storage space
w3 space create preferredSpaceName
  1. Upload files to Web3.Storage
w3 up --car output.car

Monitor outputs during development

You can monitor the outputs using nonodo by accessing the GraphQL interface at: http://localhost:8080/graphql

run cartesi send and send a generic input, in my case, sent a string 'hello'

Sample:

cartesi send
✔ Select send sub-command Send generic input to the application.
✔ Chain Foundry
✔ RPC URL http://127.0.0.1:8545
✔ Wallet Mnemonic
✔ Mnemonic test test test test test test test test test test test junk
✔ Account 0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266 9999.968655384479878868 ETH
✔ Application address 0xab7528bb862fB57E8A2BCd567a2e929a0Be56a5e
✔ Input String encoding
✔ Input (as string) hello
✔ Input sent: 0xa65e3f82179fac27ac9656f0ae82a4c5d0de5008c6977bbfb5ead18a01a20804

After sending the input, you can query both the outputs and the inputs given to them using the following GraphQL query:

query notices {
  notices {
    edges {
      node {
        index
        input {
          index
          payload
        }
        payload
      }
    }
  }
}

Execute query and the output should looks like this:

{
  "data": {
    "notices": {
      "edges": [
        {
          "node": {
            "index": 0,
            "input": {
              "index": 0,
              "payload": "0x68656c6c6f"
            },
            "payload": "0x68656c6c6f"
          }
        }
      ]
    }
  }
}

Note: In production, your program will be executed from point of snapshot every single time, there is no state saved, memory or otherwise. This is not the case during development mode.

Ensure Coprocessor has your program

Use the /ensure API with the variables you've set:

curl -X POST "https://cartesi-coprocessor-solver.fly.dev/ensure/$CID/$MACHINE_HASH/$SIZE"

Foundry setup to interact with the Coprocessor

  1. Confirm foundry installation
forge --version

If Foundry isn't installed, refer back to the Install Foundry step.

  1. Clone the foundry template repository

This template repository contains the ICoprocessor.sol interface and a sample contract:

git clone https://github.com/zippiehq/cartesi-coprocessor-contract-template.git -b old 
cd cartesi-coprocessor-contract-template
  1. Get dependencies
forge install
  1. Build the contract
forge build
  1. Deployment
Obtaining RPC Provider and Etherscan API Key

To interact with the Ethereum mainnet, you will need to obtain:

RPC Provider url
  • Visit Alchemy or Infura.
  • Sign up for an account or log in if you already have one.
  • Create a new project targeting the Ethereum mainnet.
  • Obtain the RPC URL provided for your project.
Etherscan API Key

Go to Etherscan to obtain an API key.

  • Register or log in
  • Navigate to API-Keys in your account settings
  • Generate a new API key for use with your deployments

Replace placeholders with your obtained RPC URL, private key, and Etherscan API key

COPROCESSOR_ADDRESS=0xB819BA4c5d2b64d07575ff4B30d3e0Eca219BFd5 MACHINE_HASH=0x<machine_hash_here> forge script script/Deploy.s.sol:DeployScript --rpc-url <rpc-url> --private-key xxx --etherscan-api-key xxx --broadcast --verify

(run this to get the machine hash

 echo "Machine Hash: $MACHINE_HASH"

)

Note: Deploying on the Ethereum mainnet involves real transaction costs (gas fees).

Notice Processing

The handleNotice function in the CoprocessorCaller.sol smart contract is triggered by notices that are sent from the Python application. This function is called as a direct response to the coprocessor being invoked within the smart contract.

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •  

Languages