Skip to content

Latest commit

 

History

History
427 lines (281 loc) · 19.5 KB

README-dev.md

File metadata and controls

427 lines (281 loc) · 19.5 KB

Perf report

The results of the job https://github.com/meta-introspector/o1js/actions/workflows/publish-results.yml which takes a git commit of the source data as a parameter will be written to here as an index https://meta-introspector.github.io/o1js/

you can then pass those in as parameters to the firefox profiler for reading. https://profiler.firefox.com/from-url/https%3A%2F%2Fmeta-introspector.github.io%2Fo1js%2Fmerkle-tree.test.ts.perf.data.script.txt/calltree/?globalTrackOrder=0&hiddenLocalTracksByPid=67-5w8&thread=0&timelineType=category&v=10

o1js README-dev

o1js is a TypeScript framework designed for zk-SNARKs and zkApps on the Mina blockchain.

For more information on our development process and how to contribute, see CONTRIBUTING.md. This document is meant to guide you through building o1js from source and understanding the development workflow.

It is a manual step to build the o1js Reference docs. See o1js Reference in the docs wiki style guide.

Prerequisites

Before starting, ensure you have the following tools installed:

After cloning the repository, you need to fetch the submodules:

export GIT_LFS_SKIP_SMUDGE=1
git submodule update --init --recursive

Building o1js

For most users, building o1js is as simple as running:

npm install
npm run build

This command compiles the TypeScript source files, making them ready for use. The compiled OCaml and WebAssembly artifacts are version-controlled to simplify the build process for end users. These artifacts are stored under src/bindings/compiled and contain the artifacts needed for both node and web builds. These files only have to be regenerated if there are changes to the OCaml or Rust source files.

Building Bindings

To regenerate the OCaml and WebAssembly artifacts, you can do so within the o1js repo. The bindings and Mina repos are both submodules of o1js so you can build them from within the o1js repo.

o1js depends on OCaml code that is transpiled to JavaScript using Js_of_ocaml and Rust code that is transpiled to WebAssembly using wasm-pack. These artifacts allow o1js to call into Pickles, snarky, and Kimchi to write zk-SNARKs and zkApps.

The compiled artifacts are stored under src/bindings/compiled and are version-controlled to simplify the build process for end-users.

If you want to rebuild the OCaml and Rust artifacts, you must be able to build the mina repo before building the bindings. See the Mina Dev Readme for more information. After you have configured your environment to build mina, you can build the bindings:

npm run build:update-bindings

This command builds the OCaml and Rust artifacts and copies them to the src/bindings/compiled directory.

Build Scripts

The root build script which kicks off the build process is under src/bindings/scripts/update-o1js-bindings.sh. This script is responsible for building the Node.js and web artifacts for o1js, and places them under src/bindings/compiled, to be used by o1js.

OCaml Bindings

o1js depends on Pickles, snarky, and parts of the Mina transaction logic, all of which are compiled to JavaScript and stored as artifacts to be used by o1js natively. The OCaml bindings are located under src/bindings. See the OCaml Bindings README for more information.

To compile the OCaml code, a build tool called Dune is used. Dune is a build system for OCaml projects, and is used in addition with Js_of_ocaml to compile the OCaml code to JavaScript. The dune file that is responsible for compiling the OCaml code is located under src/bindings/ocaml/dune. There are two build targets: o1js_node and o1js_web, which compile the Mina dependencies as well as link the wasm artifacts to build the Node.js and web artifacts, respectively. The output file is o1js_node.bc.js, which is used by o1js.

WebAssembly Bindings

o1js additionally depends on Kimchi, which is compiled to WebAssembly. Kimchi is located in the Mina repo under src/mina. See the Kimchi README for more information.

To compile the Wasm code, a combination of Cargo and Dune is used. Both build files are located under src/mina/src/lib/crypto/kimchi, where the wasm folder contains the Rust code that is compiled to Wasm, and the js folder that contains a wrapper around the Wasm code which allows Js_of_ocaml to compile against the Wasm backend.

