diff --git a/Cargo.lock b/Cargo.lock index d5dc2208d8..0f8aee3fcf 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -10437,6 +10437,7 @@ dependencies = [ "blockifier", "cairo-lang-sierra-to-casm", "cairo-lang-starknet-classes", + "criterion", "futures", "mempool_test_utils", "mockall", diff --git a/crates/starknet_gateway/Cargo.toml b/crates/starknet_gateway/Cargo.toml index 6420c9ba5e..e94b288aaf 100644 --- a/crates/starknet_gateway/Cargo.toml +++ b/crates/starknet_gateway/Cargo.toml @@ -39,6 +39,7 @@ validator.workspace = true [dev-dependencies] assert_matches.workspace = true cairo-lang-sierra-to-casm.workspace = true +criterion.workspace = true mockall.workspace = true mockito.workspace = true num-bigint.workspace = true @@ -49,3 +50,8 @@ rstest.workspace = true starknet_mempool.workspace = true starknet_mempool_types = { workspace = true, features = ["testing"] } tracing-test.workspace = true + +[[bench]] +harness = false +name = "gateway_bench" +path = "bench/gateway_bench.rs" diff --git a/crates/starknet_gateway/bench/gateway_bench.rs b/crates/starknet_gateway/bench/gateway_bench.rs new file mode 100644 index 0000000000..3c0acf6360 --- /dev/null +++ b/crates/starknet_gateway/bench/gateway_bench.rs @@ -0,0 +1,19 @@ +//! Benchmark module for the starknet gateway crate. It provides functionalities to benchmark +//! the performance of the gateway service, including declare, deploy account and invoke +//! transactions. +//! +//! There are four benchmark functions in this flow: `declare_benchmark`, +//! `deploy_account_benchmark`, `invoke_benchmark` and `gateway_benchmark` which combines all of the +//! types. Each of the functions measure the performance of the gateway handling randomly created +//! txs of the respective type. +//! +//! Run the benchmarks using `cargo bench --bench gateway_bench`. + +use criterion::{criterion_group, criterion_main, Criterion}; + +fn invoke_benchmark(criterion: &mut Criterion) { + criterion.bench_function("invokes", |benchmark| benchmark.iter(|| {})); +} + +criterion_group!(benches, invoke_benchmark); +criterion_main!(benches); diff --git a/crates/starknet_gateway/src/gateway.rs b/crates/starknet_gateway/src/gateway.rs index 102e2cf0b6..f7577aa036 100644 --- a/crates/starknet_gateway/src/gateway.rs +++ b/crates/starknet_gateway/src/gateway.rs @@ -26,25 +26,21 @@ use crate::utils::compile_contract_and_build_executable_tx; #[path = "gateway_test.rs"] pub mod gateway_test; -pub struct Gateway { - pub config: GatewayConfig, +struct GatewayBusinessLogic { pub stateless_tx_validator: Arc, pub stateful_tx_validator: Arc, pub state_reader_factory: Arc, pub gateway_compiler: Arc, - pub mempool_client: SharedMempoolClient, pub chain_info: ChainInfo, } -impl Gateway { +impl GatewayBusinessLogic { pub fn new( config: GatewayConfig, state_reader_factory: Arc, gateway_compiler: GatewayCompiler, - mempool_client: SharedMempoolClient, ) -> Self { Self { - config: config.clone(), stateless_tx_validator: Arc::new(StatelessTransactionValidator { config: config.stateless_tx_validator_config.clone(), }), @@ -53,31 +49,49 @@ impl Gateway { }), state_reader_factory, gateway_compiler: Arc::new(gateway_compiler), - mempool_client, chain_info: config.chain_info.clone(), } } + pub async fn add_tx(&self, tx: RpcTransaction) -> GatewayResult { + info!("Processing tx"); + let blocking_task = ProcessTxBlockingTask::new(self, tx); + // Run the blocking task in the current span. + let curr_span = Span::current(); + tokio::task::spawn_blocking(move || curr_span.in_scope(|| blocking_task.process_tx())) + .await + .map_err(|join_err| { + error!("Failed to process tx: {}", join_err); + GatewaySpecError::UnexpectedError { data: "Internal server error".to_owned() } + })? + } +} + +pub struct Gateway { + pub mempool_client: SharedMempoolClient, + business_logic: GatewayBusinessLogic, +} + +impl Gateway { + pub fn new( + config: GatewayConfig, + state_reader_factory: Arc, + gateway_compiler: GatewayCompiler, + mempool_client: SharedMempoolClient, + ) -> Self { + let business_logic = + GatewayBusinessLogic::new(config, state_reader_factory, gateway_compiler); + Self { business_logic, mempool_client } + } + #[instrument(skip(self), ret)] pub async fn add_tx( &self, tx: RpcTransaction, p2p_message_metadata: Option, ) -> GatewayResult { - info!("Processing tx"); - let blocking_task = ProcessTxBlockingTask::new(self, tx); - // Run the blocking task in the current span. - let curr_span = Span::current(); - let add_tx_args = - tokio::task::spawn_blocking(move || curr_span.in_scope(|| blocking_task.process_tx())) - .await - .map_err(|join_err| { - error!("Failed to process tx: {}", join_err); - GatewaySpecError::UnexpectedError { data: "Internal server error".to_owned() } - })??; - + let add_tx_args = self.business_logic.add_tx(tx).await?; let tx_hash = add_tx_args.tx.tx_hash(); - let add_tx_args = AddTransactionArgsWrapper { args: add_tx_args, p2p_message_metadata }; self.mempool_client.add_tx(add_tx_args).await.map_err(|e| { error!("Failed to send tx to mempool: {}", e); @@ -100,7 +114,7 @@ struct ProcessTxBlockingTask { } impl ProcessTxBlockingTask { - pub fn new(gateway: &Gateway, tx: RpcTransaction) -> Self { + pub fn new(gateway: &GatewayBusinessLogic, tx: RpcTransaction) -> Self { Self { stateless_tx_validator: gateway.stateless_tx_validator.clone(), stateful_tx_validator: gateway.stateful_tx_validator.clone(), @@ -123,7 +137,7 @@ impl ProcessTxBlockingTask { &self.chain_info.chain_id, )?; - // Perfom post compilation validations. + // Perform post compilation validations. if let AccountTransaction::Declare(executable_declare_tx) = &executable_tx { if !executable_declare_tx.validate_compiled_class_hash() { return Err(GatewaySpecError::CompiledClassHashMismatch); diff --git a/crates/starknet_gateway/src/stateful_transaction_validator_test.rs b/crates/starknet_gateway/src/stateful_transaction_validator_test.rs index de734b71cd..366ccd4431 100644 --- a/crates/starknet_gateway/src/stateful_transaction_validator_test.rs +++ b/crates/starknet_gateway/src/stateful_transaction_validator_test.rs @@ -88,7 +88,7 @@ fn test_instantiate_validator(stateful_validator: StatefulTransactionValidator) let mut mock_state_reader_factory = MockStateReaderFactory::new(); - // Make sure stateful_validator uses the latest block in the initiall call. + // Make sure stateful_validator uses the latest block in the initial call. let latest_state_reader = state_reader_factory.get_state_reader_from_latest_block(); mock_state_reader_factory .expect_get_state_reader_from_latest_block()