From 4836cf1a9145e9c9217721359dd35e079c0ac921 Mon Sep 17 00:00:00 2001 From: AvivYossef-starkware Date: Wed, 27 Nov 2024 15:36:59 +0200 Subject: [PATCH] feat(starknet_api): add sierra version to class info --- Cargo.lock | 1 + .../blockifier/src/transaction/test_utils.rs | 10 +-- .../state_reader/reexecution_state_reader.rs | 19 ++++- .../native_blockifier/src/py_transaction.rs | 2 + crates/papyrus_execution/src/lib.rs | 85 ++++++++++++++----- crates/papyrus_execution/src/test_utils.rs | 2 + crates/papyrus_rpc/src/v0_8/api/mod.rs | 18 +++- crates/starknet_api/Cargo.toml | 1 + crates/starknet_api/src/contract_class.rs | 81 +++++++++++++++++- crates/starknet_api/src/lib.rs | 2 + crates/starknet_api/src/transaction_test.rs | 2 +- crates/starknet_gateway/src/compilation.rs | 7 +- 12 files changed, 195 insertions(+), 35 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 3fddafbc6b..35c86c2ee5 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -10227,6 +10227,7 @@ dependencies = [ "pretty_assertions", "primitive-types", "rstest", + "semver 1.0.23", "serde", "serde_json", "sha3", diff --git a/crates/blockifier/src/transaction/test_utils.rs b/crates/blockifier/src/transaction/test_utils.rs index fdefb138e6..88c9ce4028 100644 --- a/crates/blockifier/src/transaction/test_utils.rs +++ b/crates/blockifier/src/transaction/test_utils.rs @@ -1,7 +1,7 @@ use rstest::fixture; use starknet_api::abi::abi_utils::get_fee_token_var_address; use starknet_api::block::GasPrice; -use starknet_api::contract_class::{ClassInfo, ContractClass}; +use starknet_api::contract_class::{ClassInfo, ContractClass, SierraVersion}; use starknet_api::core::{ClassHash, ContractAddress, Nonce}; use starknet_api::execution_resources::GasAmount; use starknet_api::test_utils::deploy_account::DeployAccountTxArgs; @@ -348,11 +348,11 @@ pub fn create_all_resource_bounds( } pub fn calculate_class_info_for_testing(contract_class: ContractClass) -> ClassInfo { - let sierra_program_length = match contract_class { - ContractClass::V0(_) => 0, - ContractClass::V1(_) => 100, + let (sierra_program_length, sierra_version) = match contract_class { + ContractClass::V0(_) => (0, SierraVersion::zero()), + ContractClass::V1(_) => (100, SierraVersion::new(2, 8, 4)), }; - ClassInfo::new(&contract_class, sierra_program_length, 100).unwrap() + ClassInfo::new(&contract_class, sierra_program_length, 100, sierra_version).unwrap() } pub fn emit_n_events_tx( diff --git a/crates/blockifier_reexecution/src/state_reader/reexecution_state_reader.rs b/crates/blockifier_reexecution/src/state_reader/reexecution_state_reader.rs index c500344c84..2b18895449 100644 --- a/crates/blockifier_reexecution/src/state_reader/reexecution_state_reader.rs +++ b/crates/blockifier_reexecution/src/state_reader/reexecution_state_reader.rs @@ -3,7 +3,7 @@ use blockifier::test_utils::MAX_FEE; use blockifier::transaction::transaction_execution::Transaction as BlockifierTransaction; use papyrus_execution::DEPRECATED_CONTRACT_SIERRA_SIZE; use starknet_api::block::{BlockHash, BlockNumber}; -use starknet_api::contract_class::ClassInfo; +use starknet_api::contract_class::{ClassInfo, SierraVersion}; use starknet_api::core::ClassHash; use starknet_api::transaction::{Transaction, TransactionHash}; use starknet_core::types::ContractClass as StarknetContractClass; @@ -20,7 +20,21 @@ pub trait ReexecutionStateReader { StarknetContractClass::Sierra(sierra) => { let abi_length = sierra.abi.len(); let sierra_length = sierra.sierra_program.len(); - Ok(ClassInfo::new(&sierra_to_contact_class_v1(sierra)?, sierra_length, abi_length)?) + + // Extract the version from the sierra program. + let (minor, major, path): (u64, u64, u64) = ( + sierra.sierra_program[0].try_into().unwrap(), + sierra.sierra_program[1].try_into().unwrap(), + sierra.sierra_program[2].try_into().unwrap(), + ); + let sierra_version: SierraVersion = SierraVersion::new(major, minor, path); + + Ok(ClassInfo::new( + &sierra_to_contact_class_v1(sierra)?, + sierra_length, + abi_length, + sierra_version, + )?) } StarknetContractClass::Legacy(legacy) => { let abi_length = @@ -29,6 +43,7 @@ pub trait ReexecutionStateReader { &legacy_to_contract_class_v0(legacy)?, DEPRECATED_CONTRACT_SIERRA_SIZE, abi_length, + SierraVersion::zero(), )?) } } diff --git a/crates/native_blockifier/src/py_transaction.rs b/crates/native_blockifier/src/py_transaction.rs index bdf28426b0..0f4036cacd 100644 --- a/crates/native_blockifier/src/py_transaction.rs +++ b/crates/native_blockifier/src/py_transaction.rs @@ -164,6 +164,7 @@ pub struct PyClassInfo { raw_contract_class: String, sierra_program_length: usize, abi_length: usize, + sierra_version: String, } impl PyClassInfo { @@ -185,6 +186,7 @@ impl PyClassInfo { &contract_class, py_class_info.sierra_program_length, py_class_info.abi_length, + py_class_info.sierra_version.parse()?, )?; Ok(class_info) } diff --git a/crates/papyrus_execution/src/lib.rs b/crates/papyrus_execution/src/lib.rs index 4e7fada6bd..138c6ce5bc 100644 --- a/crates/papyrus_execution/src/lib.rs +++ b/crates/papyrus_execution/src/lib.rs @@ -12,7 +12,6 @@ mod execution_test; pub mod execution_utils; mod state_reader; - #[cfg(test)] mod test_utils; #[cfg(any(feature = "testing", test))] @@ -52,7 +51,7 @@ use papyrus_storage::header::HeaderStorageReader; use papyrus_storage::{StorageError, StorageReader}; use serde::{Deserialize, Serialize}; use starknet_api::block::{BlockHashAndNumber, BlockNumber, NonzeroGasPrice, StarknetVersion}; -use starknet_api::contract_class::{ClassInfo, EntryPointType}; +use starknet_api::contract_class::{ClassInfo, EntryPointType, SierraVersion}; use starknet_api::core::{ChainId, ClassHash, ContractAddress, EntryPointSelector}; use starknet_api::data_availability::L1DataAvailabilityMode; use starknet_api::deprecated_contract_class::ContractClass as DeprecatedContractClass; @@ -422,8 +421,22 @@ pub enum ExecutableTransactionInput { // todo(yair): Do we need to support V0? DeclareV0(DeclareTransactionV0V1, DeprecatedContractClass, AbiSize, OnlyQuery), DeclareV1(DeclareTransactionV0V1, DeprecatedContractClass, AbiSize, OnlyQuery), - DeclareV2(DeclareTransactionV2, CasmContractClass, SierraSize, AbiSize, OnlyQuery), - DeclareV3(DeclareTransactionV3, CasmContractClass, SierraSize, AbiSize, OnlyQuery), + DeclareV2( + DeclareTransactionV2, + CasmContractClass, + SierraSize, + AbiSize, + OnlyQuery, + SierraVersion, + ), + DeclareV3( + DeclareTransactionV3, + CasmContractClass, + SierraSize, + AbiSize, + OnlyQuery, + SierraVersion, + ), DeployAccount(DeployAccountTransaction, OnlyQuery), L1Handler(L1HandlerTransaction, Fee, OnlyQuery), } @@ -479,13 +492,24 @@ impl ExecutableTransactionInput { sierra_program_length, abi_length, only_query, + sierra_version, ) => { let as_transaction = Transaction::Declare(DeclareTransaction::V2(tx)); let res = func(&as_transaction, only_query); let Transaction::Declare(DeclareTransaction::V2(tx)) = as_transaction else { unreachable!("Should be declare v2 transaction.") }; - (Self::DeclareV2(tx, class, sierra_program_length, abi_length, only_query), res) + ( + Self::DeclareV2( + tx, + class, + sierra_program_length, + abi_length, + only_query, + sierra_version, + ), + res, + ) } ExecutableTransactionInput::DeclareV3( tx, @@ -493,13 +517,24 @@ impl ExecutableTransactionInput { sierra_program_length, abi_length, only_query, + sierra_version, ) => { let as_transaction = Transaction::Declare(DeclareTransaction::V3(tx)); let res = func(&as_transaction, only_query); let Transaction::Declare(DeclareTransaction::V3(tx)) = as_transaction else { unreachable!("Should be declare v3 transaction.") }; - (Self::DeclareV3(tx, class, sierra_program_length, abi_length, only_query), res) + ( + Self::DeclareV3( + tx, + class, + sierra_program_length, + abi_length, + only_query, + sierra_version, + ), + res, + ) } ExecutableTransactionInput::DeployAccount(tx, only_query) => { let as_transaction = Transaction::DeployAccount(tx); @@ -789,6 +824,7 @@ fn to_blockifier_tx( &deprecated_class.into(), DEPRECATED_CONTRACT_SIERRA_SIZE, abi_length, + SierraVersion::zero(), ) .map_err(|err| ExecutionError::BadDeclareTransaction { tx: DeclareTransaction::V0(declare_tx.clone()), @@ -815,6 +851,7 @@ fn to_blockifier_tx( &deprecated_class.into(), DEPRECATED_CONTRACT_SIERRA_SIZE, abi_length, + SierraVersion::zero(), ) .map_err(|err| ExecutionError::BadDeclareTransaction { tx: DeclareTransaction::V1(declare_tx.clone()), @@ -836,14 +873,18 @@ fn to_blockifier_tx( sierra_program_length, abi_length, only_query, + sierra_version, ) => { - let class_info = - ClassInfo::new(&compiled_class.into(), sierra_program_length, abi_length).map_err( - |err| ExecutionError::BadDeclareTransaction { - tx: DeclareTransaction::V2(declare_tx.clone()), - err, - }, - )?; + let class_info = ClassInfo::new( + &compiled_class.into(), + sierra_program_length, + abi_length, + sierra_version, + ) + .map_err(|err| ExecutionError::BadDeclareTransaction { + tx: DeclareTransaction::V2(declare_tx.clone()), + err, + })?; BlockifierTransaction::from_api( Transaction::Declare(DeclareTransaction::V2(declare_tx)), tx_hash, @@ -860,14 +901,18 @@ fn to_blockifier_tx( sierra_program_length, abi_length, only_query, + sierra_version, ) => { - let class_info = - ClassInfo::new(&compiled_class.into(), sierra_program_length, abi_length).map_err( - |err| ExecutionError::BadDeclareTransaction { - tx: DeclareTransaction::V3(declare_tx.clone()), - err, - }, - )?; + let class_info = ClassInfo::new( + &compiled_class.into(), + sierra_program_length, + abi_length, + sierra_version, + ) + .map_err(|err| ExecutionError::BadDeclareTransaction { + tx: DeclareTransaction::V3(declare_tx.clone()), + err, + })?; BlockifierTransaction::from_api( Transaction::Declare(DeclareTransaction::V3(declare_tx)), tx_hash, diff --git a/crates/papyrus_execution/src/test_utils.rs b/crates/papyrus_execution/src/test_utils.rs index 861fc4b231..563a144769 100644 --- a/crates/papyrus_execution/src/test_utils.rs +++ b/crates/papyrus_execution/src/test_utils.rs @@ -21,6 +21,7 @@ use starknet_api::block::{ GasPrice, GasPricePerToken, }; +use starknet_api::contract_class::SierraVersion; use starknet_api::core::{ ChainId, ClassHash, @@ -312,6 +313,7 @@ impl TxsScenarioBuilder { DUMMY_SIERRA_SIZE, 0, false, + SierraVersion::default(), ); self.txs.push(tx); self diff --git a/crates/papyrus_rpc/src/v0_8/api/mod.rs b/crates/papyrus_rpc/src/v0_8/api/mod.rs index 1437933c93..23b1bc379e 100644 --- a/crates/papyrus_rpc/src/v0_8/api/mod.rs +++ b/crates/papyrus_rpc/src/v0_8/api/mod.rs @@ -18,6 +18,7 @@ use papyrus_storage::state::StateStorageReader; use papyrus_storage::StorageTxn; use serde::{Deserialize, Serialize}; use starknet_api::block::{BlockHashAndNumber, BlockNumber}; +use starknet_api::contract_class::SierraVersion; use starknet_api::core::{ClassHash, ContractAddress, Nonce}; use starknet_api::deprecated_contract_class::{ ContractClass as StarknetApiDeprecatedContractClass, @@ -379,7 +380,7 @@ pub(crate) fn stored_txn_to_executable_txn( value.class_hash )) })?; - let (sierra_program_length, abi_length) = + let (sierra_program_length, abi_length, sierra_version) = get_class_lengths(storage_txn, state_number, value.class_hash)?; Ok(ExecutableTransactionInput::DeclareV2( value, @@ -387,6 +388,7 @@ pub(crate) fn stored_txn_to_executable_txn( sierra_program_length, abi_length, false, + sierra_version, )) } starknet_api::transaction::Transaction::Declare( @@ -401,7 +403,7 @@ pub(crate) fn stored_txn_to_executable_txn( value.class_hash )) })?; - let (sierra_program_length, abi_length) = + let (sierra_program_length, abi_length, sierra_version) = get_class_lengths(storage_txn, state_number, value.class_hash)?; Ok(ExecutableTransactionInput::DeclareV3( value, @@ -409,6 +411,7 @@ pub(crate) fn stored_txn_to_executable_txn( sierra_program_length, abi_length, false, + sierra_version, )) } starknet_api::transaction::Transaction::Deploy(_) => { @@ -453,9 +456,10 @@ fn get_class_lengths( storage_txn: &StorageTxn<'_, RO>, state_number: StateNumber, class_hash: ClassHash, -) -> Result<(SierraSize, AbiSize), ErrorObjectOwned> { +) -> Result<(SierraSize, AbiSize, SierraVersion), ErrorObjectOwned> { let state_number_after_block = StateNumber::unchecked_right_after_block(state_number.block_after()); + storage_txn .get_state_reader() .map_err(internal_server_error)? @@ -464,7 +468,13 @@ fn get_class_lengths( .ok_or_else(|| { internal_server_error(format!("Missing deprecated class definition of {class_hash}.")) }) - .map(|contract_class| (contract_class.sierra_program.len(), contract_class.abi.len())) + .and_then(|contract_class| { + let sierra_program_len = contract_class.sierra_program.len(); + let abi_len = contract_class.abi.len(); + let sierra_program = + (&contract_class.sierra_program).try_into().map_err(internal_server_error)?; + Ok((sierra_program_len, abi_len, sierra_program)) + }) } impl TryFrom for ExecutableTransactionInput { diff --git a/crates/starknet_api/Cargo.toml b/crates/starknet_api/Cargo.toml index 5fac49f236..4fd905b38c 100644 --- a/crates/starknet_api/Cargo.toml +++ b/crates/starknet_api/Cargo.toml @@ -21,6 +21,7 @@ itertools.workspace = true num-bigint.workspace = true pretty_assertions.workspace = true primitive-types = { workspace = true, features = ["serde"] } +semver.workspace = true serde = { workspace = true, features = ["derive", "rc"] } serde_json.workspace = true sha3.workspace = true diff --git a/crates/starknet_api/src/contract_class.rs b/crates/starknet_api/src/contract_class.rs index 3a85573a06..df713cee94 100644 --- a/crates/starknet_api/src/contract_class.rs +++ b/crates/starknet_api/src/contract_class.rs @@ -1,5 +1,10 @@ +use std::str::FromStr; + use cairo_lang_starknet_classes::casm_contract_class::CasmContractClass; +use derive_more::Deref; +use semver::Version; use serde::{Deserialize, Serialize}; +use starknet_types_core::felt::Felt; use crate::core::CompiledClassHash; use crate::deprecated_contract_class::ContractClass as DeprecatedContractClass; @@ -42,7 +47,72 @@ impl ContractClass { } } } -/// All relevant information about a declared contract class, including the compiled class + +#[derive(Deref, Serialize, Deserialize, Clone, Debug, Eq, PartialEq)] +pub struct SierraVersion(Version); + +impl SierraVersion { + pub fn new(major: u64, minor: u64, patch: u64) -> Self { + Self(Version::new(major, minor, patch)) + } + + /// Version of deprecated contract class. + pub fn zero() -> Self { + Self(Version::new(0, 0, 0)) + } +} + +impl Default for SierraVersion { + fn default() -> Self { + Self::new(1, 0, 0) + } +} + +impl FromStr for SierraVersion { + type Err = StarknetApiError; + + fn from_str(s: &str) -> Result { + Ok(Self( + Version::parse(s) + .map_err(|_| StarknetApiError::ParseSierraVersionError(s.to_string()))?, + )) + } +} + +impl TryFrom<&Vec> for SierraVersion { + type Error = StarknetApiError; + + /// The Sierra version is determined from the first 3 Felts in the Sierra program. + fn try_from(sierra_program: &Vec) -> Result { + if sierra_program.len() < 3 { + return Err(StarknetApiError::ParseSierraVersionError( + "Sierra program length must be at least 3 Felts.".to_string(), + )); + } + + // Closure to map a Felt error to a StarknetApiError. + let map_felt_to_api_error = |(felt, index): (Felt, usize)| { + move |err| { + StarknetApiError::ParseSierraVersionError(format!( + "Failed to parse Sierra program to Sierra version. Sierra program index: {}, \ + Felt value: {}, Error: {}", + index, felt, err + )) + } + }; + + let major = + sierra_program[0].try_into().map_err(map_felt_to_api_error((sierra_program[0], 0)))?; + let minor = + sierra_program[1].try_into().map_err(map_felt_to_api_error((sierra_program[1], 1)))?; + let patch = + sierra_program[2].try_into().map_err(map_felt_to_api_error((sierra_program[2], 2)))?; + + Ok(Self::new(major, minor, patch)) + } +} + +/// All relevant information about a declared contract class, including the compiled contract class /// and other parameters derived from the original declare transaction required for billing. #[derive(Clone, Debug, Deserialize, Eq, PartialEq, Serialize)] // TODO(Ayelet,10/02/2024): Change to bytes. @@ -51,6 +121,7 @@ pub struct ClassInfo { pub contract_class: ContractClass, pub sierra_program_length: usize, pub abi_length: usize, + pub sierra_version: SierraVersion, } impl ClassInfo { @@ -84,6 +155,7 @@ impl ClassInfo { contract_class: &ContractClass, sierra_program_length: usize, abi_length: usize, + sierra_version: SierraVersion, ) -> Result { let (contract_class_version, condition) = match contract_class { ContractClass::V0(_) => (0, sierra_program_length == 0), @@ -91,7 +163,12 @@ impl ClassInfo { }; if condition { - Ok(Self { contract_class: contract_class.clone(), sierra_program_length, abi_length }) + Ok(Self { + contract_class: contract_class.clone(), + sierra_program_length, + abi_length, + sierra_version, + }) } else { Err(StarknetApiError::ContractClassVersionSierraProgramLengthMismatch { contract_class_version, diff --git a/crates/starknet_api/src/lib.rs b/crates/starknet_api/src/lib.rs index 2531709e67..0ac6cb2c04 100644 --- a/crates/starknet_api/src/lib.rs +++ b/crates/starknet_api/src/lib.rs @@ -65,6 +65,8 @@ pub enum StarknetApiError { version {cairo_version:?}.", **declare_version )] ContractClassVersionMismatch { declare_version: TransactionVersion, cairo_version: u64 }, + #[error("Failed to parse Sierra version: {0}")] + ParseSierraVersionError(String), } pub type StarknetApiResult = Result; diff --git a/crates/starknet_api/src/transaction_test.rs b/crates/starknet_api/src/transaction_test.rs index 3c029603b5..82759324a9 100644 --- a/crates/starknet_api/src/transaction_test.rs +++ b/crates/starknet_api/src/transaction_test.rs @@ -49,7 +49,7 @@ fn convert_executable_transaction_and_back() { } } }; - let executable_tx: ExecutableTransaction = (tx.clone(), tx_hash.clone()).into(); + let executable_tx: ExecutableTransaction = (tx.clone(), tx_hash).into(); let tx_back = Transaction::from(executable_tx); assert_eq!(tx, tx_back); } diff --git a/crates/starknet_gateway/src/compilation.rs b/crates/starknet_gateway/src/compilation.rs index 5e8a4db171..a9c3bc106b 100644 --- a/crates/starknet_gateway/src/compilation.rs +++ b/crates/starknet_gateway/src/compilation.rs @@ -2,7 +2,7 @@ use std::sync::Arc; use cairo_lang_starknet_classes::casm_contract_class::CasmContractClass; use cairo_lang_starknet_classes::contract_class::ContractClass as CairoLangContractClass; -use starknet_api::contract_class::{ClassInfo, ContractClass}; +use starknet_api::contract_class::{ClassInfo, ContractClass, SierraVersion}; use starknet_api::rpc_transaction::RpcDeclareTransaction; use starknet_gateway_types::errors::GatewaySpecError; use starknet_sierra_compile::command_line_compiler::CommandLineCompiler; @@ -41,10 +41,15 @@ impl GatewayCompiler { let casm_contract_class = self.compile(cairo_lang_contract_class)?; + let sierra_version: SierraVersion = (&rpc_contract_class.sierra_program) + .try_into() + .map_err(|_| GatewaySpecError::UnsupportedContractClassVersion)?; + Ok(ClassInfo { contract_class: ContractClass::V1(casm_contract_class), sierra_program_length: rpc_contract_class.sierra_program.len(), abi_length: rpc_contract_class.abi.len(), + sierra_version, }) }