For the Wasm build, the output files are:

  • plonk_wasm_bg.wasm: The compiled WebAssembly binary.
  • plonk_wasm_bg.wasm.d.ts: TypeScript definition files describing the types of .wasm or .js files.
  • plonk_wasm.js: JavaScript file that wraps the Wasm code for use in Node.js.
  • plonk_wasm.d.ts: TypeScript definition file for plonk_wasm.js.

Generated Constant Types

In addition to building the OCaml and Rust code, the build script also generates TypeScript types for constants used in the Mina protocol. These types are generated from the OCaml source files, and are located under src/bindings/crypto/constants.ts and src/bindings/mina-transaction/gen. When building the bindings, these constants are auto-generated by Dune. If you wish to add a new constant, you can edit the src/bindings/ocaml/o1js_constants file, and then run npm run build:bindings to regenerate the TypeScript files.

o1js uses these types to ensure that the constants used in the protocol are consistent with the OCaml source files.

Development

Branching Policy

o1js base branches Is default?
main Yes
develop No

When you start your work on o1js, please create the feature branch off of one of the above base branches. It's encouraged to submit your work-in-progress as a draft PR to raise visibility! When working with submodules and various interconnected parts of the stack, ensure you are on the correct branches that are compatible with each other.

Default to main as the base branch.

Other base branches (currently develop only) are used in specific scenarios where you want to adapt o1js to changes in the sibling repos on those other branches. Even then, consider whether it is feasible to land your changes to main and merge to develop afterwards. Only changes in main will ever be released, so anything in other branches has to be backported and reconciled with the main branch eventually.

Relationship Between Repositories and Branches

Repository o1js → o1js-bindings → mina
Branches main main compatible
develop develop develop

Where:

  • compatible: This is the Mina repository branch. It corresponds to the main branch in both o1js and o1js-bindings repositories. This branch is where stable releases and soft-fork features are maintained.

  • develop: This branch is maintained across all three repositories. It is used for ongoing (next hard-fork) development, testing new features and integration work.

Running Tests

To ensure your changes don't break existing functionality, run the test suite:

npm run test
npm run test:unit

In order for the mina-signer tests to run you must also build from inside its subdirectory:

cd src/mina-signer
npm run build
cd ../..

This runs all the unit tests and provides you with a summary of the test results.

Note that you can run individual jest tests via the command:

./jest <path/to/test.ts>

You can also run integration tests by running:

npm run test:integration

Finally, a set of end-to-end tests are run against the browser. These tests are not run by default, but you can run them by running:

npm install
npm run e2e:install
npm run build:web

npm run e2e:prepare-server
npm run test:e2e
npm run e2e:show-report

Run the GitHub actions locally

You can execute the CI locally by using act. First, generate a GitHub token and use:

act -j Build-And-Test-Server --matrix test_type:"Simple integration tests" -s $GITHUB_TOKEN

Releasing

To release a new version of o1js, you must first update the version number in package.json. Then, you can create a new pull request to merge your changes into the main branch. After the pull request is merged, a CI job automatically publishes the new version to npm.

Testing and Debugging

Test zkApps against Lightnet network

Use the lightweight Mina blockchain network (Lightnet) to test on a local blockchain before you test with a live network. To test zkApps against the local blockchain, first spin up Lightnet.

The easiest way is to use zkApp CLI sub-commands:

zk lightnet start # start the local network
# Do your tests and other interactions with the network
zk lightnet logs # manage the logs of the local network
zk lightnet explorer # visualize the local network state
zk lightnet stop # stop the local network

Use zk lightnet --help for more information.

You can also use the corresponding Docker image manually:

docker run --rm --pull=missing -it \
  --env NETWORK_TYPE="single-node" \
  --env PROOF_LEVEL="none" \
  --env LOG_LEVEL="Trace" \
  -p 3085:3085 \
  -p 5432:5432 \
  -p 8080:8080 \
  -p 8181:8181 \
  -p 8282:8282 \
  o1labs/mina-local-network:compatible-latest-lightnet

See the Docker Hub repository for more information.

Next up, get the Mina blockchain accounts information to be used in your zkApp. After the local network is up and running, you can use the Lightnet o1js API namespace to get the accounts information. See the corresponding example in src/examples/zkapps/hello-world/run-live.ts.

Profiling o1js

