Skip to content

Commit d35219e

Browse files
DelevoXDGFroyaTheHenmaciektr
authored
Add scarb verify (#1922)
## Stack - #1924 - #1900 - #1922 ## Issues Closes #1815 --------- Signed-off-by: Maksim Zdobnikau <[email protected]> Co-authored-by: FroyaTheHen <[email protected]> Co-authored-by: maciektr <[email protected]>
1 parent 4e02b14 commit d35219e

File tree

7 files changed

+321
-18
lines changed

7 files changed

+321
-18
lines changed

.github/workflows/ci.yml

+13-13
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ jobs:
2929
- uses: Swatinem/rust-cache@v2
3030
- uses: taiki-e/install-action@nextest
3131
- name: nextest archive
32-
run: cargo nextest archive --workspace --all-features --cargo-profile ci --archive-file 'nextest-archive-${{ matrix.platform.os }}.tar.zst' --exclude scarb-prove
32+
run: cargo nextest archive --workspace --all-features --cargo-profile ci --archive-file 'nextest-archive-${{ matrix.platform.os }}.tar.zst' --exclude scarb-prove --exclude scarb-verify
3333
- uses: actions/upload-artifact@v4
3434
with:
3535
name: nextest-archive-${{ matrix.platform.os }}
@@ -82,8 +82,8 @@ jobs:
8282
- name: run tests
8383
run: cargo test -p scarb-metadata
8484

85-
test-scarb-prove:
86-
name: test scarb-prove ${{ matrix.platform.name }}
85+
test-nightly-crates:
86+
name: test nightly crates ${{ matrix.platform.name }}
8787
runs-on: ${{ matrix.platform.os }}
8888
env:
8989
# TODO(#1915): Use stable toolchain once stwo is stable.
@@ -104,9 +104,9 @@ jobs:
104104
toolchain: ${{ env.RUST_TOOLCHAIN }}
105105
- uses: Swatinem/rust-cache@v2
106106
- name: Build
107-
run: cargo +stable build --workspace --all-features --exclude scarb-prove
108-
- name: Run scarb-prove tests
109-
run: cargo +${{ env.RUST_TOOLCHAIN }} test -p scarb-prove
107+
run: cargo +stable build --workspace --all-features --exclude scarb-prove --exclude scarb-verify
108+
- name: Run scarb-prove and scarb-verify tests
109+
run: cargo +${{ env.RUST_TOOLCHAIN }} test -p scarb-prove -p scarb-verify
110110

111111
test-prebuilt-plugins:
112112
name: test prebuilt plugins ${{ matrix.platform.name }}
@@ -140,17 +140,17 @@ jobs:
140140
- uses: Swatinem/rust-cache@v2
141141
- run: cargo fmt --check
142142
# TODO(#1915): Build all crates with stable toolchain once stwo is stable.
143-
- run: cargo clippy --all-targets --all-features --workspace --exclude scarb-prove -- --no-deps
143+
- run: cargo clippy --all-targets --all-features --workspace --exclude scarb-prove --exclude scarb-verify -- --no-deps
144144
env:
145145
# Make sure CI fails on all warnings, including Clippy lints.
146146
RUSTFLAGS: "-Dwarnings"
147-
- run: cargo doc --all-features --no-deps --workspace --exclude scarb-prove
147+
- run: cargo doc --all-features --no-deps --workspace --exclude scarb-prove --exclude scarb-verify
148148
env:
149149
# Make sure CI fails on all warnings, including Clippy lints.
150150
RUSTDOCFLAGS: "-Dwarnings"
151151

152-
check-rust-scarb-prove:
153-
name: check-rust (scarb-prove)
152+
check-rust-nightly:
153+
name: check-rust (nightly)
154154
runs-on: ubuntu-latest
155155
env:
156156
# TODO(#1915): Use stable toolchain once stwo is stable.
@@ -162,14 +162,14 @@ jobs:
162162
toolchain: ${{ env.RUST_TOOLCHAIN }}
163163
components: rustfmt, clippy
164164
- uses: Swatinem/rust-cache@v2
165-
- run: cargo +${{ env.RUST_TOOLCHAIN }} clippy --all-targets --all-features -p scarb-prove -- --no-deps
165+
- run: cargo +${{ env.RUST_TOOLCHAIN }} clippy --all-targets --all-features -p scarb-prove -p scarb-verify -- --no-deps
166166
env:
167167
# Make sure CI fails on all warnings, including Clippy lints.
168168
RUSTFLAGS: "-Dwarnings"
169-
- run: cargo +${{ env.RUST_TOOLCHAIN }} doc --all-features --no-deps -p scarb-prove
169+
- run: cargo +${{ env.RUST_TOOLCHAIN }} doc --all-features --no-deps -p scarb-prove -p scarb-verify
170170
env:
171171
# Make sure CI fails on all warnings, including Clippy lints.
172-
RUSTDOCFLAGS: "-Dwarnings"
172+
RUSTDOCFLAGS: "-Dwarnings"
173173

174174
check-website:
175175
runs-on: ubuntu-latest

Cargo.lock

+19
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

Cargo.toml

+3
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ members = [
55
"extensions/scarb-doc",
66
"extensions/scarb-execute",
77
"extensions/scarb-prove",
8+
"extensions/scarb-verify",
89
"extensions/scarb-cairo-language-server",
910
"extensions/scarb-cairo-run",
1011
"extensions/scarb-cairo-test",
@@ -76,6 +77,8 @@ cairo-lang-utils = { version = "*", features = ["env_logger"] }
7677
cairo-language-server = "*"
7778
cairo-lint-core = "*"
7879
cairo-vm = "1.0.1"
80+
stwo_cairo_prover = { git = "https://github.com/starkware-libs/stwo-cairo", rev = "a27287c21e8b7e677b07cc40988c8c145533c55d" }
81+
stwo-prover = { git = "https://github.com/starkware-libs/stwo", rev = "af5475cb", features = ["parallel"] }
7982
camino = { version = "1", features = ["serde1"] }
8083
cargo_metadata = ">=0.18"
8184
clap = { version = "4", features = ["derive", "env", "string"] }

extensions/scarb-prove/Cargo.toml

+2-5
Original file line numberDiff line numberDiff line change
@@ -20,15 +20,12 @@ serde.workspace = true
2020
camino.workspace = true
2121
serde_json.workspace = true
2222
create-output-dir = { path = "../../utils/create-output-dir" }
23-
stwo_cairo_prover = { git = "https://github.com/starkware-libs/stwo-cairo", rev = "a27287c21e8b7e677b07cc40988c8c145533c55d" }
24-
stwo-prover = { git = "https://github.com/starkware-libs/stwo", rev = "af5475cb", features = [
25-
"parallel",
26-
] }
23+
stwo_cairo_prover.workspace = true
24+
stwo-prover.workspace = true
2725
scarb-execute = { path = "../scarb-execute" }
2826

2927
[dev-dependencies]
3028
assert_fs.workspace = true
3129
scarb-test-support = { path = "../../utils/scarb-test-support" }
3230
snapbox.workspace = true
3331
predicates.workspace = true
34-

extensions/scarb-verify/Cargo.toml

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
[package]
2+
name = "scarb-verify"
3+
publish = false
4+
5+
authors.workspace = true
6+
edition.workspace = true
7+
homepage.workspace = true
8+
license.workspace = true
9+
readme.workspace = true
10+
repository.workspace = true
11+
version.workspace = true
12+
13+
[dependencies]
14+
anyhow.workspace = true
15+
indoc.workspace = true
16+
scarb-metadata = { path = "../../scarb-metadata" }
17+
scarb-ui = { path = "../../utils/scarb-ui" }
18+
clap.workspace = true
19+
serde.workspace = true
20+
camino.workspace = true
21+
serde_json.workspace = true
22+
stwo_cairo_prover.workspace = true
23+
stwo-prover.workspace = true
24+
25+
[dev-dependencies]
26+
assert_fs.workspace = true
27+
scarb-test-support = { path = "../../utils/scarb-test-support" }
28+
snapbox.workspace = true

extensions/scarb-verify/src/main.rs

+119
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,119 @@
1+
use anyhow::{ensure, Context, Result};
2+
use camino::{Utf8Path, Utf8PathBuf};
3+
use clap::Parser;
4+
use indoc::formatdoc;
5+
use scarb_metadata::{MetadataCommand, PackageMetadata};
6+
use scarb_ui::args::{PackagesFilter, VerbositySpec};
7+
use scarb_ui::components::Status;
8+
use scarb_ui::{OutputFormat, Ui};
9+
use std::env;
10+
use std::fs;
11+
use std::process::ExitCode;
12+
use stwo_cairo_prover::cairo_air::air::CairoProof;
13+
use stwo_cairo_prover::cairo_air::verify_cairo;
14+
use stwo_prover::core::vcs::blake2_merkle::{Blake2sMerkleChannel, Blake2sMerkleHasher};
15+
16+
/// Verifies `scarb prove` output using Stwo verifier.
17+
#[derive(Parser, Clone, Debug)]
18+
#[clap(version, verbatim_doc_comment)]
19+
struct Args {
20+
/// Name of the package.
21+
#[command(flatten)]
22+
packages_filter: PackagesFilter,
23+
24+
/// ID of `scarb execute` output for given package, for which proof was generated using `scarb prove`.
25+
#[arg(long)]
26+
execution_id: Option<u32>,
27+
28+
/// Proof file path.
29+
#[arg(
30+
long,
31+
required_unless_present = "execution_id",
32+
conflicts_with = "execution_id"
33+
)]
34+
proof_file: Option<Utf8PathBuf>,
35+
36+
/// Logging verbosity.
37+
#[command(flatten)]
38+
pub verbose: VerbositySpec,
39+
}
40+
41+
fn main() -> ExitCode {
42+
let args = Args::parse();
43+
let ui = Ui::new(args.verbose.clone().into(), OutputFormat::Text);
44+
45+
match main_inner(args, ui.clone()) {
46+
Ok(()) => ExitCode::SUCCESS,
47+
Err(error) => {
48+
ui.error(format!("{error:#}"));
49+
ExitCode::FAILURE
50+
}
51+
}
52+
}
53+
54+
fn main_inner(args: Args, ui: Ui) -> Result<()> {
55+
let scarb_target_dir = Utf8PathBuf::from(env::var("SCARB_TARGET_DIR")?);
56+
57+
let metadata = MetadataCommand::new().inherit_stderr().exec()?;
58+
let package = args.packages_filter.match_one(&metadata)?;
59+
60+
let proof_path = if let Some(execution_id) = args.execution_id {
61+
ui.print(Status::new("Verifying", &package.name));
62+
resolve_proof_path_from_package(&scarb_target_dir, &package, execution_id)?
63+
} else {
64+
ui.print(Status::new("Verifying", "proof"));
65+
args.proof_file.unwrap()
66+
};
67+
68+
let proof = load_proof(&proof_path)?;
69+
70+
verify_cairo::<Blake2sMerkleChannel>(proof).with_context(|| "failed to verify proof")?;
71+
72+
ui.print(Status::new("Verified", "proof successfully"));
73+
74+
Ok(())
75+
}
76+
77+
fn load_proof(path: &Utf8Path) -> Result<CairoProof<Blake2sMerkleHasher>> {
78+
ensure!(
79+
path.exists(),
80+
format!("proof file does not exist at path: {path}")
81+
);
82+
83+
let proof_contents =
84+
fs::read_to_string(path).with_context(|| format!("failed to read proof file: {path}"))?;
85+
let proof = serde_json::from_str(&proof_contents)
86+
.with_context(|| format!("failed to deserialize proof file: {path}"))?;
87+
Ok(proof)
88+
}
89+
90+
fn resolve_proof_path_from_package(
91+
scarb_target_dir: &Utf8Path,
92+
package: &PackageMetadata,
93+
execution_id: u32,
94+
) -> Result<Utf8PathBuf> {
95+
let execution_dir = scarb_target_dir
96+
.join("execute")
97+
.join(&package.name)
98+
.join(format!("execution{execution_id}"));
99+
100+
ensure!(
101+
execution_dir.exists(),
102+
formatdoc! {r#"
103+
execution directory does not exist at path: {execution_dir}
104+
help: make sure to run `scarb prove --execute` first
105+
and that the execution ID is correct
106+
"#}
107+
);
108+
109+
let proof_path = execution_dir.join("proof").join("proof.json");
110+
ensure!(
111+
proof_path.exists(),
112+
formatdoc! {r#"
113+
proof file does not exist at path: {proof_path}
114+
help: run `scarb prove --execution-id={execution_id}` first
115+
"#}
116+
);
117+
118+
Ok(proof_path)
119+
}

0 commit comments

Comments
 (0)