Skip to content

Commit

Permalink
Merge branch 'main' into notlesh/no-nullptrs-except-deprecated-deploy
Browse files Browse the repository at this point in the history
  • Loading branch information
whichqua authored Dec 6, 2024
2 parents 1453cb3 + dee6508 commit 6299e96
Show file tree
Hide file tree
Showing 31 changed files with 2,499 additions and 104 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/check.yml
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,7 @@ jobs:
pip install cairo-lang==0.13.2 "sympy<1.13.0"
- name: "Prepare test environment"
run: |
bash ./scripts/reset-tests.sh
bash ./setup-scripts/reset-tests.sh
- name: "Run tests"
run: |
RUSTFLAGS="-D warnings" cargo test
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/prove_blocks.yml
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ jobs:
run: |
source venv/bin/activate
pip install cairo-lang==0.13.2 "sympy<1.13.0"
bash scripts/setup-tests.sh
bash setup-scripts/setup-tests.sh
- name: Prove Blocks
env:
Expand Down
2 changes: 2 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,8 @@ build/*
# Cairo dependencies
cairo-lang
venv
snos-env
.python-version
# Hint Dependency symlink
starkware

Expand Down
29 changes: 29 additions & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

3 changes: 2 additions & 1 deletion Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,7 @@ members = [
"crates/bin/hint_tool",
"crates/bin/output_segment",
"crates/bin/prove_block",
"crates/bin/cairo0_run",
"crates/cairo-type-derive",
"crates/rpc-client",
"crates/rpc-replay",
Expand Down Expand Up @@ -36,7 +37,7 @@ cairo-lang-starknet-classes = { version = "=2.8.2" }
cairo-lang-utils = { version = "=2.8.2" }
cairo-lang-casm = { version = "=2.8.2" }
cairo-type-derive = { version = "0.1.0", path = "crates/cairo-type-derive" }
cairo-vm = { git = "https://github.com/Moonsong-Labs/cairo-vm", branch = "herman/fix-pie-serialization", features = ["cairo-1-hints", "extensive_hints", "mod_builtin"] }
cairo-vm = { git = "https://github.com/Moonsong-Labs/cairo-vm", branch = "notlesh/snos-2024-11-04", features = ["cairo-1-hints", "extensive_hints", "mod_builtin"] }
clap = { version = "4.5.4", features = ["derive"] }
c-kzg = { version = "1.0.3" }
env_logger = "0.11.3"
Expand Down
150 changes: 135 additions & 15 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,38 +1,158 @@
<div align="center">
<h1>
SNOS
<br>
<img src="./docs/images/SNOS.png" height="400" width="500">

### ✨ SNOS ✨

A Rust Library for running the [Starknet OS](https://github.com/starkware-libs/cairo-lang/blob/master/src/starkware/starknet/core/os/os.cairo).

![SN_Ver_0.12.2](https://img.shields.io/badge/Starknet-0.12.2-0C0C4C.svg?labelColor=282d33&logo=)
[![Check Workflow Status](https://github.com/keep-starknet-strange/snos/actions/workflows/check.yml/badge.svg)](https://github.com/keep-starknet-strange/snos/actions/workflows/check.yml)
[Report Bug](https://github.com/keep-starknet-strange/snos/issues/new?assignees=&labels=bug&projects=&template=bug_report.md&title=bug%3A+) · [Request Feature](https://github.com/keep-starknet-strange/snos/issues/new?labels=enhancement&title=feat%3A+)

[![Exploration_Team](https://img.shields.io/badge/Exploration_Team-29296E.svg?&style=for-the-badge&logo=)](https://github.com/keep-starknet-strange)
[![Check Workflow Status](https://github.com/keep-starknet-strange/snos/actions/workflows/check.yml/badge.svg)](https://github.com/keep-starknet-strange/snos/actions/workflows/check.yml)
[![license](https://img.shields.io/github/license/keep-starknet-strange/snos)](/LICENSE)
[![pr-welcome]](#-contributing)

[pr-welcome]: https://img.shields.io/static/v1?color=blue&label=PRs&style=flat&message=welcome

</h1>
</div>

Rust Library for running the [Starknet OS](https://hackmd.io/@pragma/ByP-iux1T) via the [Cairo VM](https://github.com/lambdaclass/cairo-vm).
## Table of Contents
- [Table of Contents](#table-of-contents)
- [📖 About](#-about)
- [🛠️ Getting Started](#️-getting-started)
- [Prerequisites](#prerequisites)
- [Installation](#installation)
- [🧪 Running Tests](#-running-tests)
- [Run Tests](#run-tests)
- [Reset Tests](#reset-tests)
- [🚀 Usage](#-usage)
- [Adding SNOS as a Dependency](#adding-snos-as-a-dependency)
- [Using the **prove_block** Binary](#using-the-prove_block-binary)
- [🤝 Related Projects](#-related-projects)
- [📚 Documentation](#-documentation)
- [📜 License](#-license)


## 📖 About

[Starknet OS](https://github.com/starkware-libs/cairo-lang/blob/master/src/starkware/starknet/core/os/os.cairo) is a [Cairo](https://www.cairo-lang.org/) program designed to prove the integrity of state transitions between blocks on Starknet.

By re-executing transactions from a block and verifying consistency, it produces a [PIE](https://github.com/starkware-libs/cairo-lang/blob/a86e92bfde9c171c0856d7b46580c66e004922f3/src/starkware/cairo/lang/vm/cairo_pie.py#L219-L225) (Program Independent Execution) result. This PIE can be used to generate a STARK proof of integrity, which, if accepted by Starknet L1 verifiers, confirms block validity and updates the Starknet state root in the [StarknetCore contract](https://etherscan.io/address/0xc662c410c0ecf747543f5ba90660f6abebd9c8c4#code).

## 🛠️ Getting Started

### Prerequisites

Ensure you have the following dependencies installed:
- [Rust 1.76.0 or newer](https://www.rust-lang.org/tools/install)

#### Optional
- [pyenv](https://github.com/pyenv/pyenv-installer?tab=readme-ov-file#install) (recommended for managing Python versions and setting up environment)

- [rclone](https://rclone.org/install/) (recommended for downloading Pathfinder's database and being able to quickly use [`prove_block`](#using-the-prove_block-binary) binary)

### Installation

## Test Setup
1. **Clone the Repository**

**Cairo Env**
(see: https://docs.cairo-lang.org/0.12.0/quickstart.html)
Clone this repository and its submodules:
```bash
git clone https://github.com/keep-starknet-strange/snos.git --recursive
```

#### Install project dependencies
In order to compile the Starknet OS Cairo program, you’ll need the Cairo compiler:

- Follow the [Cairo documentation](https://docs.cairo-lang.org/quickstart.html)
- Or simply run:
```bash
poetry install
poetry shell
./setup-scripts/setup-cairo.sh
```

**Run Tests**
This will create a virtual environment and download needed dependencies to compile cairo programs. You will need to activate it to compile Cairo programs.

## 🧪 Running Tests
To verify your setup, follow these steps:

### Activate snos-env:
```bash
source ./snos-env/bin/activate
```

### Run Tests

```bash
./scripts/setup-tests.sh

cargo test
```

**Reset Tests**
### Reset Tests

If you need to reset the test environment:

```bash
./scripts/reset-tests.sh
```

## 🚀 Usage
### Adding SNOS as a dependency

You can add the following to your rust project's `Cargo.toml`:

```toml
starknet-os = { git = "https://github.com/keep-starknet-strange/snos", rev = "662d1706f5855044e52ebf688a18dd80016c8700" }
```

### Using the `prove_block` Binary

To execute correctly, SNOS requires detailed block information, including:

- **State changes**: Information about new classes, contracts, and any modifications to contract storage. (See [StateDiff](https://github.com/xJonathanLEI/starknet-rs/blob/5c676a64031901b5a203168fd8ef8d6b40a5862f/starknet-core/src/types/codegen.rs#L1723-L1737))
- **Storage proofs**: [Merkle Proofs](https://www.quicknode.com/docs/starknet/pathfinder_getProof) from both class and contract tries, needed for validating that updated values match the global state root.
- **Transaction execution Information**: Data on [calls, subcalls](https://github.com/starkware-libs/sequencer/blob/7aa546acde88c94825992501662788e716db5fe0/crates/blockifier/src/transaction/objects.rs#L168-L183), and specific program counters visited ([VisitedPCs](https://github.com/starkware-libs/sequencer/blob/7aa546acde88c94825992501662788e716db5fe0/crates/blockifier/src/state/cached_state.rs#L34-L35)) during execution.

The `prove_block` binary handles this entire process by collecting, formatting, and feeding the necessary data into the OS, ensuring the correct `OSInput` is passed for execution.

To accomplish this, it queries the required information from a full node. Currently, Pathfinder is the only full node implementing all the necessary RPC methods, so a synced [Pathfinder](https://github.com/eqlabs/pathfinder) instance running as an [**archive node**](https://github.com/eqlabs/pathfinder?tab=readme-ov-file#state-trie-pruning) (to provide access to storage proofs) is required to execute this binary successfully.

For example, you can run Pathfinder executing:

```bash
PATHFINDER_ETHEREUM_API_URL="YOUR_KEY" ./target/release/pathfinder --data-directory /home/herman/pathfinder-data --http-rpc 0.0.0.0:9545 --storage.state-tries archive
```

Once you have a synced full node, you can start generating PIEs of a given block by running:

```bash
cargo run --release -p prove_block -- --block-number 200000 --rpc-provider http://0.0.0.0:9545
```

## 🤝 Related Projects

- [cairo compiler](https://github.com/starkware-libs/cairo): A blazing fast compiler for Cairo, written in Rust
- [cairo vm](https://github.com/lambdaclass/cairo-vm): A faster and safer implementation of the Cairo VM in Rust
- [blockifier](https://github.com/starkware-libs/sequencer/tree/7218aa1f7ca3fe21c0a2bede2570820939ffe069/crates/blockifier): The transaction-executing component in the Starknet sequencer.
- [pathfinder](https://github.com/eqlabs/pathfinder): A Starknet full node written in Rust
- [madara](https://github.com/madara-alliance/madara): A powerful Starknet client written in Rust.

## 📚 Documentation

### Cairo:
- [The Cairo Book](https://book.cairo-lang.org/)
- [How Cairo Works](https://docs.cairo-lang.org/how_cairo_works/index.html)
- [Cairo – a Turing-complete STARK-friendly CPU architecture](https://eprint.iacr.org/2021/1063)
- [A Verified Algebraic Representation of Cairo Program Execution](https://arxiv.org/pdf/2109.14534)

### Starknet
- [Starknet Docs](https://docs.starknet.io/)
- [Starknet State](https://docs.starknet.io/architecture-and-concepts/network-architecture/starknet-state/)
- [MoonsongLabs talk in StarknetCC](https://www.youtube.com/watch?v=xHc_pKXN9h8)

### StarknetOS
- [Pragma Article on os.cairo](https://hackmd.io/@pragma/ByP-iux1T)
- [os.cairo code](https://github.com/starkware-libs/cairo-lang/blob/master/src/starkware/starknet/core/os/os.cairo)


## 📜 License

This project is licensed under the MIT License. See the [LICENSE](./LICENSE) file for details.
32 changes: 32 additions & 0 deletions crates/bin/cairo0_run/Cargo.toml
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
[package]
name = "cairo0_run"
version.workspace = true
edition.workspace = true
repository.workspace = true
license-file.workspace = true


[dependencies]
anyhow = { workspace = true }
blockifier = { workspace = true }
cairo-lang-starknet-classes = { workspace = true }
cairo-lang-utils = { workspace = true }
cairo-vm = { workspace = true }
clap = { workspace = true }
log = { workspace = true }
env_logger = { workspace = true }
futures-core = { workspace = true }
pathfinder-common = { workspace = true }
pathfinder-crypto = { workspace = true }
pathfinder-serde = { workspace = true }
reqwest = { workspace = true }
serde = { workspace = true }
serde_json = { workspace = true }
serde_with = { workspace = true }
starknet_api = { workspace = true }
starknet-os = { workspace = true }
starknet-os-types = { workspace = true }
starknet-types-core = { workspace = true }
thiserror = { workspace = true }
tokio = { workspace = true }
num-bigint = { workspace = true }
87 changes: 87 additions & 0 deletions crates/bin/cairo0_run/src/main.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
use cairo_vm::cairo_run::CairoRunConfig;
use cairo_vm::types::layout_name::LayoutName;
use cairo_vm::types::program::Program;
use cairo_vm::vm::errors::vm_exception::VmException;
use cairo_vm::vm::runners::cairo_runner::CairoRunner;
use cairo_vm::Felt252;
use clap::Parser;
use starknet_os::error::SnOsError;
use starknet_os::hints;
use starknet_os::starknet::starknet_storage::{CommitmentInfo, CommitmentInfoError, PerContractStorage};
use starknet_os::starkware_utils::commitment_tree::base_types::TreeIndex;

#[derive(Parser, Debug)]
struct Args {
#[arg(long)]
input: String,
}

fn init_logging() {
env_logger::builder()
.filter_level(log::LevelFilter::Trace)
.format_timestamp(None)
.try_init()
.expect("Failed to configure env_logger");
}

fn main() -> Result<(), Box<dyn std::error::Error>> {
init_logging();

let args = Args::parse();

let program_bytes = std::fs::read_to_string(args.input).expect("Failed to read input file");

let layout = LayoutName::all_cairo;

// Init CairoRunConfig
let cairo_run_config = CairoRunConfig { layout, relocate_mem: true, trace_enabled: true, ..Default::default() };
let allow_missing_builtins = cairo_run_config.allow_missing_builtins.unwrap_or(false);

// Load the Starknet OS Program
let program = Program::from_bytes(program_bytes.as_bytes(), Some(cairo_run_config.entrypoint))
.map_err(|e| SnOsError::Runner(e.into()))?;

// Init cairo runner
let mut cairo_runner = CairoRunner::new(
&program,
cairo_run_config.layout,
cairo_run_config.proof_mode,
cairo_run_config.trace_enabled,
)
.map_err(|e| SnOsError::Runner(e.into()))?;

// Init the Cairo VM
let end = cairo_runner.initialize(allow_missing_builtins).map_err(|e| SnOsError::Runner(e.into()))?;

// Run the Cairo VM
let mut hint_processor = hints::SnosHintProcessor::<DummyPCS>::default();
cairo_runner
.run_until_pc(end, &mut hint_processor)
.map_err(|err| VmException::from_vm_error(&cairo_runner, err))
.map_err(|e| SnOsError::Runner(e.into()))?;

// End the Cairo VM run
cairo_runner
.end_run(cairo_run_config.disable_trace_padding, false, &mut hint_processor)
.map_err(|e| SnOsError::Runner(e.into()))?;

if cairo_run_config.proof_mode {
cairo_runner.finalize_segments().map_err(|e| SnOsError::Runner(e.into()))?;
}

Ok(())
}

struct DummyPCS {}

impl PerContractStorage for DummyPCS {
async fn compute_commitment(&mut self) -> Result<CommitmentInfo, CommitmentInfoError> {
unimplemented!();
}
async fn read(&mut self, _key: TreeIndex) -> Option<Felt252> {
unimplemented!();
}
fn write(&mut self, _key: TreeIndex, _value: Felt252) {
unimplemented!();
}
}
Loading

0 comments on commit 6299e96

Please sign in to comment.