diff --git a/Cargo.lock b/Cargo.lock index e7373942611..b49a145fc6d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -7298,6 +7298,7 @@ dependencies = [ "alloy-json-rpc", "alloy-primitives", "alloy-provider", + "alloy-rpc-types-eth", "alloy-sol-types", "alloy-transport", "alloy-transport-http", diff --git a/Cargo.toml b/Cargo.toml index a51d19f4d00..b178f1e2588 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -72,6 +72,7 @@ alloy-dyn-abi = "0.8.3" alloy-json-rpc = "0.3.5" alloy-primitives = "0.8.3" alloy-provider = "0.3.5" +alloy-rpc-types-eth = "0.3.5" alloy-sol-types = "0.8.3" alloy-transport = "0.3.5" alloy-transport-http = "0.3.5" diff --git a/crates/papyrus_base_layer/Cargo.toml b/crates/papyrus_base_layer/Cargo.toml index dcc6387f5b5..ab1718b42c1 100644 --- a/crates/papyrus_base_layer/Cargo.toml +++ b/crates/papyrus_base_layer/Cargo.toml @@ -17,6 +17,7 @@ alloy-dyn-abi.workspace = true alloy-json-rpc.workspace = true alloy-primitives.workspace = true alloy-provider.workspace = true +alloy-rpc-types-eth.workspace = true alloy-sol-types = { workspace = true, features = ["json"] } alloy-transport.workspace = true alloy-transport-http.workspace = true diff --git a/crates/papyrus_base_layer/src/eth_events.rs b/crates/papyrus_base_layer/src/eth_events.rs new file mode 100644 index 00000000000..5fd6eae0411 --- /dev/null +++ b/crates/papyrus_base_layer/src/eth_events.rs @@ -0,0 +1,87 @@ +use std::sync::Arc; + +use alloy_primitives::{Address as EthereumContractAddress, U256}; +use alloy_rpc_types_eth::Log; +use alloy_sol_types::SolEvent; +use starknet_api::core::{EntryPointSelector, Nonce}; +use starknet_api::transaction::fields::Calldata; +use starknet_types_core::felt::Felt; + +use crate::ethereum_base_layer_contract::{ + EthereumBaseLayerError, + EthereumBaseLayerResult, + Starknet, +}; +use crate::{MessageData, StarknetEvent}; + +impl TryFrom for StarknetEvent { + type Error = EthereumBaseLayerError; + + fn try_from(log: Log) -> EthereumBaseLayerResult { + let event_signature = + *log.topic0().ok_or_else(|| Self::Error::UnhandledL1Event(log.inner.clone()))?; + let validate = true; + let log = log.inner; + + Ok(match event_signature { + sig if sig == Starknet::LogMessageToL2::SIGNATURE_HASH => { + todo!() + } + sig if sig == Starknet::MessageToL2CancellationStarted::SIGNATURE_HASH => { + let decoded = Starknet::MessageToL2CancellationStarted::decode_log(&log, validate)?; + StarknetEvent::MessageToL2CancellationStarted(decoded.try_into()?) + } + sig if sig == Starknet::MessageToL2Canceled::SIGNATURE_HASH => { + todo!() + } + sig if sig == Starknet::ConsumedMessageToL1::SIGNATURE_HASH => { + todo!() + } + _ => return Err(EthereumBaseLayerError::UnhandledL1Event(log)), + }) + } +} + +impl TryFrom> for MessageData { + type Error = EthereumBaseLayerError; + + fn try_from( + decoded: alloy_primitives::Log, + ) -> EthereumBaseLayerResult { + create_l1_message_data( + decoded.fromAddress, + decoded.toAddress, + decoded.selector, + &decoded.payload, + decoded.nonce, + ) + } +} + +pub fn create_l1_message_data( + from_address: EthereumContractAddress, + to_address: U256, + selector: U256, + payload: &[U256], + nonce: U256, +) -> EthereumBaseLayerResult { + Ok(MessageData { + from_address: felt_from_eth_address(from_address) + .try_into() + .map_err(EthereumBaseLayerError::StarknetApiParsingError)?, + to_address: felt_from_u256(to_address) + .try_into() + .map_err(EthereumBaseLayerError::StarknetApiParsingError)?, + entry_point_selector: EntryPointSelector(felt_from_u256(selector)), + payload: Calldata(Arc::new(payload.iter().map(|&x| felt_from_u256(x)).collect())), + nonce: Nonce(felt_from_u256(nonce)), + }) +} + +pub fn felt_from_eth_address(address: EthereumContractAddress) -> Felt { + Felt::from_bytes_be_slice(address.0.as_slice()) +} + +pub fn felt_from_u256(num: U256) -> Felt { + Felt::from_bytes_be(&num.to_be_bytes()) +} diff --git a/crates/papyrus_base_layer/src/ethereum_base_layer_contract.rs b/crates/papyrus_base_layer/src/ethereum_base_layer_contract.rs index d76fddaed38..410614c1443 100644 --- a/crates/papyrus_base_layer/src/ethereum_base_layer_contract.rs +++ b/crates/papyrus_base_layer/src/ethereum_base_layer_contract.rs @@ -3,9 +3,10 @@ use std::future::IntoFuture; use alloy_dyn_abi::SolType; use alloy_json_rpc::RpcError; -pub(crate) use alloy_primitives::Address as EthereumContractAddress; +use alloy_primitives::Address as EthereumContractAddress; use alloy_provider::network::Ethereum; use alloy_provider::{Provider, ProviderBuilder, RootProvider}; +use alloy_rpc_types_eth::{BlockNumberOrTag, Filter as EthEventFilter}; use alloy_sol_types::{sol, sol_data}; use alloy_transport::TransportErrorKind; use alloy_transport_http::{Client, Http}; @@ -15,7 +16,7 @@ use papyrus_config::{ParamPath, ParamPrivacyInput, SerializationType, Serialized use serde::{Deserialize, Serialize}; use starknet_api::block::{BlockHash, BlockHashAndNumber, BlockNumber}; use starknet_api::hash::StarkHash; -use starknet_types_core::felt; +use starknet_api::StarknetApiError; use url::Url; use crate::{BaseLayerContract, StarknetEvent}; @@ -81,11 +82,23 @@ impl BaseLayerContract for EthereumBaseLayerContract { async fn events( &self, - _from_block: u64, - _until_block: u64, - _event_identifiers: Vec<&str>, + from_block: u64, + until_block: u64, + events: Vec<&str>, ) -> EthereumBaseLayerResult> { - todo!("Implmeneted in a subsequent commit") + let filter = EthEventFilter::new() + .from_block(BlockNumberOrTag::Number(from_block)) + .events(events) + .to_block(until_block); + + // Get all logs from the latest block that match the filter. + self.contract + .provider() + .get_logs(&filter) + .await? + .into_iter() + .map(TryInto::try_into) + .collect() } async fn latest_l1_block_number(&self, finality: u64) -> EthereumBaseLayerResult> { @@ -98,13 +111,13 @@ pub enum EthereumBaseLayerError { #[error(transparent)] Contract(#[from] alloy_contract::Error), #[error(transparent)] - FeltParseError(#[from] felt::FromStrError), - #[error(transparent)] RpcError(#[from] RpcError), #[error(transparent)] - Serde(#[from] serde_json::Error), - #[error(transparent)] TypeError(#[from] alloy_sol_types::Error), + #[error("{0}")] + StarknetApiParsingError(StarknetApiError), + #[error("{0:?}")] + UnhandledL1Event(alloy_primitives::Log), } #[derive(Debug, Deserialize, Serialize, Clone, PartialEq)] diff --git a/crates/papyrus_base_layer/src/lib.rs b/crates/papyrus_base_layer/src/lib.rs index 8679801e8f1..1b157bfe7ea 100644 --- a/crates/papyrus_base_layer/src/lib.rs +++ b/crates/papyrus_base_layer/src/lib.rs @@ -7,6 +7,8 @@ use starknet_api::transaction::L1HandlerTransaction; pub mod ethereum_base_layer_contract; +pub(crate) mod eth_events; + #[cfg(any(feature = "testing", test))] pub mod test_utils;