To enhance the development experience and optimize the performance of o1js, use the Chrome Debugger alongside Node.js. This setup is particularly useful when you want to profile the performance of your zkApp or o1js.

Using the run-debug script

To facilitate this process, use the provided script named run-debug. To use this script, run:

./run-debug <path-to-your-zkapp> --bundle

This script initializes a Node.js process with the --inspect-brk flag that starts the Node.js inspector and breaks before the user script starts (i.e., it pauses execution until a debugger is attached). The --enable-source-maps flag ensures that source maps are used to allow easy debugging of o1js code directly.

After the Node.js process is running, open the Chrome browser and navigate to chrome://inspect to attach the Chrome Debugger to the Node.js process. You can set breakpoints, inspect variables, and profile the performance of your zkApp or o1js. For more information on using the Chrome Debugger, see the DevTools documentation.

building

docker build . --progress=plain

docker image history o1labs/mina-local-network:compatible-latest-lightnet > history.txt

docker compose build docker compose run mina-local-network docker compose run -it mina-local-network /bin/bash

to update the pnpm lock pnpm install --no-frozen-lockfile

running test

make will run docker build . that will trigger creating of the image.

Committing test

~/2024/08/02/mina/o1js$ docker commit a4fcb9ec4901 mina-test1

sha256:9b707295a1fa7ac2541b946bc9cc68a90823a10f52b32268848c17dff3799f4a

docker images

| REPOSITORY| TAG| IMAGE| ID| CREATED| SIZE| | mina-test1| latest| 9b707295a1fa| 54| seconds| ago| 12.9GB| | o1js-build| latest| a8e0a0173b63| 4| hours| ago| 2.18GB|

This shows a huge 10 gb of data collected, the newer image is mina1-test1

Create docker hub repo to hold data

Using docker hub create a repo to hold the data.

Tag Image

docker tag mina-test1 h4ckermike/o1js-clinicjs-data:sept-17-2024

Push Image

docker push h4ckermike/o1js-clinicjs-data:sept-17-2024

docker run -v /mnt/data1/:/mnt/data1 -it h4ckermike/o1js-clinicjs-data:sept-17-2024 bash cp -r .clinic /mnt/data1/nix/time/2024/09/17/o1js/_clinic sudo chown -R mdupont: _clinic

update modules

pnpm install "https://github.com/meta-introspector/jest.git" pnpm install "https://github.com/meta-introspector/ts-jest.git" pnpm install "https://github.com/meta-introspector/node-clinic-doctor"

strace -s999 -o strace.txt -f -e execve npx jest ./src/lib/mina/token.test.ts
/home/mdupont/.config/nvm/versions/node/v18.17.0/bin/node /home/mdupont/.npm/_npx/b8d86e6551a4f492/node_modules/.bin/jest ./src/lib/mina/token.test.ts

strace -s999 -o strace.txt -f -e execve npx jest src/lib/provable/test/int.test.ts * /home/mdupont/.config/nvm/versions/node/v18.17.0/bin/node /mnt/data1/nix/time/2024/08/02/mina/o1js/node_modules/.bin/../jest/bin/jest.js src/lib/provable/test/int.test.ts oops the wrong one /home/mdupont/.config/nvm/versions/node/v20.13.1/bin/node /mnt/data1/nix/time/2024/08/02/mina/o1js/node_modules/.bin/../jest/bin/jest.js src/lib/provable/test/int.test.ts

run perf and save the data

This is working

perf record  node --perf-basic-prof ./node_modules/.bin/../jest/bin/jest.js 
perf script > report_perf1.txt

run with custom jest

This is not working

pnpm remove jest-circus
pnpm remove jest
pnpm remove ts-jest
pnpm install "https://github.com/meta-introspector/jest.git"
pnpm install "https://github.com/meta-introspector/ts-jest.git"
pnpm install
pnpm build

perf record  node --perf-basic-prof ./node_modules/.pnpm/@jest+monorepo@https+++codeload.github.com+meta-introspector+jest+tar.gz+4a58402556ecd05ca9cc01873db6fbc5596ccc67/node_modules/@jest/monorepo/packages/jest/bin/jest.js src/lib/provable/test/int.test.ts

