Integrity verifier is deployed on Starknet and can be used for verifying proofs onchain. The intended way of using the verifier is through FactRegistry contract, which besides running the verification process, also stores data for all verified proofs. (For more information see FactRegistry and Proxy contract)
There are two ways of serializing proof into calldata: monolith and split proof. The former should be used if possible, because it's easier and more efficient. The latter should only be used if monolith proof did not fit in a single transaction, either because of calldata limit or steps limit.
Calldata for monolith proof can be generated with the following command:
cargo run --release --bin proof_serializer < examples/proofs/recursive/cairo0_example_proof.json > examples/calldata
Then make sure that you have sncast
installed and snfoundry.toml
is configured correctly.
After that, you can use verify-on-starknet.sh
script to send the transaction to FactRegistry contract. Remember to select appropriate settings for your proof. For more information on supported settings, see Configure Verifier.
For example, run:
./verify-on-starknet.sh 0x7a5340bf1a500d94185cde6fc9cdc4b32c1159d1db5c056841d21bfb0d9c2bd examples/calldata recursive keccak_248_lsb stone5 cairo0
This bash script internally calls verify_proof_full_and_register_fact
function on FactRegistry contract.
To generate split calldata, please refer to Calldata Generator README. This repository also provides script for automatic transaction sending (proof verification is split into multiple transactions, for more information see Split Verifier Architecture).
To run the verifier locally, first you need to build cairo project using:
scarb build
The verifier by default is configured in recursive layout and keccak hasher. If you want to build for other layouts, refer to Configure Verifier
Because of large size of proofs, we don't store them directly in this repository, but rather in Large File Storage, so you need to have it installed and then run git lfs pull
.
Then you can use cairo runner to run the verifier on example proof:
cargo run --release --bin runner -- \
--program target/dev/cairo_verifier.sierra.json \
--cairo-version cairo0 \
--stone-version stone5 \
--hasher-bit-length 160_lsb \
< examples/proofs/recursive/cairo0_stone5_keccak_160_lsb_example_proof.json
By default, the verifier is configured for monolith version, recursive layout and keccak hash for verifier unfriendly commitment layers. You can easily change that by using scarb's features:
scarb build --no-default-features --features small,blake2s,monolith
layout
: [dex
, recursive
, recursive_with_poseidon
, small
, starknet
, starknet_with_keccak
]
hash functions: [keccak
, blake2s
]
verifier types: [monolith
, split
]
There are also additional settings that can be configured at runtime:
cairo_version
: [cairo0
, cairo1
]
stone_version
: [stone5
, stone6
]
hasher bit length: [160_lsb
, 248_lsb
]
Hash function and hasher bit length are combined into one setting:
hasher
: [keccak_160_lsb
, blake2s_160
, blake2s_248_lsb
]
For stone5
available hasher
s are keccak_160_lsb
and blake2s_160
, for stone6
- keccak_160_lsb
and blake2s_248_lsb
.
To run tests, use the following command:
scarb test
In order to launch benchmarking, just run this (it requires recursive layout configuration):
cargo run --release --bin benches -- target/dev/cairo_verifier.sierra.json
For detailed instructions and examples, refer to the Stone Prover documentation.
How to prove Cairo0 program with Stone Prover.
How to prove Cairo1 program with Stone Prover.
Because of great complexity of the verifier compared to standard starknet contracts, we encounter some limitations enforced by starknet. The most important ones are:
- Contract classhash size limit
- Transaction calldata limit
- Transaction steps limit
To overcome these limitations, we split the verifier into multiple contracts and transactions. The biggest part of classhash size is autogenerated (e.g. recursive autogenerated), so we extracted that part into separate contract (or many contracts in case of starknet
and starknet_with_keccak
layout), which is called automatically by the main verifier contract. On the other hand the biggest part of calldata is fri witness, so user can send subsequent chunks of fri witness in separate step transactions.
Since verifier can be configured in many ways and some parts of the logic changes with new stone versions, a contract which routes calls to the correct verifier is needed. This task is handled by FactRegistry contract that also stores data for all verified proofs.
After proof is verified, FactRegistered
event is emitted which contains fact_hash
, verification_hash
, security_bits
and settings
. fact_hash
is a value that represents proven program and its output (formally fact_hash = poseidon_hash(program_hash, output_hash)
). Remember that registration of some fact_hash
doesn't necessary mean that it has been verified by someone with secure enough proof. You always need to check security_bits
and settings
which is part of verification_hash
(formally verification_hash = poseidon_hash(fact_hash, security_bits, settings)
).
FactRegistry
provides two methods for checking verified proofs:
get_verification(verification_hash)
- returns fact hash, security bits and settings for givenverification_hash
.get_all_verifications_for_fact_hash(fact_hash)
- returns list of all verification hashes, security bits and settings for givenfact_hash
. This method is useful for checking if given program has been verified by someone with secure enough proof.
FactRegistry contract is trustless which means that the owner of the contract can't override or change any existing behavior, they can only add new verifiers. Proxy contract on the other hand is upgradable, so every function can be changed or removed. It has the advantage of having all future updates of the verifier logic without having to replace the address of FactRegistry contract.
TODO: how to read FactRegistered event