test lengths

PASS src/lib/mina/precondition.test.ts (11.013 s) PASS src/lib/mina/token.test.ts (46.054 s)

perf report> report.txt

Architecture

We have docker compose that is runnable locally and in github or gitlab. This allows for reproducible builds. We name our local images the same as would be used in github so we have the same references. We can test our code locally, faster.

Running test

docker compose up reporting-github

this uses the service reporting-github, to run in the container /app/perf-reporting/perf-report.sh that copies perf-reporting/perf-report.sh into a temp new docker image. If you update that you need to run docker compose build reporting-github

To test the run locally, you could use once it works, and been setup:

in the file named ".secrets" put GITHUB_TOKEN=github_pat_abcd with your personal token from github. see nektos/act#233

checkout the git repo not in a submodule because act does not like them.

act --secret-file .secrets -W .github/workflows/process-perf-results.yml --verbose

set-output:: artifacts=[{"id":1950791984,"node_id":"MDg6QXJ0aWZhY3QxOTUwNzkxOTg0","name":"perf.data","size_in_bytes":119031823,"url":"https://api.github.com/repos/meta-introspector/o1js/actions/artifacts/1950791984","archive_download_url":"","expired":false,"created_at":"2024-09-19T00:56:29Z","updated_at":"2024-09-19T00:56:29Z","expires_at":"2024-12-18T00:46:42Z","workflow_run":{"id":10932397620,"repository_id":857372173,"head_repository_id":857372173,"head_branch":"main","head_sha":"584fd53a9f50126360ab764537f4ae10bd5e3666"}}]

That means we can download the data like this gh api https://api.github.com/repos/meta-introspector/o1js/actions/artifacts/1950791984/zip > data/perf.data.zip

Now we added a way to edit a script in the data directory. o1js/perf-reporting/scripts/perf-report.sh if we put the script in there it will be run when we run docker compose up reporting-github; This allows for iterative changes.

you can also say docker compose up --build reporting-github; This allows for iterative changes.

perf report --header-only > header.txt perf report --verbose --stats

Example

Aggregated stats:
           TOTAL events:    5528837
            MMAP events:         61  ( 0.0%)
            COMM events:          8  ( 0.0%)
            EXIT events:        134  ( 0.0%)
            FORK events:        133  ( 0.0%)
          SAMPLE events:    5506199  (99.6%)
           MMAP2 events:       3148  ( 0.1%)
         KSYMBOL events:         25  ( 0.0%)
       BPF_EVENT events:         24  ( 0.0%)
: 
  FINISHED_ROUND events:      19100  ( 0.3%)
        ID_INDEX events:          1  ( 0.0%)
      THREAD_MAP events:          1  ( 0.0%)
         CPU_MAP events:          1  ( 0.0%)
       TIME_CONV events:          1  ( 0.0%)
   FINISHED_INIT events:          1  ( 0.0%)
cpu-clock:ppp stats:
          SAMPLE events:    5506199

Summary

This set of tools is for batch collecting and processing performance and coverage data customized for o1js mina.

We have different github actions to create docker images.

The long term goal is to be able to collect this performance information in the browser later, but for new we can run them locally, in github actions or on other platforms, planning on gitlab and any other cloud resources that are available. We will try other build platform first via docker images and then later on actual different hardware.

To run a local test via docker just type in make to use the Makefile to run tests via docker.

Creating docker images

Build and publish

Using docker images to run jobs

Then we take the resulting url of the docker image produces and pass it into the job defined here "run Perf Test and collect data" .github/workflows/run-docker-tests.yml to use the docker image

Useful Commands to download archives locally

We can download an artifact from github using gh ai to basically authenticate a curl for us.

This is an example of downloading an artifact directly, you can find examples in the github build logs for the jobs gh api https://api.github.com/repos/meta-introspector/o1js/actions/artifacts/1961929543/zip > data/perf.data.zip

In This version it contains perf.data.tar.gz, we have evolved the artifact contents so it will change to slightly different names in future versions. This documentation needs to be edited again to reflect that.

next step

See run-all-tests.sh for how we record and report on tests using clinic and perf and other tools as we add them.