diff --git a/.env-example b/.env-example index 11e75752ee4b..ed1e1d1fd348 100644 --- a/.env-example +++ b/.env-example @@ -5,6 +5,12 @@ CHAIN=tevmmainnet RETH_CONFIG=/path/to/datadir/config.toml RETH_RPC_ADDRESS=0.0.0.0 RETH_RPC_PORT=8545 +RETH_WS_ADDRESS=0.0.0.0 +RETH_WS_PORT=8546 +RETH_AUTH_RPC_ADDRESS=127.0.0.1 +RETH_AUTH_RPC_PORT=8551 +RETH_IPCPATH=/path/to/reth.ipc +RETH_DISCOVERY_PORT=30303 TELOS_ENDPOINT=https://mainnet.telos.net TELOS_SIGNER_ACCOUNT=rpc.evm TELOS_SIGNER_PERMISSION=rpc diff --git a/.github/assets/hive/expected_failures_experimental.yaml b/.github/assets/hive/expected_failures_experimental.yaml deleted file mode 100644 index 66a68f0c67e7..000000000000 --- a/.github/assets/hive/expected_failures_experimental.yaml +++ /dev/null @@ -1,65 +0,0 @@ -# https://github.com/paradigmxyz/reth/issues/7015 -# https://github.com/paradigmxyz/reth/issues/6332 -rpc-compat: - - debug_getRawBlock/get-invalid-number (reth) - - debug_getRawHeader/get-invalid-number (reth) - - debug_getRawReceipts/get-invalid-number (reth) - - debug_getRawTransaction/get-invalid-hash (reth) - - - eth_call/call-callenv (reth) - - eth_feeHistory/fee-history (reth) - - eth_getStorageAt/get-storage-invalid-key-too-large (reth) - - eth_getStorageAt/get-storage-invalid-key (reth) - - eth_getTransactionReceipt/get-access-list (reth) - - eth_getTransactionReceipt/get-blob-tx (reth) - - eth_getTransactionReceipt/get-dynamic-fee (reth) - -# https://github.com/paradigmxyz/reth/issues/8732 -engine-withdrawals: - - Withdrawals Fork On Genesis (Paris) (reth) - - Withdrawals Fork on Block 1 (Paris) (reth) - - Withdrawals Fork on Block 2 (Paris) (reth) - - Withdrawals Fork on Block 3 (Paris) (reth) - - Withdraw to a single account (Paris) (reth) - - Withdraw to two accounts (Paris) (reth) - - Withdraw many accounts (Paris) (reth) - - Withdraw zero amount (Paris) (reth) - - Empty Withdrawals (Paris) (reth) - - Corrupted Block Hash Payload (INVALID) (Paris) (reth) - - Withdrawals Fork on Block 1 - 8 Block Re-Org NewPayload (Paris) (reth) - - Withdrawals Fork on Block 1 - 8 Block Re-Org, Sync (Paris) (reth) - - Withdrawals Fork on Block 8 - 10 Block Re-Org NewPayload (Paris) (reth) - - Withdrawals Fork on Block 8 - 10 Block Re-Org Sync (Paris) (reth) - - Withdrawals Fork on Canonical Block 8 / Side Block 7 - 10 Block Re-Org (Paris) (reth) - - Withdrawals Fork on Canonical Block 8 / Side Block 7 - 10 Block Re-Org Sync (Paris) (reth) - - Withdrawals Fork on Canonical Block 8 / Side Block 9 - 10 Block Re-Org (Paris) (reth) - - Withdrawals Fork on Canonical Block 8 / Side Block 9 - 10 Block Re-Org Sync (Paris) (reth) - -# https://github.com/paradigmxyz/reth/issues/8305 -# https://github.com/paradigmxyz/reth/issues/6217 -engine-api: - - Re-org to Previously Validated Sidechain Payload (Paris) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=False, Invalid P9 (Paris) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=True, Invalid P9 (Paris) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=False, Invalid P10 (Paris) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=True, Invalid P10 (Paris) (reth) - -# https://github.com/paradigmxyz/reth/issues/8305 -# https://github.com/paradigmxyz/reth/issues/6217 -# https://github.com/paradigmxyz/reth/issues/8306 -# https://github.com/paradigmxyz/reth/issues/7144 -engine-cancun: - - Blob Transaction Ordering, Multiple Clients (Cancun) (reth) - - Invalid PayloadAttributes, Missing BeaconRoot, Syncing=True (Cancun) (reth) - - Invalid NewPayload, BlobGasUsed, Syncing=True, EmptyTxs=False, DynFeeTxs=False (Cancun) (reth) - - Invalid NewPayload, Blob Count on BlobGasUsed, Syncing=True, EmptyTxs=False, DynFeeTxs=False (Cancun) (reth) - - Invalid NewPayload, ExcessBlobGas, Syncing=True, EmptyTxs=False, DynFeeTxs=False (Cancun) (reth) - - Re-org to Previously Validated Sidechain Payload (Cancun) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=False, Invalid P9 (Cancun) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=True, Invalid P9 (Cancun) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=False, Invalid P10 (Cancun) (reth) - - Invalid Missing Ancestor ReOrg, StateRoot, EmptyTxs=True, Invalid P10 (Cancun) (reth) - -# https://github.com/paradigmxyz/reth/issues/8579 -sync: - - sync reth -> reth diff --git a/.github/workflows/telos_ci.yml b/.github/workflows/telos_ci.yml index e69de29bb2d1..22dfb0192667 100644 --- a/.github/workflows/telos_ci.yml +++ b/.github/workflows/telos_ci.yml @@ -0,0 +1,67 @@ +on: [pull_request] + +name: Continuous integration + +jobs: +# check: +# name: Check +# runs-on: ubuntu-latest +# steps: +# - uses: actions/checkout@v2 +# - uses: actions-rs/toolchain@v1 +# with: +# profile: minimal +# toolchain: stable +# override: true +# - uses: actions-rs/cargo@v1 +# with: +# command: check +# args: --workspace +# + test: + name: Test Suite + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v2 + - uses: actions-rs/toolchain@v1 + with: + profile: minimal + toolchain: stable + override: true + - uses: actions-rs/cargo@v1 + with: + command: test + args: --package reth-node-telos --features telos --test integration testing_chain_sync -- --exact --nocapture + +# fmt: +# name: Rustfmt +# runs-on: ubuntu-latest +# steps: +# - uses: actions/checkout@v2 +# - uses: actions-rs/toolchain@v1 +# with: +# profile: minimal +# toolchain: stable +# override: true +# - run: rustup component add rustfmt +# - uses: actions-rs/cargo@v1 +# with: +# command: fmt +# args: --all -- --check + + +# clippy: +# name: Clippy +# runs-on: ubuntu-latest +# steps: +# - uses: actions/checkout@v2 +# - uses: actions-rs/toolchain@v1 +# with: +# profile: minimal +# toolchain: stable +# override: true +# - run: rustup component add clippy +# - uses: actions-rs/cargo@v1 +# with: +# command: clippy +# args: --workspace -- -D warnings \ No newline at end of file diff --git a/Cargo.lock b/Cargo.lock index c20194a01a30..c6bfed15739f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -886,39 +886,7 @@ dependencies = [ [[package]] name = "antelope-client" version = "0.2.1" -source = "git+https://github.com/telosnetwork/antelope-rs.git?branch=development#2587e09099f75b23b90849e00057ce67a2be684b" -dependencies = [ - "antelope-client-macros", - "async-trait", - "base64 0.21.7", - "bs58", - "chrono", - "digest 0.10.7", - "ecdsa", - "flate2", - "hex", - "hmac 0.12.1", - "k256", - "log", - "once_cell", - "p256", - "rand 0.8.5", - "rand_core 0.6.4", - "reqwest 0.11.27", - "ripemd", - "serde", - "serde-big-array", - "serde_json", - "sha2 0.10.8", - "signature", - "thiserror", - "tokio", -] - -[[package]] -name = "antelope-client" -version = "0.2.1" -source = "git+https://github.com/telosnetwork/antelope-rs?branch=finish_table_rows_params#feb151b28499946698c25c3411b574801cc869e6" +source = "git+https://github.com/telosnetwork/antelope-rs?branch=development#ed53c75b6b9ce14aa72002ddd491a6697ae563d9" dependencies = [ "antelope-client-macros", "async-trait", @@ -4229,7 +4197,7 @@ dependencies = [ "httpdate", "itoa", "pin-project-lite", - "socket2 0.5.7", + "socket2 0.4.10", "tokio", "tower-service", "tracing", @@ -5131,7 +5099,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4979f22fdb869068da03c9f7528f8297c6fd2606bc3a4affe42e6a823fdb8da4" dependencies = [ "cfg-if", - "windows-targets 0.48.5", + "windows-targets 0.52.6", ] [[package]] @@ -7871,6 +7839,7 @@ dependencies = [ "reth-stages-api", "reth-static-file", "reth-tasks", + "reth-telos-rpc-engine-api 1.0.6", "reth-tracing", "reth-trie", "thiserror", @@ -7898,6 +7867,7 @@ dependencies = [ "reth-revm", "reth-rpc-types 1.0.6", "reth-rpc-types-compat", + "reth-telos-primitives-traits", "reth-telos-rpc-engine-api 1.0.6", "reth-trie", "revm-primitives", @@ -8064,6 +8034,7 @@ dependencies = [ "reth-primitives 1.0.6", "reth-provider", "reth-revm", + "reth-telos-primitives-traits", "reth-transaction-pool", "reth-trie", "revm", @@ -8094,6 +8065,7 @@ dependencies = [ "reth-primitives 1.0.6", "reth-prune-types", "reth-storage-errors 1.0.6", + "reth-telos-primitives-traits", "reth-telos-rpc-engine-api 1.0.6", "revm", "revm-primitives", @@ -8113,6 +8085,7 @@ dependencies = [ "reth-primitives 1.0.6", "reth-prune-types", "reth-revm", + "reth-telos-primitives-traits", "reth-telos-rpc-engine-api 1.0.6", "reth-testing-utils", "revm-primitives", @@ -8764,15 +8737,18 @@ dependencies = [ name = "reth-node-telos" version = "1.0.6" dependencies = [ + "alloy-consensus", "alloy-contract", "alloy-network", "alloy-provider", + "alloy-rpc-client", "alloy-rpc-types", "alloy-signer-local", "alloy-sol-types", "alloy-transport-http", - "antelope-client 0.2.1 (git+https://github.com/telosnetwork/antelope-rs?branch=finish_table_rows_params)", + "antelope-client", "clap", + "derive_more 1.0.0", "env_logger 0.11.5", "eyre", "reqwest 0.12.7", @@ -8978,6 +8954,7 @@ dependencies = [ "reth-primitives 1.0.6", "reth-rpc-types 1.0.6", "reth-rpc-types-compat", + "reth-telos-rpc-engine-api 1.0.6", ] [[package]] @@ -9012,6 +8989,7 @@ dependencies = [ "reth-optimism-chainspec", "reth-primitives-traits 1.0.6", "reth-static-file-types 1.0.6", + "reth-telos-primitives-traits", "reth-trie-common 1.0.6", "revm-primitives", "secp256k1", @@ -9075,6 +9053,7 @@ dependencies = [ "proptest-arbitrary-interop", "rand 0.8.5", "reth-codecs 1.0.6", + "reth-telos-primitives-traits", "revm-primitives", "roaring", "serde", @@ -9253,6 +9232,7 @@ dependencies = [ "reth-rpc-types 1.0.6", "reth-rpc-types-compat", "reth-tasks", + "reth-telos-primitives-traits", "reth-testing-utils", "reth-transaction-pool", "reth-trie", @@ -9408,6 +9388,8 @@ dependencies = [ "reth-rpc-types 1.0.6", "reth-rpc-types-compat", "reth-tasks", + "reth-telos-primitives-traits", + "reth-telos-rpc-engine-api 1.0.6", "reth-transaction-pool", "reth-trie", "revm", @@ -9534,6 +9516,7 @@ dependencies = [ "alloy-rpc-types", "reth-primitives 1.0.6", "reth-rpc-types 1.0.6", + "reth-telos-primitives-traits", "reth-trie-common 1.0.6", "serde_json", ] @@ -9730,13 +9713,25 @@ dependencies = [ "tracing-futures", ] +[[package]] +name = "reth-telos-primitives-traits" +version = "1.0.6" +dependencies = [ + "alloy-primitives 0.8.3", + "arbitrary", + "bytes", + "modular-bitfield", + "reth-codecs 1.0.6", + "serde", +] + [[package]] name = "reth-telos-rpc" version = "1.0.6" dependencies = [ "alloy-network", "alloy-primitives 0.8.3", - "antelope-client 0.2.1 (git+https://github.com/telosnetwork/antelope-rs?branch=finish_table_rows_params)", + "antelope-client", "async-trait", "jsonrpsee-types", "log", @@ -11190,7 +11185,7 @@ dependencies = [ "alloy", "alloy-consensus", "alloy-rlp", - "antelope-client 0.2.1 (git+https://github.com/telosnetwork/antelope-rs.git?branch=development)", + "antelope-client", "arrowbatch", "base64 0.22.1", "bytes", @@ -11227,7 +11222,7 @@ dependencies = [ "alloy-consensus", "alloy-eips", "alloy-rlp", - "antelope-client 0.2.1 (git+https://github.com/telosnetwork/antelope-rs.git?branch=development)", + "antelope-client", "bytes", "clap", "dashmap 5.5.3", @@ -12350,7 +12345,7 @@ version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.48.0", + "windows-sys 0.59.0", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index b3102446b6da..003103367d64 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -115,6 +115,7 @@ members = [ "crates/storage/storage-api/", "crates/tasks/", "crates/telos/node", + "crates/telos/primitives-traits", "crates/telos/rpc", "crates/tokio-util/", "crates/tracing/", @@ -568,5 +569,6 @@ test-fuzz = "5" # telos reth-node-telos = { path = "crates/telos/node" } reth-telos-rpc = { path = "crates/telos/rpc" } +reth-telos-primitives-traits = { path = "crates/telos/primitives-traits" } reth-telos-rpc-engine-api = { path = "crates/telos/rpc-engine-api" } -antelope-client = { git = "https://github.com/telosnetwork/antelope-rs", branch = "finish_table_rows_params" } +antelope-client = { git = "https://github.com/telosnetwork/antelope-rs", branch = "development" } diff --git a/crates/blockchain-tree/src/blockchain_tree.rs b/crates/blockchain-tree/src/blockchain_tree.rs index d7b90a99f465..428ee5f96d48 100644 --- a/crates/blockchain-tree/src/blockchain_tree.rs +++ b/crates/blockchain-tree/src/blockchain_tree.rs @@ -22,11 +22,7 @@ use reth_primitives::{ BlockHash, BlockNumHash, BlockNumber, EthereumHardfork, ForkBlock, Receipt, SealedBlock, SealedBlockWithSenders, SealedHeader, StaticFileSegment, B256, U256, }; -use reth_provider::{ - BlockExecutionWriter, BlockNumReader, BlockWriter, CanonStateNotification, - CanonStateNotificationSender, CanonStateNotifications, ChainSpecProvider, ChainSplit, - ChainSplitTarget, DisplayBlocksChain, HeaderProvider, ProviderError, StaticFileProviderFactory, -}; +use reth_provider::{BlockExecutionWriter, BlockNumReader, BlockReader, BlockWriter, CanonStateNotification, CanonStateNotificationSender, CanonStateNotifications, ChainSpecProvider, ChainSplit, ChainSplitTarget, DisplayBlocksChain, HeaderProvider, ProviderError, StaticFileProviderFactory}; use reth_prune_types::PruneModes; use reth_stages_api::{MetricEvent, MetricEventsSender}; #[cfg(not(feature = "telos"))] @@ -41,6 +37,7 @@ use std::{ sync::Arc, }; use tracing::{debug, error, info, instrument, trace, warn}; +use reth_revm::interpreter::instructions::system::gas; #[cfg_attr(doc, aquamarine::aquamarine)] /// A Tree of chains. @@ -804,6 +801,50 @@ where _ => {} } + #[cfg(feature = "telos")] + let parent_telos_ext; + #[cfg(feature = "telos")] + let mut gas_price_change; + #[cfg(feature = "telos")] + let mut revision_number_change; + #[cfg(feature = "telos")] + { + gas_price_change = None; + revision_number_change = None; + if let Some(telos_extra_fields) = telos_extra_fields.as_ref() { + gas_price_change = telos_extra_fields.gasprice_changes; + revision_number_change = telos_extra_fields.revision_changes; + } + let parent_hash = block.block.header.parent_hash; + let provider = self.externals.provider_factory.provider(); + let block_by_hash = provider.unwrap().block_by_hash(parent_hash); + if let Some(block_by_hash) = block_by_hash.unwrap() { + parent_telos_ext = block_by_hash.header.telos_block_extension; + } else { + let sidechain_block = self.sidechain_block_by_hash(parent_hash); + if let Some(sidechain_block) = sidechain_block { + parent_telos_ext = sidechain_block.header.telos_block_extension.clone(); + } else { + panic!("Parent block not found"); + } + } + } + #[cfg(feature = "telos")] + let block = SealedBlockWithSenders { + block: SealedBlock { + header: SealedHeader::new(block.block.header.clone_with_telos( + parent_telos_ext, + gas_price_change, + revision_number_change, + ), block.block.header.hash()), + body: block.block.body.clone(), + ommers: block.block.ommers.clone(), + withdrawals: block.block.withdrawals.clone(), + requests: block.block.requests.clone(), + }, + senders: block.senders, + }; + // validate block consensus rules if let Err(err) = self.validate_block(&block) { return Err(InsertBlockError::consensus_error(err, block.block)) diff --git a/crates/consensus/beacon/src/engine/mod.rs b/crates/consensus/beacon/src/engine/mod.rs index 81bad7e81b85..75a68eb9f441 100644 --- a/crates/consensus/beacon/src/engine/mod.rs +++ b/crates/consensus/beacon/src/engine/mod.rs @@ -1083,6 +1083,8 @@ where &mut self, payload: ExecutionPayload, cancun_fields: Option, + #[cfg(feature = "telos")] + telos_extra_fields: TelosEngineAPIExtraFields, ) -> Result, BeaconOnNewPayloadError> { self.metrics.new_payload_messages.increment(1); @@ -1114,7 +1116,7 @@ where let parent_hash = payload.parent_hash(); let block = match self .payload_validator - .ensure_well_formed_payload(payload, cancun_fields.into()) + .ensure_well_formed_payload(payload, cancun_fields.into(), #[cfg(feature = "telos")] telos_extra_fields) { Ok(block) => block, Err(error) => { @@ -1866,7 +1868,7 @@ where this.on_forkchoice_updated(state, payload_attrs, tx); } BeaconEngineMessage::NewPayload { payload, cancun_fields, tx, #[cfg(feature = "telos")] telos_extra_fields } => { - match this.on_new_payload(payload, cancun_fields) { + match this.on_new_payload(payload, cancun_fields, #[cfg(feature = "telos")] telos_extra_fields.clone().unwrap_or_default()) { Ok(Either::Right(block)) => { this.set_blockchain_tree_action( BlockchainTreeAction::InsertNewPayload { block, tx, #[cfg(feature = "telos")] telos_extra_fields }, diff --git a/crates/engine/tree/Cargo.toml b/crates/engine/tree/Cargo.toml index a457dd1f969b..59a01dede963 100644 --- a/crates/engine/tree/Cargo.toml +++ b/crates/engine/tree/Cargo.toml @@ -54,6 +54,9 @@ reth-stages = { workspace = true, optional = true } reth-static-file = { workspace = true, optional = true } reth-tracing = { workspace = true, optional = true } +# telos +reth-telos-rpc-engine-api = { workspace = true, optional = true } + [dev-dependencies] # reth reth-db = { workspace = true, features = ["test-utils"] } @@ -86,4 +89,4 @@ test-utils = [ "reth-static-file", "reth-tracing", ] -telos = [] +telos = ["dep:reth-telos-rpc-engine-api"] diff --git a/crates/engine/tree/src/tree/mod.rs b/crates/engine/tree/src/tree/mod.rs index 574a29f0e7ac..b5da7f66fefd 100644 --- a/crates/engine/tree/src/tree/mod.rs +++ b/crates/engine/tree/src/tree/mod.rs @@ -60,6 +60,7 @@ mod config; mod metrics; use crate::{engine::EngineApiRequest, tree::metrics::EngineApiMetrics}; pub use config::TreeConfig; +use reth_telos_rpc_engine_api::structs::TelosEngineAPIExtraFields; /// Keeps track of the state of the tree. /// @@ -593,6 +594,8 @@ where &mut self, payload: ExecutionPayload, cancun_fields: Option, + #[cfg(feature = "telos")] + telos_extra_fields: TelosEngineAPIExtraFields ) -> Result, InsertBlockFatalError> { trace!(target: "engine", "invoked new payload"); self.metrics.new_payload_messages.increment(1); @@ -625,7 +628,7 @@ where let parent_hash = payload.parent_hash(); let block = match self .payload_validator - .ensure_well_formed_payload(payload, cancun_fields.into()) + .ensure_well_formed_payload(payload, cancun_fields.into(), telos_extra_fields) { Ok(block) => block, Err(error) => { @@ -956,8 +959,8 @@ where error!("Failed to send event: {err:?}"); } } - BeaconEngineMessage::NewPayload { payload, cancun_fields, tx, #[cfg(feature = "telos")] telos_extra_fields: _} => { - let output = self.on_new_payload(payload, cancun_fields); + BeaconEngineMessage::NewPayload { payload, cancun_fields, tx, #[cfg(feature = "telos")] telos_extra_fields} => { + let output = self.on_new_payload(payload, cancun_fields, #[cfg(feature = "telos")] telos_extra_fields.unwrap_or_default()); if let Err(err) = tx.send(output.map(|o| o.outcome).map_err(|e| { reth_beacon_consensus::BeaconOnNewPayloadError::Internal( Box::new(e), diff --git a/crates/engine/util/Cargo.toml b/crates/engine/util/Cargo.toml index 932ac8f01899..80dfb8eb9198 100644 --- a/crates/engine/util/Cargo.toml +++ b/crates/engine/util/Cargo.toml @@ -46,6 +46,7 @@ tracing.workspace = true # telos reth-telos-rpc-engine-api = { workspace = true, optional = true } +reth-telos-primitives-traits = { workspace = true, optional = true } [features] optimism = [ @@ -55,4 +56,5 @@ optimism = [ telos = [ "revm-primitives/telos", "dep:reth-telos-rpc-engine-api", + "dep:reth-telos-primitives-traits" ] diff --git a/crates/engine/util/src/reorg.rs b/crates/engine/util/src/reorg.rs index 7c4db61624a1..70067b722a86 100644 --- a/crates/engine/util/src/reorg.rs +++ b/crates/engine/util/src/reorg.rs @@ -33,6 +33,8 @@ use std::{ }; use tokio::sync::oneshot; use tracing::*; +use reth_telos_primitives_traits::TelosBlockExtension; +use reth_telos_rpc_engine_api::structs::TelosEngineAPIExtraFields; #[derive(Debug)] enum EngineReorgState { @@ -167,6 +169,8 @@ where *this.depth, payload.clone(), cancun_fields.clone(), + #[cfg(feature = "telos")] + telos_extra_fields.clone(), ) { Ok(result) => result, Err(error) => { @@ -237,6 +241,8 @@ fn create_reorg_head( mut depth: usize, next_payload: ExecutionPayload, next_cancun_fields: Option, + #[cfg(feature = "telos")] + telos_extra_fields: Option, ) -> RethResult<(ExecutionPayload, Option)> where Provider: BlockReader + StateProviderFactory, @@ -246,7 +252,7 @@ where // Ensure next payload is valid. let next_block = payload_validator - .ensure_well_formed_payload(next_payload, next_cancun_fields.into()) + .ensure_well_formed_payload(next_payload, next_cancun_fields.into(), #[cfg(feature = "telos")] telos_extra_fields.clone().unwrap_or_default()) .map_err(RethError::msg)?; // Fetch reorg target block depending on its depth and its parent. @@ -302,6 +308,8 @@ where &mut evm, )?; + #[cfg(feature = "telos")] + let mut tx_index = 0; let mut cumulative_gas_used = 0; let mut sum_blob_gas_used = 0; let mut transactions = Vec::new(); @@ -317,7 +325,7 @@ where let tx_recovered = tx.clone().try_into_ecrecovered().map_err(|_| { BlockExecutionError::Validation(BlockValidationError::SenderRecoveryError) })?; - evm_config.fill_tx_env(evm.tx_mut(), &tx_recovered, tx_recovered.signer()); + evm_config.fill_tx_env(evm.tx_mut(), &tx_recovered, tx_recovered.signer(), #[cfg(feature = "telos")] reorg_target.header.telos_block_extension.tx_env_at(tx_index)); let exec_result = match evm.transact() { Ok(result) => result, error @ Err(EVMError::Transaction(_) | EVMError::Header(_)) => { @@ -350,6 +358,11 @@ where // append transaction to the list of executed transactions transactions.push(tx); + + #[cfg(feature = "telos")] + { + tx_index += 1; + } } drop(evm); @@ -412,6 +425,12 @@ where blob_gas_used, excess_blob_gas, state_root: state_provider.state_root(hashed_state)?, + #[cfg(feature = "telos")] + telos_block_extension: TelosBlockExtension::from_parent_and_changes( + &reorg_target_parent.header.telos_block_extension, + telos_extra_fields.clone().unwrap_or_default().gasprice_changes, + telos_extra_fields.clone().unwrap_or_default().revision_changes, + ), }, body: transactions, ommers: reorg_target.ommers, diff --git a/crates/ethereum/engine-primitives/Cargo.toml b/crates/ethereum/engine-primitives/Cargo.toml index 8a1f25808937..09d765d18ef4 100644 --- a/crates/ethereum/engine-primitives/Cargo.toml +++ b/crates/ethereum/engine-primitives/Cargo.toml @@ -28,3 +28,6 @@ sha2.workspace = true [dev-dependencies] serde_json.workspace = true + +[features] +telos = [] diff --git a/crates/ethereum/evm/Cargo.toml b/crates/ethereum/evm/Cargo.toml index cdd0b9a3f29b..5697871cdf1c 100644 --- a/crates/ethereum/evm/Cargo.toml +++ b/crates/ethereum/evm/Cargo.toml @@ -30,6 +30,7 @@ alloy-sol-types.workspace = true # telos reth-telos-rpc-engine-api = { workspace = true, optional = true } +reth-telos-primitives-traits = { workspace = true, optional = true } [dev-dependencies] reth-testing-utils.workspace = true @@ -43,5 +44,6 @@ default = ["std"] std = [] telos = [ "dep:reth-telos-rpc-engine-api", + "dep:reth-telos-primitives-traits", "reth-ethereum-consensus/telos", ] \ No newline at end of file diff --git a/crates/ethereum/evm/src/execute.rs b/crates/ethereum/evm/src/execute.rs index 6983c0f98b19..896be9f1d5e1 100644 --- a/crates/ethereum/evm/src/execute.rs +++ b/crates/ethereum/evm/src/execute.rs @@ -207,7 +207,7 @@ where .into()) } - self.evm_config.fill_tx_env(evm.tx_mut(), transaction, *sender); + self.evm_config.fill_tx_env(evm.tx_mut(), transaction, *sender, #[cfg(feature = "telos")] block.header.telos_block_extension.tx_env_at(tx_index)); // Execute transaction. let ResultAndState { result, state } = evm.transact().map_err(move |err| { diff --git a/crates/ethereum/evm/src/lib.rs b/crates/ethereum/evm/src/lib.rs index bead8ae3923c..52c369695668 100644 --- a/crates/ethereum/evm/src/lib.rs +++ b/crates/ethereum/evm/src/lib.rs @@ -22,6 +22,7 @@ use alloc::vec::Vec; mod config; pub use config::{revm_spec, revm_spec_by_timestamp_after_merge}; +use reth_telos_primitives_traits::TelosTxEnv; pub mod execute; @@ -61,8 +62,8 @@ impl ConfigureEvmEnv for EthEvmConfig { cfg_env.handler_cfg.spec_id = spec_id; } - fn fill_tx_env(&self, tx_env: &mut TxEnv, transaction: &TransactionSigned, sender: Address) { - transaction.fill_tx_env(tx_env, sender); + fn fill_tx_env(&self, tx_env: &mut TxEnv, transaction: &TransactionSigned, sender: Address, #[cfg(feature = "telos")] telos_tx_env: TelosTxEnv) { + transaction.fill_tx_env(tx_env, sender, #[cfg(feature = "telos")] telos_tx_env); } fn fill_tx_env_system_contract_call( diff --git a/crates/ethereum/payload/Cargo.toml b/crates/ethereum/payload/Cargo.toml index 39dfc23613af..54d52e485966 100644 --- a/crates/ethereum/payload/Cargo.toml +++ b/crates/ethereum/payload/Cargo.toml @@ -31,8 +31,12 @@ revm.workspace = true # misc tracing.workspace = true +# telos +reth-telos-primitives-traits = { workspace = true, optional = true } + [features] telos = [ "revm/telos", "reth-evm-ethereum/telos", + "dep:reth-telos-primitives-traits", ] diff --git a/crates/ethereum/payload/src/lib.rs b/crates/ethereum/payload/src/lib.rs index 2e9a2fdaeda9..47b5d4b1da50 100644 --- a/crates/ethereum/payload/src/lib.rs +++ b/crates/ethereum/payload/src/lib.rs @@ -46,6 +46,7 @@ use revm::{ DatabaseCommit, State, }; use tracing::{debug, trace, warn}; +use reth_telos_primitives_traits::TelosTxEnv; /// Ethereum payload builder #[derive(Debug, Clone, Copy, PartialEq, Eq)] @@ -249,6 +250,9 @@ where excess_blob_gas, parent_beacon_block_root: attributes.parent_beacon_block_root, requests_root, + #[cfg(feature = "telos")] + // Ok to use Default here because Telos will never build a payload in reth + telos_block_extension: Default::default(), }; let block = Block { header, body: vec![], ommers: vec![], withdrawals, requests }; @@ -376,7 +380,8 @@ where let env = EnvWithHandlerCfg::new_with_cfg_env( initialized_cfg.clone(), initialized_block_env.clone(), - evm_config.tx_env(&tx), + // Telos will never build payloads here, so using Default below is OK + evm_config.tx_env(&tx, #[cfg(feature = "telos")] TelosTxEnv::default()), ); // Configure the environment for the block. @@ -556,6 +561,9 @@ where blob_gas_used, excess_blob_gas, requests_root, + #[cfg(feature = "telos")] + // Ok to use Default here because Telos will never build a payload in reth + telos_block_extension: Default::default(), }; // seal the block diff --git a/crates/evm/Cargo.toml b/crates/evm/Cargo.toml index 054c7085345b..c8c7e2efba37 100644 --- a/crates/evm/Cargo.toml +++ b/crates/evm/Cargo.toml @@ -27,6 +27,7 @@ futures-util.workspace = true parking_lot = { workspace = true, optional = true } reth-telos-rpc-engine-api = { workspace = true, optional = true } +reth-telos-primitives-traits = { workspace = true, optional = true } [dev-dependencies] parking_lot.workspace = true @@ -38,4 +39,5 @@ test-utils = ["dep:parking_lot"] telos = [ "revm/telos", "dep:reth-telos-rpc-engine-api", + "dep:reth-telos-primitives-traits", ] diff --git a/crates/evm/src/lib.rs b/crates/evm/src/lib.rs index 7b52a8accfc7..b402341558e8 100644 --- a/crates/evm/src/lib.rs +++ b/crates/evm/src/lib.rs @@ -21,6 +21,8 @@ use revm::{Database, Evm, GetInspector}; use revm_primitives::{ BlockEnv, Bytes, CfgEnvWithHandlerCfg, Env, EnvWithHandlerCfg, SpecId, TxEnv, }; +#[cfg(feature = "telos")] +use reth_telos_primitives_traits::TelosTxEnv; pub mod builder; pub mod either; @@ -109,14 +111,14 @@ pub trait ConfigureEvm: ConfigureEvmEnv { #[auto_impl::auto_impl(&, Arc)] pub trait ConfigureEvmEnv: Send + Sync + Unpin + Clone + 'static { /// Returns a [`TxEnv`] from a [`TransactionSignedEcRecovered`]. - fn tx_env(&self, transaction: &TransactionSignedEcRecovered) -> TxEnv { + fn tx_env(&self, transaction: &TransactionSignedEcRecovered, #[cfg(feature = "telos")] telos_tx_env: TelosTxEnv) -> TxEnv { let mut tx_env = TxEnv::default(); - self.fill_tx_env(&mut tx_env, transaction.deref(), transaction.signer()); + self.fill_tx_env(&mut tx_env, transaction.deref(), transaction.signer(), #[cfg(feature = "telos")] telos_tx_env); tx_env } /// Fill transaction environment from a [`TransactionSigned`] and the given sender address. - fn fill_tx_env(&self, tx_env: &mut TxEnv, transaction: &TransactionSigned, sender: Address); + fn fill_tx_env(&self, tx_env: &mut TxEnv, transaction: &TransactionSigned, sender: Address, #[cfg(feature = "telos")] telos_tx_env: TelosTxEnv); /// Fill transaction environment with a system contract call. fn fill_tx_env_system_contract_call( diff --git a/crates/node/builder/Cargo.toml b/crates/node/builder/Cargo.toml index df6aa0fb8ecd..7036ee41735a 100644 --- a/crates/node/builder/Cargo.toml +++ b/crates/node/builder/Cargo.toml @@ -95,4 +95,5 @@ telos = [ "reth-rpc-engine-api/telos", "reth-engine-tree/telos", "reth-auto-seal-consensus/telos", + "reth-payload-validator/telos", ] diff --git a/crates/payload/validator/Cargo.toml b/crates/payload/validator/Cargo.toml index 66efe865ceba..f9b21afba3e3 100644 --- a/crates/payload/validator/Cargo.toml +++ b/crates/payload/validator/Cargo.toml @@ -17,3 +17,9 @@ reth-chainspec.workspace = true reth-primitives.workspace = true reth-rpc-types.workspace = true reth-rpc-types-compat.workspace = true + +# telos +reth-telos-rpc-engine-api = { workspace = true, optional = true } + +[features] +telos = ["dep:reth-telos-rpc-engine-api"] \ No newline at end of file diff --git a/crates/payload/validator/src/lib.rs b/crates/payload/validator/src/lib.rs index 6e21111e62f4..6c8a18a0f95e 100644 --- a/crates/payload/validator/src/lib.rs +++ b/crates/payload/validator/src/lib.rs @@ -13,6 +13,7 @@ use reth_primitives::SealedBlock; use reth_rpc_types::{engine::MaybeCancunPayloadFields, ExecutionPayload, PayloadError}; use reth_rpc_types_compat::engine::payload::try_into_block; use std::sync::Arc; +use reth_telos_rpc_engine_api::structs::TelosEngineAPIExtraFields; /// Execution payload validator. #[derive(Clone, Debug)] @@ -112,6 +113,8 @@ impl ExecutionPayloadValidator { &self, payload: ExecutionPayload, cancun_fields: MaybeCancunPayloadFields, + #[cfg(feature = "telos")] + telos_extra_fields: TelosEngineAPIExtraFields, ) -> Result { let expected_hash = payload.block_hash(); diff --git a/crates/primitives-traits/Cargo.toml b/crates/primitives-traits/Cargo.toml index 06802d5adbd0..33ad9d536ee3 100644 --- a/crates/primitives-traits/Cargo.toml +++ b/crates/primitives-traits/Cargo.toml @@ -38,6 +38,9 @@ arbitrary = { workspace = true, features = ["derive"], optional = true } proptest = { workspace = true, optional = true } proptest-arbitrary-interop = { workspace = true, optional = true } +# telos +reth-telos-primitives-traits = { workspace = true, optional = true } + [dev-dependencies] alloy-primitives = { workspace = true, features = ["arbitrary"] } alloy-consensus = { workspace = true, features = ["arbitrary"] } @@ -59,6 +62,9 @@ arbitrary = [ "dep:arbitrary", "dep:proptest", "dep:proptest-arbitrary-interop", + "reth-telos-primitives-traits/arbitrary", ] alloy-compat = ["alloy-rpc-types-eth"] -telos = [] \ No newline at end of file +telos = [ + "dep:reth-telos-primitives-traits", +] \ No newline at end of file diff --git a/crates/primitives-traits/src/header/mod.rs b/crates/primitives-traits/src/header/mod.rs index 4a950c563f78..336634b195c0 100644 --- a/crates/primitives-traits/src/header/mod.rs +++ b/crates/primitives-traits/src/header/mod.rs @@ -19,6 +19,8 @@ use core::mem; use reth_codecs::{add_arbitrary_tests, Compact}; use revm_primitives::{calc_blob_gasprice, calc_excess_blob_gas}; use serde::{Deserialize, Serialize}; +#[cfg(feature = "telos")] +use reth_telos_primitives_traits::{GasPrice, Revision, TelosBlockExtension}; /// Block header #[derive(Debug, Clone, PartialEq, Eq, Hash, Serialize, Deserialize, Compact)] @@ -96,6 +98,9 @@ pub struct Header { /// /// [EIP-7685]: https://eips.ethereum.org/EIPS/eip-7685 pub requests_root: Option, + #[cfg(feature = "telos")] + /// Telos specific block fields, to be stored in the database but not used for block hash + pub telos_block_extension: TelosBlockExtension, /// An arbitrary byte array containing data relevant to this block. This must be 32 bytes or /// fewer; formally Hx. pub extra_data: Bytes, @@ -131,6 +136,8 @@ impl Default for Header { excess_blob_gas: None, parent_beacon_block_root: None, requests_root: None, + #[cfg(feature = "telos")] + telos_block_extension: Default::default(), } } } @@ -354,6 +361,52 @@ impl Header { length } + + #[cfg(feature = "telos")] + /// Creates a clone of the header, adding Telos block extension fields + pub fn clone_with_telos(&self, + parent_telos_extension: TelosBlockExtension, + gas_price_change: Option<(u64, U256)>, + revision_change: Option<(u64, u64)>) -> Self { + let gas_price_change = if let Some(gas_price_change) = gas_price_change { + Some(gas_price_change) + } else { + None + }; + + let revision_change = if let Some(revision_change) = revision_change { + Some(revision_change) + } else { + None + }; + + Self { + parent_hash: self.parent_hash, + ommers_hash: self.ommers_hash, + beneficiary: self.beneficiary, + state_root: self.state_root, + transactions_root: self.transactions_root, + receipts_root: self.receipts_root, + logs_bloom: self.logs_bloom, + difficulty: self.difficulty, + number: self.number, + gas_limit: self.gas_limit, + gas_used: self.gas_used, + timestamp: self.timestamp, + extra_data: self.extra_data.clone(), + mix_hash: self.mix_hash, + nonce: self.nonce, + base_fee_per_gas: self.base_fee_per_gas, + withdrawals_root: self.withdrawals_root, + blob_gas_used: self.blob_gas_used, + excess_blob_gas: self.excess_blob_gas, + parent_beacon_block_root: self.parent_beacon_block_root, + requests_root: self.requests_root, + telos_block_extension: TelosBlockExtension::from_parent_and_changes( + &parent_telos_extension, gas_price_change, revision_change + ), + } + } } impl Encodable for Header { @@ -449,6 +502,8 @@ impl Decodable for Header { excess_blob_gas: None, parent_beacon_block_root: None, requests_root: None, + #[cfg(feature = "telos")] + telos_block_extension: Default::default(), }; if started_len - buf.len() < rlp_head.payload_length { this.base_fee_per_gas = Some(u64::decode(buf)?); @@ -516,6 +571,8 @@ impl<'a> arbitrary::Arbitrary<'a> for Header { parent_beacon_block_root: u.arbitrary()?, requests_root: u.arbitrary()?, withdrawals_root: u.arbitrary()?, + #[cfg(feature = "telos")] + telos_block_extension: u.arbitrary()?, }; Ok(test_utils::generate_valid_header( diff --git a/crates/primitives/Cargo.toml b/crates/primitives/Cargo.toml index 3ffb4f94fa11..fa884f1c9ec6 100644 --- a/crates/primitives/Cargo.toml +++ b/crates/primitives/Cargo.toml @@ -59,6 +59,8 @@ zstd = { workspace = true, features = ["experimental"], optional = true } arbitrary = { workspace = true, features = ["derive"], optional = true } proptest = { workspace = true, optional = true } +reth-telos-primitives-traits = { workspace = true, optional = true } + [dev-dependencies] # eth reth-primitives-traits = { workspace = true, features = ["arbitrary"] } @@ -111,6 +113,8 @@ alloy-compat = ["reth-primitives-traits/alloy-compat", "dep:alloy-rpc-types", "d test-utils = ["reth-primitives-traits/test-utils"] telos = [ "reth-chainspec/telos", + "reth-codecs/telos", + "dep:reth-telos-primitives-traits", "reth-primitives-traits/telos", "revm-primitives/telos", ] diff --git a/crates/primitives/src/transaction/compat.rs b/crates/primitives/src/transaction/compat.rs index 2cd1f0c82590..5330113e3894 100644 --- a/crates/primitives/src/transaction/compat.rs +++ b/crates/primitives/src/transaction/compat.rs @@ -3,15 +3,16 @@ use revm_primitives::{AuthorizationList, TxEnv}; #[cfg(all(not(feature = "std"), feature = "optimism"))] use alloc::vec::Vec; +use reth_telos_primitives_traits::TelosTxEnv; /// Implements behaviour to fill a [`TxEnv`] from another transaction. pub trait FillTxEnv { /// Fills [`TxEnv`] with an [`Address`] and transaction. - fn fill_tx_env(&self, tx_env: &mut TxEnv, sender: Address); + fn fill_tx_env(&self, tx_env: &mut TxEnv, sender: Address, #[cfg(feature = "telos")] telos_tx_env: TelosTxEnv); } impl FillTxEnv for TransactionSigned { - fn fill_tx_env(&self, tx_env: &mut TxEnv, sender: Address) { +fn fill_tx_env(&self, tx_env: &mut TxEnv, sender: Address, #[cfg(feature = "telos")] telos_tx_env: TelosTxEnv) { #[cfg(feature = "optimism")] let envelope = { let mut envelope = Vec::with_capacity(self.length_without_header()); @@ -19,6 +20,10 @@ impl FillTxEnv for TransactionSigned { envelope }; + #[cfg(feature = "telos")] { + tx_env.fixed_gas_price = telos_tx_env.gas_price; + tx_env.revision_number = telos_tx_env.revision; + } #[cfg(feature = "telos")] { tx_env.first_new_address = None; } diff --git a/crates/rpc/rpc-eth-api/Cargo.toml b/crates/rpc/rpc-eth-api/Cargo.toml index 5c18bfef46ce..d2cdf83abc92 100644 --- a/crates/rpc/rpc-eth-api/Cargo.toml +++ b/crates/rpc/rpc-eth-api/Cargo.toml @@ -52,6 +52,10 @@ auto_impl.workspace = true dyn-clone.workspace = true tracing.workspace = true +# telos +reth-telos-rpc-engine-api = { workspace = true, optional = true } +reth-telos-primitives-traits = { workspace = true, optional = true } + [features] js-tracer = ["revm-inspectors/js-tracer", "reth-rpc-eth-types/js-tracer"] client = ["jsonrpsee/client", "jsonrpsee/async-client"] @@ -59,4 +63,8 @@ optimism = [ "reth-primitives/optimism", "revm/optimism", "reth-provider/optimism", +] +telos = [ + "dep:reth-telos-rpc-engine-api", + "dep:reth-telos-primitives-traits", ] \ No newline at end of file diff --git a/crates/rpc/rpc-eth-api/src/helpers/call.rs b/crates/rpc/rpc-eth-api/src/helpers/call.rs index 0d309e6e0863..6d9b619c9e6d 100644 --- a/crates/rpc/rpc-eth-api/src/helpers/call.rs +++ b/crates/rpc/rpc-eth-api/src/helpers/call.rs @@ -34,8 +34,11 @@ use reth_rpc_types::{ use revm::{Database, DatabaseCommit}; use revm_inspectors::access_list::AccessListInspector; use tracing::trace; - +use reth_telos_primitives_traits::{TelosBlockExtension, TelosTxEnv}; use super::{LoadBlock, LoadPendingBlock, LoadState, LoadTransaction, SpawnBlocking, Trace}; +#[cfg(feature = "telos")] +use reth_provider::BlockIdReader; + /// Execution related functions for the [`EthApiServer`](crate::EthApiServer) trait in /// the `eth_` namespace. @@ -131,6 +134,11 @@ pub trait EthCall: Call + LoadPendingBlock { let mut results = Vec::with_capacity(transactions.len()); let mut db = CacheDB::new(StateProviderDatabase::new(state)); + #[cfg(feature = "telos")] + let telos_extension = block.header.telos_block_extension.clone(); + #[cfg(feature = "telos")] + let mut tx_index = 0; + if replay_block_txs { // only need to replay the transactions in the block if not all transactions are // to be replayed @@ -139,10 +147,13 @@ pub trait EthCall: Call + LoadPendingBlock { let env = EnvWithHandlerCfg::new_with_cfg_env( cfg.clone(), block_env.clone(), - Call::evm_config(&this).tx_env(&tx), + Call::evm_config(&this).tx_env(&tx, #[cfg(feature = "telos")] telos_extension.tx_env_at(tx_index)), ); let (res, _) = this.transact(&mut db, env)?; db.commit(res.state); + #[cfg(feature = "telos")] { + tx_index += 1; + } } } @@ -162,6 +173,8 @@ pub trait EthCall: Call + LoadPendingBlock { gas_limit, &mut db, overrides, + #[cfg(feature = "telos")] + telos_extension.tx_env_at(tx_index), ) .map(Into::into)?; let (res, _) = this.transact(&mut db, env)?; @@ -183,6 +196,9 @@ pub trait EthCall: Call + LoadPendingBlock { // call db.commit(res.state); } + #[cfg(feature = "telos")] { + tx_index += 1; + } } Ok(results) @@ -205,8 +221,11 @@ pub trait EthCall: Call + LoadPendingBlock { let block_id = block_number.unwrap_or_default(); let (cfg, block, at) = self.evm_env_at(block_id).await?; + #[cfg(feature = "telos")] + let telos_tx_env = self.telos_tx_env_at(at).await?; + self.spawn_blocking_io(move |this| { - this.create_access_list_with(cfg, block, at, request) + this.create_access_list_with(cfg, block, at, request, #[cfg(feature = "telos")] telos_tx_env) }) .await } @@ -220,13 +239,15 @@ pub trait EthCall: Call + LoadPendingBlock { block: BlockEnv, at: BlockId, mut request: TransactionRequest, + #[cfg(feature = "telos")] + telos_tx_env: TelosTxEnv, ) -> Result where Self: Trace, { let state = self.state_at_block_id(at)?; - let mut env = self.build_call_evm_env(cfg, block, request.clone())?; + let mut env = self.build_call_evm_env(cfg, block, request.clone(), #[cfg(feature = "telos")] telos_tx_env)?; // we want to disable this in eth_createAccessList, since this is common practice used by // other node impls and providers @@ -334,6 +355,30 @@ pub trait Call: LoadState + SpawnBlocking { Ok((res, env)) } + #[cfg(feature = "telos")] + /// Fetch a TelosTxEnv for a given block + fn telos_tx_env_at(&self, at: BlockId) -> impl Future> + Send + where + Self: LoadPendingBlock, + { + async move { + let block_hash = LoadPendingBlock::provider(self) + .block_hash_for_id(at) + .map_err(Self::Error::from_eth_err)? + .ok_or_else(|| EthApiError::UnknownBlockNumber)?; + let block_at_result = self.cache().get_block(block_hash).await.map_err(Self::Error::from_eth_err)?; + let block_at = block_at_result.ok_or_else(|| { + EthApiError::UnknownBlockNumber + })?; + let parent_block_result = self.cache().get_block(block_at.parent_hash).await; + let parent_block = parent_block_result.unwrap_or(None).ok_or_else( + || EthApiError::UnknownBlockNumber, + )?; + + Ok(parent_block.header.telos_block_extension.tx_env_at(block_at.body.len() as u64)) + } + } + /// Executes the call request at the given [`BlockId`]. fn transact_call_at( &self, @@ -385,6 +430,9 @@ pub trait Call: LoadState + SpawnBlocking { { async move { let (cfg, block_env, at) = self.evm_env_at(at).await?; + #[cfg(feature = "telos")] + let telos_tx_env = self.telos_tx_env_at(at).await?; + let this = self.clone(); self.spawn_tracing(move |_| { let state = this.state_at_block_id(at)?; @@ -398,6 +446,8 @@ pub trait Call: LoadState + SpawnBlocking { this.call_gas_limit(), &mut db, overrides, + #[cfg(feature = "telos")] + telos_tx_env, )?; f(StateCacheDbRefMutWrapper(&mut db), env) @@ -439,12 +489,15 @@ pub trait Call: LoadState + SpawnBlocking { // we need to get the state of the parent block because we're essentially replaying the // block the transaction is included in let parent_block = block.parent_hash; + #[cfg(feature = "telos")] + let telos_block_extension = block.header.telos_block_extension.clone(); let block_txs = block.into_transactions_ecrecovered(); let this = self.clone(); self.spawn_with_state_at_block(parent_block.into(), move |state| { let mut db = CacheDB::new(StateProviderDatabase::new(state)); + #[cfg(not(feature = "telos"))] // replay all transactions prior to the targeted transaction this.replay_transactions_until( &mut db, @@ -454,10 +507,20 @@ pub trait Call: LoadState + SpawnBlocking { tx.hash, )?; + #[cfg(feature = "telos")] + let index = this.replay_transactions_until( + &mut db, + cfg.clone(), + block_env.clone(), + block_txs, + tx.hash, + &telos_block_extension, + )?; + let env = EnvWithHandlerCfg::new_with_cfg_env( cfg, block_env, - Call::evm_config(&this).tx_env(&tx), + Call::evm_config(&this).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(index as u64)), ); let (res, _) = this.transact(&mut db, env)?; @@ -482,6 +545,8 @@ pub trait Call: LoadState + SpawnBlocking { block_env: BlockEnv, transactions: impl IntoIterator, target_tx_hash: B256, + #[cfg(feature = "telos")] + telos_extension: &TelosBlockExtension, ) -> Result where DB: DatabaseRef, @@ -498,7 +563,7 @@ pub trait Call: LoadState + SpawnBlocking { } let sender = tx.signer(); - self.evm_config().fill_tx_env(evm.tx_mut(), &tx.into_signed(), sender); + self.evm_config().fill_tx_env(evm.tx_mut(), &tx.into_signed(), sender, #[cfg(feature = "telos")] telos_extension.tx_env_at(index as u64)); evm.transact_commit().map_err(Self::Error::from_evm_err)?; index += 1; } @@ -518,9 +583,12 @@ pub trait Call: LoadState + SpawnBlocking { async move { let (cfg, block_env, at) = self.evm_env_at(at).await?; + #[cfg(feature = "telos")] + let telos_tx_env = self.telos_tx_env_at(at).await?; + self.spawn_blocking_io(move |this| { let state = this.state_at_block_id(at)?; - this.estimate_gas_with(cfg, block_env, request, state, state_override) + this.estimate_gas_with(cfg, block_env, request, state, state_override, #[cfg(feature = "telos")] telos_tx_env) }) .await } @@ -544,6 +612,8 @@ pub trait Call: LoadState + SpawnBlocking { mut request: TransactionRequest, state: S, state_override: Option, + #[cfg(feature = "telos")] + telos_tx_env: TelosTxEnv, ) -> Result where S: StateProvider, @@ -572,7 +642,7 @@ pub trait Call: LoadState + SpawnBlocking { .unwrap_or(block_env_gas_limit); // Configure the evm env - let mut env = self.build_call_evm_env(cfg, block, request)?; + let mut env = self.build_call_evm_env(cfg, block, request, #[cfg(feature = "telos")] telos_tx_env)?; let mut db = CacheDB::new(StateProviderDatabase::new(state)); // Apply any state overrides if specified. @@ -833,6 +903,8 @@ pub trait Call: LoadState + SpawnBlocking { &self, block_env: &BlockEnv, request: TransactionRequest, + #[cfg(feature = "telos")] + telos_tx_env: TelosTxEnv ) -> Result { // Ensure that if versioned hashes are set, they're not empty if request.blob_versioned_hashes.as_ref().map_or(false, |hashes| hashes.is_empty()) { @@ -876,6 +948,12 @@ pub trait Call: LoadState + SpawnBlocking { .try_into() .map_err(|_| RpcInvalidTransactionError::GasUintOverflow) .map_err(Self::Error::from_eth_err)?, + #[cfg(feature = "telos")] + first_new_address: None, + #[cfg(feature = "telos")] + revision_number: telos_tx_env.revision, + #[cfg(feature = "telos")] + fixed_gas_price: telos_tx_env.gas_price, nonce, caller: from.unwrap_or_default(), gas_price, @@ -908,8 +986,10 @@ pub trait Call: LoadState + SpawnBlocking { cfg: CfgEnvWithHandlerCfg, block: BlockEnv, request: TransactionRequest, + #[cfg(feature = "telos")] + telos_tx_env: TelosTxEnv ) -> Result { - let tx = self.create_txn_env(&block, request)?; + let tx = self.create_txn_env(&block, request, #[cfg(feature = "telos")] telos_tx_env)?; Ok(EnvWithHandlerCfg::new_with_cfg_env(cfg, block, tx)) } @@ -935,6 +1015,8 @@ pub trait Call: LoadState + SpawnBlocking { gas_limit: u64, db: &mut CacheDB, overrides: EvmOverrides, + #[cfg(feature = "telos")] + telos_tx_env: TelosTxEnv, ) -> Result where DB: DatabaseRef, @@ -968,7 +1050,7 @@ pub trait Call: LoadState + SpawnBlocking { } let request_gas = request.gas; - let mut env = self.build_call_evm_env(cfg, block, request)?; + let mut env = self.build_call_evm_env(cfg, block, request, #[cfg(feature = "telos")] telos_tx_env)?; // apply state overrides if let Some(state_overrides) = overrides.state { diff --git a/crates/rpc/rpc-eth-api/src/helpers/fee.rs b/crates/rpc/rpc-eth-api/src/helpers/fee.rs index a8fbc7e7fa73..874d6cd96fb2 100644 --- a/crates/rpc/rpc-eth-api/src/helpers/fee.rs +++ b/crates/rpc/rpc-eth-api/src/helpers/fee.rs @@ -333,7 +333,10 @@ pub trait LoadFee: LoadBlock { let suggested_tip = self.suggested_priority_fee(); async move { let (header, suggested_tip) = futures::try_join!(header, suggested_tip)?; - let base_fee = header.and_then(|h| h.base_fee_per_gas).unwrap_or_default(); + #[cfg(not(feature = "telos"))] + let base_fee = header.clone().and_then(|h| h.base_fee_per_gas).unwrap_or_default(); + #[cfg(feature = "telos")] + let base_fee = header.and_then(|h| Some(h.telos_block_extension.get_last_gas_price())).unwrap_or_default(); Ok(suggested_tip + U256::from(base_fee)) } } diff --git a/crates/rpc/rpc-eth-api/src/helpers/pending_block.rs b/crates/rpc/rpc-eth-api/src/helpers/pending_block.rs index bcbbf576a763..bc6327250ef0 100644 --- a/crates/rpc/rpc-eth-api/src/helpers/pending_block.rs +++ b/crates/rpc/rpc-eth-api/src/helpers/pending_block.rs @@ -35,7 +35,7 @@ use reth_trie::HashedPostState; use revm::{db::states::bundle_state::BundleRetention, DatabaseCommit, State}; use tokio::sync::Mutex; use tracing::debug; - +use reth_telos_primitives_traits::TelosTxEnv; use super::SpawnBlocking; /// Loads a pending block from database. @@ -328,7 +328,8 @@ pub trait LoadPendingBlock: EthApiTypes { let env = Env::boxed( cfg.cfg_env.clone(), block_env.clone(), - Self::evm_config(self).tx_env(&tx), + // Telos will never build blocks in reth, using Default below is ok + Self::evm_config(self).tx_env(&tx, #[cfg(feature = "telos")] TelosTxEnv::default()), ); let mut evm = revm::Evm::builder().with_env(env).with_db(&mut db).build(); @@ -453,6 +454,9 @@ pub trait LoadPendingBlock: EthApiTypes { extra_data: Default::default(), parent_beacon_block_root, requests_root, + #[cfg(feature = "telos")] + // Ok to use Default here, as Telos will never build a block in reth + telos_block_extension: Default::default(), }; // Convert Vec> to Vec diff --git a/crates/rpc/rpc-eth-api/src/helpers/trace.rs b/crates/rpc/rpc-eth-api/src/helpers/trace.rs index 09ad7f22fa21..481d0a79d85e 100644 --- a/crates/rpc/rpc-eth-api/src/helpers/trace.rs +++ b/crates/rpc/rpc-eth-api/src/helpers/trace.rs @@ -186,6 +186,9 @@ pub trait Trace: LoadState { let (cfg, block_env, _) = self.evm_env_at(block.hash().into()).await?; + #[cfg(feature = "telos")] + let telos_block_extension = block.header.telos_block_extension.clone(); + // we need to get the state of the parent block because we're essentially replaying the // block the transaction is included in let parent_block = block.parent_hash; @@ -195,6 +198,7 @@ pub trait Trace: LoadState { self.spawn_with_state_at_block(parent_block.into(), move |state| { let mut db = CacheDB::new(StateProviderDatabase::new(state)); + #[cfg(not(feature = "telos"))] // replay all transactions prior to the targeted transaction this.replay_transactions_until( &mut db, @@ -204,10 +208,21 @@ pub trait Trace: LoadState { tx.hash, )?; + #[cfg(feature = "telos")] + let tx_index = this.replay_transactions_until( + &mut db, + cfg.clone(), + block_env.clone(), + block_txs, + tx.hash, + #[cfg(feature = "telos")] + &telos_block_extension, + )?; + let env = EnvWithHandlerCfg::new_with_cfg_env( cfg, block_env, - Call::evm_config(&this).tx_env(&tx), + Call::evm_config(&this).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(tx_index as u64)), ); let (res, _) = this.inspect(StateCacheDbRefMutWrapper(&mut db), env, &mut inspector)?; @@ -295,6 +310,8 @@ pub trait Trace: LoadState { return Ok(Some(Vec::new())) } + #[cfg(feature = "telos")] + let telos_block_extension = block.header.telos_block_extension.clone(); // replay all transactions of the block self.spawn_tracing(move |this| { // we need to get the state of the parent block because we're replaying this block @@ -325,7 +342,7 @@ pub trait Trace: LoadState { block_number: Some(block_number), base_fee: Some(base_fee), }; - let tx_env = Trace::evm_config(&this).tx_env(&tx); + let tx_env = Trace::evm_config(&this).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(idx as u64)); (tx_info, tx_env) }) .peekable(); diff --git a/crates/rpc/rpc-types-compat/Cargo.toml b/crates/rpc/rpc-types-compat/Cargo.toml index 1dfb7d87db50..564fa9fe15c9 100644 --- a/crates/rpc/rpc-types-compat/Cargo.toml +++ b/crates/rpc/rpc-types-compat/Cargo.toml @@ -18,9 +18,12 @@ reth-trie-common.workspace = true alloy-rlp.workspace = true alloy-rpc-types.workspace = true +# telos +reth-telos-primitives-traits = { workspace = true, optional = true } + [dev-dependencies] serde_json.workspace = true [features] optimism = ["reth-primitives/optimism"] -telos = [] +telos = ["dep:reth-telos-primitives-traits"] diff --git a/crates/rpc/rpc-types-compat/src/block.rs b/crates/rpc/rpc-types-compat/src/block.rs index 3b3c22c83b28..0186c44fa3b5 100644 --- a/crates/rpc/rpc-types-compat/src/block.rs +++ b/crates/rpc/rpc-types-compat/src/block.rs @@ -124,6 +124,8 @@ pub fn from_primitive_with_hash(primitive_header: reth_primitives::SealedHeader) excess_blob_gas, parent_beacon_block_root, requests_root, + #[cfg(feature = "telos")] + .. } = header; Header { diff --git a/crates/rpc/rpc-types-compat/src/engine/payload.rs b/crates/rpc/rpc-types-compat/src/engine/payload.rs index 435270b39f58..ed20e951c156 100644 --- a/crates/rpc/rpc-types-compat/src/engine/payload.rs +++ b/crates/rpc/rpc-types-compat/src/engine/payload.rs @@ -13,6 +13,7 @@ use reth_rpc_types::engine::{ ExecutionPayload, ExecutionPayloadBodyV2, ExecutionPayloadV1, ExecutionPayloadV2, ExecutionPayloadV3, ExecutionPayloadV4, PayloadError, }; +use reth_telos_primitives_traits::TelosBlockExtension; /// Converts [`ExecutionPayloadV1`] to [`Block`] pub fn try_payload_v1_to_block(payload: ExecutionPayloadV1) -> Result { @@ -66,6 +67,12 @@ pub fn try_payload_v1_to_block(payload: ExecutionPayloadV1) -> Result Result, Eth::Error> { if transactions.is_empty() { // nothing to trace @@ -98,6 +101,7 @@ where // replay all transactions of the block let this = self.clone(); + let telos_block_extension = telos_block_extension.clone(); self.eth_api() .spawn_with_state_at_block(at, move |state| { let block_hash = at.as_block_hash(); @@ -111,7 +115,7 @@ where env: Env::boxed( cfg.cfg_env.clone(), block_env.clone(), - Call::evm_config(this.eth_api()).tx_env(&tx), + Call::evm_config(this.eth_api()).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(index as u64)), ), handler_cfg: cfg.handler_cfg, }; @@ -157,6 +161,13 @@ where // we trace on top the block's parent block let parent = block.parent_hash; + #[cfg(feature = "telos")] + let telos_block_extension = self + .inner.provider + .block_by_hash(parent.into()) + .unwrap().unwrap() // TODO: this better + .header.telos_block_extension.to_child(); + // Depending on EIP-2 we need to recover the transactions differently let transactions = if self.inner.provider.chain_spec().is_homestead_active_at_block(block.number) { @@ -181,7 +192,7 @@ where .collect::, Eth::Error>>()? }; - self.trace_block(parent.into(), transactions, cfg, block_env, opts).await + self.trace_block(parent.into(), transactions, cfg, block_env, opts, #[cfg(feature = "telos")] &telos_block_extension).await } /// Replays a block and returns the trace of each transaction. @@ -206,6 +217,7 @@ where // we need to get the state of the parent block because we're replaying this block on top of // its parent block's state let state_at = block.parent_hash; + let telos_block_extension = &block.header.telos_block_extension.clone(); self.trace_block( state_at.into(), @@ -213,6 +225,8 @@ where cfg, block_env, opts, + #[cfg(feature = "telos")] + telos_block_extension ) .await } @@ -229,6 +243,7 @@ where None => return Err(EthApiError::TransactionNotFound.into()), Some(res) => res, }; + let telos_block_extension = block.header.telos_block_extension.clone(); let (cfg, block_env, _) = self.inner.eth_api.evm_env_at(block.hash().into()).await?; // we need to get the state of the parent block because we're essentially replaying the @@ -252,13 +267,15 @@ where block_env.clone(), block_txs, tx.hash, + #[cfg(feature = "telos")] + &telos_block_extension )?; let env = EnvWithHandlerCfg { env: Env::boxed( cfg.cfg_env.clone(), block_env, - Call::evm_config(this.eth_api()).tx_env(&tx), + Call::evm_config(this.eth_api()).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(index as u64)), ), handler_cfg: cfg.handler_cfg, }; @@ -469,6 +486,7 @@ where let opts = opts.unwrap_or_default(); let block = block.ok_or_else(|| EthApiError::UnknownBlockNumber)?; + let telos_block_extension = block.header.telos_block_extension.clone(); let GethDebugTracingCallOptions { tracing_options, mut state_overrides, .. } = opts; let gas_limit = self.inner.eth_api.call_gas_limit(); @@ -496,6 +514,9 @@ where let mut all_bundles = Vec::with_capacity(bundles.len()); let mut db = CacheDB::new(StateProviderDatabase::new(state)); + #[cfg(feature = "telos")] + let mut tx_index = 0; + if replay_block_txs { // only need to replay the transactions in the block if not all transactions are // to be replayed @@ -507,12 +528,16 @@ where env: Env::boxed( cfg.cfg_env.clone(), block_env.clone(), - Call::evm_config(this.eth_api()).tx_env(&tx), + Call::evm_config(this.eth_api()).tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(tx_index)), ), handler_cfg: cfg.handler_cfg, }; let (res, _) = this.inner.eth_api.transact(&mut db, env)?; db.commit(res.state); + #[cfg(feature = "telos")] + { + tx_index += 1; + } } } @@ -537,6 +562,8 @@ where gas_limit, &mut db, overrides, + #[cfg(feature = "telos")] + telos_block_extension.tx_env_at(tx_index), )?; let (trace, state) = @@ -548,6 +575,10 @@ where db.commit(state); } results.push(trace); + #[cfg(feature = "telos")] + { + tx_index += 1; + } } // Increment block_env number and timestamp for the next bundle block_env.number += U256::from(1); @@ -573,6 +604,8 @@ where self.inner.eth_api.block_with_senders(block_id.into()), )?; let block = maybe_block.ok_or(EthApiError::UnknownBlockNumber)?; + #[cfg(feature = "telos")] + let telos_block_extension = block.header.telos_block_extension.clone(); let this = self.clone(); @@ -606,6 +639,9 @@ where ) .map_err(|err| EthApiError::Internal(err.into()))?; + #[cfg(feature = "telos")] + let mut tx_index = 0; + // Re-execute all of the transactions in the block to load all touched accounts into // the cache DB. for tx in block.into_transactions_ecrecovered() { @@ -613,13 +649,17 @@ where env: Env::boxed( cfg.cfg_env.clone(), block_env.clone(), - evm_config.tx_env(&tx), + evm_config.tx_env(&tx, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(tx_index)), ), handler_cfg: cfg.handler_cfg, }; let (res, _) = this.inner.eth_api.transact(&mut db, env)?; db.commit(res.state); + #[cfg(feature = "telos")] + { + tx_index += 1; + } } // Merge all state transitions diff --git a/crates/rpc/rpc/src/eth/bundle.rs b/crates/rpc/rpc/src/eth/bundle.rs index 9cabc1f6f5fd..44821dce332a 100644 --- a/crates/rpc/rpc/src/eth/bundle.rs +++ b/crates/rpc/rpc/src/eth/bundle.rs @@ -143,6 +143,18 @@ where block_env.number = U256::from(block_number); let eth_api = self.inner.eth_api.clone(); + #[cfg(feature = "telos")] + let telos_block_extension; + #[cfg(feature = "telos")] + { + let parent_block = block_env.number.saturating_to::(); + // here we need to fetch the _next_ block's basefee based on the parent block + let parent = LoadPendingBlock::provider(&self.inner.eth_api) + .header_by_number(parent_block) + .map_err(Eth::Error::from_eth_err)? + .ok_or_else(|| EthApiError::UnknownBlockNumber)?; + telos_block_extension = parent.telos_block_extension.to_child(); + } self.inner .eth_api @@ -167,6 +179,9 @@ where let mut results = Vec::with_capacity(transactions.len()); let mut transactions = transactions.into_iter().peekable(); + #[cfg(feature = "telos")] + let mut tx_index = 0; + while let Some((tx, signer)) = transactions.next() { // Verify that the given blob data, commitments, and proofs are all valid for // this transaction. @@ -183,7 +198,7 @@ where .effective_tip_per_gas(basefee) .ok_or_else(|| RpcInvalidTransactionError::FeeCapTooLow) .map_err(Eth::Error::from_eth_err)?; - Call::evm_config(ð_api).fill_tx_env(evm.tx_mut(), &tx, signer); + Call::evm_config(ð_api).fill_tx_env(evm.tx_mut(), &tx, signer, #[cfg(feature = "telos")] telos_block_extension.tx_env_at(tx_index)); let ResultAndState { result, state } = evm.transact().map_err(Eth::Error::from_evm_err)?; @@ -226,6 +241,11 @@ where }; results.push(tx_res); + #[cfg(feature = "telos")] + { + tx_index += 1; + } + // need to apply the state changes of this call before executing the // next call if transactions.peek().is_some() { diff --git a/crates/rpc/rpc/src/trace.rs b/crates/rpc/rpc/src/trace.rs index cb9f0ab269bd..0b4a4880d088 100644 --- a/crates/rpc/rpc/src/trace.rs +++ b/crates/rpc/rpc/src/trace.rs @@ -1,5 +1,5 @@ use std::{collections::HashSet, sync::Arc}; - +use std::hash::Hash; use async_trait::async_trait; use jsonrpsee::core::RpcResult; use reth_chainspec::{ChainSpec, EthereumHardforks}; @@ -8,7 +8,7 @@ use reth_consensus_common::calc::{ }; use reth_evm::ConfigureEvmEnv; use reth_primitives::{BlockId, Bytes, Header, B256, U256}; -use reth_provider::{BlockReader, ChainSpecProvider, EvmEnvProvider, StateProviderFactory}; +use reth_provider::{BlockReader, ChainSpecProvider, EvmEnvProvider, HeaderProvider, StateProviderFactory}; use reth_revm::database::StateProviderDatabase; use reth_rpc_api::TraceApiServer; use reth_rpc_eth_api::{ @@ -16,16 +16,12 @@ use reth_rpc_eth_api::{ FromEthApiError, }; use reth_rpc_eth_types::{error::EthApiError, utils::recover_raw_transaction}; -use reth_rpc_types::{ - state::{EvmOverrides, StateOverride}, - trace::{ - filter::TraceFilter, - opcode::{BlockOpcodeGas, TransactionOpcodeGas}, - parity::*, - tracerequest::TraceCallRequest, - }, - BlockOverrides, Index, TransactionRequest, -}; +use reth_rpc_types::{state::{EvmOverrides, StateOverride}, trace::{ + filter::TraceFilter, + opcode::{BlockOpcodeGas, TransactionOpcodeGas}, + parity::*, + tracerequest::TraceCallRequest, +}, BlockHashOrNumber, BlockOverrides, Index, TransactionRequest}; use reth_tasks::pool::BlockingTaskGuard; use revm::{ db::{CacheDB, DatabaseCommit}, @@ -36,6 +32,7 @@ use revm_inspectors::{ tracing::{parity::populate_state_diff, TracingInspector, TracingInspectorConfig}, }; use tokio::sync::{AcquireError, OwnedSemaphorePermit}; +use reth_telos_primitives_traits::TelosTxEnv; /// `trace` API implementation. /// @@ -120,10 +117,17 @@ where let (cfg, block, at) = self.inner.eth_api.evm_env_at(block_id.unwrap_or_default()).await?; + #[cfg(feature = "telos")] + let telox_tx_env = self.provider() + .block(BlockHashOrNumber::Hash(block_id.unwrap_or_default().as_block_hash().unwrap())) + .unwrap().unwrap() // TODO: Fix this + .header + .telos_block_extension + .tx_env_at(0); let env = EnvWithHandlerCfg::new_with_cfg_env( cfg, block, - Call::evm_config(self.eth_api()).tx_env(&tx.into_ecrecovered_transaction()), + Call::evm_config(self.eth_api()).tx_env(&tx.into_ecrecovered_transaction(), #[cfg(feature = "telos")] telox_tx_env), ); let config = TracingInspectorConfig::from_parity_config(&trace_types); @@ -151,6 +155,14 @@ where let at = block_id.unwrap_or(BlockId::pending()); let (cfg, block_env, at) = self.inner.eth_api.evm_env_at(at).await?; + #[cfg(feature = "telos")] + let telos_tx_env = self.provider() + .block(BlockHashOrNumber::Hash(at.as_block_hash().unwrap())) + .unwrap().unwrap() // TODO: Fix this + .header + .telos_block_extension + .tx_env_at(0); + let gas_limit = self.inner.eth_api.call_gas_limit(); let this = self.clone(); // execute all transactions on top of each other and record the traces @@ -169,6 +181,8 @@ where gas_limit, &mut db, Default::default(), + #[cfg(feature = "telos")] + telos_tx_env.clone(), )?; let config = TracingInspectorConfig::from_parity_config(&trace_types); let mut inspector = TracingInspector::new(config); diff --git a/crates/storage/codecs/Cargo.toml b/crates/storage/codecs/Cargo.toml index 7ecf75208ac1..25ed0704775b 100644 --- a/crates/storage/codecs/Cargo.toml +++ b/crates/storage/codecs/Cargo.toml @@ -51,3 +51,6 @@ alloy = [ "dep:alloy-trie", "dep:serde" ] +telos = [ + "reth-codecs-derive/telos", +] \ No newline at end of file diff --git a/crates/storage/codecs/derive/Cargo.toml b/crates/storage/codecs/derive/Cargo.toml index f62587faa938..df97f6e4fed3 100644 --- a/crates/storage/codecs/derive/Cargo.toml +++ b/crates/storage/codecs/derive/Cargo.toml @@ -23,3 +23,6 @@ syn.workspace = true [dev-dependencies] similar-asserts.workspace = true syn = { workspace = true, features = ["full", "extra-traits"] } + +[features] +telos =[] diff --git a/crates/storage/codecs/derive/src/compact/generator.rs b/crates/storage/codecs/derive/src/compact/generator.rs index 9b1b09e47689..bfa33ee9f94b 100644 --- a/crates/storage/codecs/derive/src/compact/generator.rs +++ b/crates/storage/codecs/derive/src/compact/generator.rs @@ -96,6 +96,8 @@ fn generate_from_compact(fields: &FieldList, ident: &Ident, is_zstd: bool) -> To // // This removes the requirement of the field to be placed last in the struct. known_types.extend_from_slice(&["TxKind", "AccessList", "Signature", "CheckpointBlockRange"]); + #[cfg(feature = "telos")] + known_types.push("TelosBlockExtension"); // let mut handle = FieldListHandler::new(fields); let is_enum = fields.iter().any(|field| matches!(field, FieldTypes::EnumVariant(_))); diff --git a/crates/telos/node/Cargo.toml b/crates/telos/node/Cargo.toml index 6473a923c72c..f1e68e7ed948 100644 --- a/crates/telos/node/Cargo.toml +++ b/crates/telos/node/Cargo.toml @@ -58,13 +58,16 @@ telos = [ "reth-transaction-pool/telos", "reth-primitives/telos", "reth-stages/telos", + "reth-ethereum-engine-primitives/telos", "reth/telos", ] [dev-dependencies] +alloy-consensus.workspace = true alloy-contract = "0.3.0" alloy-provider.workspace = true alloy-network.workspace = true +alloy-rpc-client.workspace = true alloy-rpc-types.workspace = true alloy-signer-local.workspace = true alloy-sol-types.workspace = true @@ -83,4 +86,5 @@ telos-translator-rs = { git = "https://github.com/telosnetwork/telos-consensus-c env_logger = "0.11.5" testcontainers = "0.21.1" +derive_more.workspace = true diff --git a/crates/telos/node/tests/integration.rs b/crates/telos/node/tests/integration.rs index 25e111f58cd8..472459705a68 100644 --- a/crates/telos/node/tests/integration.rs +++ b/crates/telos/node/tests/integration.rs @@ -13,17 +13,19 @@ use std::fs; use std::path::PathBuf; use std::sync::Arc; use std::time::Duration; +use alloy_network::{AnyTxType, Network}; +use alloy_network::eip2718::Eip2718Error; use reqwest::Url; use telos_consensus_client::client::ConsensusClient; use telos_consensus_client::config::{AppConfig, CliArgs}; -use telos_consensus_client::data::Block; use telos_translator_rs::{block::TelosEVMBlock, types::translator_types::ChainId}; use telos_consensus_client::main_utils::build_consensus_client; use telos_translator_rs::translator::Translator; use testcontainers::core::ContainerPort::Tcp; use testcontainers::{runners::AsyncRunner, ContainerAsync, GenericImage}; use tokio::sync::mpsc; -use tokio::time::sleep; +use reth::rpc::types::{AnyTransactionReceipt, Header, Transaction, TransactionRequest, WithOtherFields}; +use derive_more::Display; pub mod live_test_runner; @@ -32,8 +34,10 @@ struct TelosRethNodeHandle { jwt_secret: String, } -const CONTAINER_TAG: &str = "v0.1.9@sha256:6d4946f112e5c26712a938ea332b76e742035e64af93149227d97dd67e1a9012"; -const CONTAINER_LAST_EVM_BLOCK: u64 = 50; +const CONTAINER_TAG: &str = "v0.1.11@sha256:d138f2e08db108d5d420b4db99a57fb9d45a3ee3e0f0faa7d4c4a065f7f018ce"; + +// This is the last block in the container, after this block the node is done syncing and is running live +const CONTAINER_LAST_EVM_BLOCK: u64 = 1010; async fn start_ship() -> ContainerAsync { // Change this container to a local image if using new ship data, @@ -111,7 +115,7 @@ async fn build_consensus_and_translator( jwt_secret: reth_handle.jwt_secret, ship_endpoint: format!("ws://localhost:{ship_port}"), chain_endpoint: format!("http://localhost:{chain_port}"), - batch_size: 10, + batch_size: 1, prev_hash: "b25034033c9ca7a40e879ddcc29cf69071a22df06688b5fe8cc2d68b4e0528f9".to_string(), validate_hash: None, evm_start_block: 1, @@ -194,9 +198,6 @@ async fn testing_chain_sync() { let rpc_url = Url::from(format!("http://localhost:{}", rpc_port).parse().unwrap()); let provider = ProviderBuilder::new().on_http(rpc_url.clone()); - // todo wait time or anything else to wait for shutdown - println!("Waiting to send shutdown signal..."); - loop { tokio::time::sleep(Duration::from_secs(1)).await; let latest_block = provider.get_block_number().await.unwrap(); @@ -206,19 +207,21 @@ async fn testing_chain_sync() { break; } if latest_block > CONTAINER_LAST_EVM_BLOCK { - _ = translator_shutdown.shutdown().await.unwrap(); - _ = consensus_shutdown.shutdown().await.unwrap(); break; } } - _ = tokio::join!(client_handle, translator_handle); - println!("Translator shutdown done."); - println!("Client shutdown done."); - live_test_runner::run_tests( &rpc_url.clone().to_string(), "87ef69a835f8cd0c44ab99b7609a20b2ca7f1c8470af4f0e5b44db927d542084", ) .await; + + _ = translator_shutdown.shutdown().await.unwrap(); + _ = consensus_shutdown.shutdown().await.unwrap(); + + println!("Client shutdown done."); + + _ = tokio::join!(client_handle, translator_handle); + println!("Translator shutdown done."); } diff --git a/crates/telos/node/tests/live_test_runner.rs b/crates/telos/node/tests/live_test_runner.rs index 9a941aa3a92c..81650c2b476e 100644 --- a/crates/telos/node/tests/live_test_runner.rs +++ b/crates/telos/node/tests/live_test_runner.rs @@ -1,12 +1,13 @@ use std::str::FromStr; -use alloy_network::{Ethereum, Network}; +use alloy_network::{Ethereum, Network, ReceiptResponse, TransactionBuilder}; use alloy_provider::network::EthereumWallet; use alloy_provider::{Provider, ProviderBuilder, ReqwestProvider}; use alloy_provider::fillers::{FillProvider, JoinFill, WalletFiller}; -use alloy_rpc_types::TransactionRequest; +use alloy_rpc_client::ClientBuilder; +use alloy_rpc_types::{BlockNumberOrTag, TransactionRequest}; use alloy_signer_local::PrivateKeySigner; -use alloy_sol_types::private::primitives::TxKind::Create; -use alloy_sol_types::{sol, SolValue}; +use alloy_sol_types::private::primitives::TxKind::{Call, Create}; +use alloy_sol_types::{sol, SolEvent, SolValue}; use alloy_transport_http::Http; use reth::primitives::{AccessList, Address, BlockId, U256}; @@ -29,7 +30,12 @@ pub async fn run_local() { pub async fn run_tests(url: &str, private_key: &str) { let signer = PrivateKeySigner::from_str(private_key).unwrap(); let wallet = EthereumWallet::from(signer.clone()); - let provider = ProviderBuilder::new().wallet(wallet.clone()).on_http(Url::from_str(url).unwrap()); + + let provider = ProviderBuilder::new() + //.network::() + .wallet(wallet.clone()) + .on_http(Url::from_str(url).unwrap()); + let signer_address = signer.address(); let balance = provider.get_balance(signer_address).await.unwrap(); @@ -64,19 +70,18 @@ pub async fn test_blocknum_onchain(url: &str, private_key: &str) { let provider = ProviderBuilder::new().wallet(wallet.clone()).on_http(Url::from_str(url).unwrap()); - info!("Deploying contract using address {address}"); let nonce = provider.get_transaction_count(address).await.unwrap(); let chain_id = provider.get_chain_id().await.unwrap(); - let gas_price: u128 = 600 * (GWEI_TO_WEI as u128); + let gas_price = provider.get_gas_price().await.unwrap(); let legacy_tx = TxLegacy { chain_id: Some(chain_id), nonce, gas_price: gas_price.into(), gas_limit: 20_000_000, - to: reth::primitives::TxKind::Create, + to: Create, value: U256::ZERO, input: BlockNumChecker::BYTECODE.to_vec().into(), }; @@ -89,7 +94,6 @@ pub async fn test_blocknum_onchain(url: &str, private_key: &str) { value: Some(legacy_tx.value), input: TransactionInput::from(legacy_tx.input), nonce: Some(legacy_tx.nonce), - access_list: Some(AccessList::default()), chain_id: legacy_tx.chain_id, ..Default::default() }; @@ -100,4 +104,38 @@ pub async fn test_blocknum_onchain(url: &str, private_key: &str) { info!("Deployed contract with tx hash: {deploy_tx_hash}"); let receipt = deploy_result.get_receipt().await.unwrap(); info!("Receipt: {:?}", receipt); + + let contract_address = receipt.contract_address().unwrap(); + let block_num_checker = BlockNumChecker::new(contract_address, provider.clone()); + + let legacy_tx_request = TransactionRequest::default() + .with_from(address) + .with_to(contract_address) + .with_gas_limit(20_000_000) + .with_gas_price(gas_price) + .with_input(block_num_checker.logBlockNum().calldata().clone()) + .with_nonce(provider.get_transaction_count(address).await.unwrap()) + .with_chain_id(chain_id); + + let log_block_num_tx_result = provider.send_transaction(legacy_tx_request).await.unwrap(); + + let log_block_num_tx_hash = log_block_num_tx_result.tx_hash(); + info!("Called contract with tx hash: {log_block_num_tx_hash}"); + let receipt = log_block_num_tx_result.get_receipt().await.unwrap(); + info!("log block number receipt: {:?}", receipt); + let rpc_block_num = receipt.block_number().unwrap(); + let receipt = receipt.inner; + let logs = receipt.logs(); + let first_log = logs[0].clone().inner; + let block_num_event = BlockNumChecker::BlockNumber::decode_log(&first_log, true).unwrap(); + assert_eq!(U256::from(rpc_block_num), block_num_event.number); + info!("Block numbers match inside transaction event"); + + // The below needs to be done using LegacyTransaction style call... with the current code it's including base_fee_per_gas and being rejected by reth + // let block_num_latest = block_num_checker.getBlockNum().call().await.unwrap(); + // assert!(block_num_latest._0 > U256::from(rpc_block_num), "Latest block number via call to getBlockNum is not greater than the block number in the previous log event"); + // + // let block_num_five_back = block_num_checker.getBlockNum().call().block(BlockId::number(rpc_block_num - 5)).await.unwrap(); + // assert!(block_num_five_back._0 == U256::from(rpc_block_num - 5), "Block number 5 blocks back via historical eth_call is not correct"); + } \ No newline at end of file diff --git a/crates/telos/primitives-traits/Cargo.toml b/crates/telos/primitives-traits/Cargo.toml new file mode 100644 index 000000000000..0ac09a9841b3 --- /dev/null +++ b/crates/telos/primitives-traits/Cargo.toml @@ -0,0 +1,23 @@ +[package] +name = "reth-telos-primitives-traits" +version.workspace = true +edition.workspace = true +rust-version.workspace = true +license.workspace = true +homepage.workspace = true +repository.workspace = true +exclude.workspace = true + +[dependencies] +arbitrary = { workspace = true, optional = true } +alloy-primitives = { workspace = true } +bytes.workspace = true +modular-bitfield.workspace = true +reth-codecs = { workspace = true } +serde.workspace = true + +[lints] +workspace = true + +[features] +arbitrary = ["dep:arbitrary"] \ No newline at end of file diff --git a/crates/telos/primitives-traits/src/lib.rs b/crates/telos/primitives-traits/src/lib.rs new file mode 100644 index 000000000000..b97c367e1e82 --- /dev/null +++ b/crates/telos/primitives-traits/src/lib.rs @@ -0,0 +1,161 @@ +//! Crate for Telos specific primitive traits + +#![doc( + html_logo_url = "https://raw.githubusercontent.com/paradigmxyz/reth/main/assets/reth-docs.png", + html_favicon_url = "https://avatars0.githubusercontent.com/u/97369466?s=256", + issue_tracker_base_url = "https://github.com/telosnetwork/telos-reth/issues/" +)] +#![cfg_attr(docsrs, feature(doc_cfg, doc_auto_cfg))] + +use alloy_primitives::U256; +use serde::{Deserialize, Serialize}; +use reth_codecs::Compact; + +/// Telos block extension fields, included in Headers table as part of Header +#[cfg_attr(any(test, feature = "arbitrary"), derive(arbitrary::Arbitrary))] +#[derive(Debug, Clone, Default, PartialEq, Eq, Hash, Serialize, Deserialize, Compact)] +pub struct TelosBlockExtension { + /// Initial gas price for this block + pub starting_gas_price: U256, + /// Initial revision number for this block + pub starting_revision_number: u64, + /// Changed gas price for this block + pub gas_price_change: Option, + /// Changed revision number for this block + pub revision_change: Option, +} + +impl TelosBlockExtension { + /// Create a new `TelosBlockExtension` using a parent extension to fetch the starting price/revision + /// plus the `TelosExtraAPIFields` for the current block + pub fn from_parent_and_changes( + parent: &Self, + gas_price_change: Option<(u64, U256)>, + revision_change: Option<(u64, u64)> + ) -> Self { + let mut starting_gas_price = parent.get_last_gas_price(); + let mut starting_revision_number = parent.get_last_revision(); + let gas_price_change = if let Some(price_change) = gas_price_change { + // if transaction index is > 0 this means it changed after the starting value at index 0, + // we keep both values in the vector + if price_change.0 > 0 { + Some(GasPrice { + height: price_change.0, + price: price_change.1, + }) + } else { + // price change at index 0, we only keep the new value, don't care about the value from + // the parent block + starting_gas_price = price_change.1; + None + } + } else { + // no change in this block, we keep the last value from the parent block + None + }; + + let revision_change = if let Some(revision) = revision_change { + // if transaction index is > 0 this means it changed after the starting value at index 0, + // we keep both values in the vector + if revision.0 > 0 { + Some(Revision { + height: revision.0, + revision: revision.1, + }) + } else { + // price change at index 0, we only keep the new value, don't care about the value from + // the parent block + starting_revision_number = revision.1; + None + } + } else { + // no change in this block, we keep the last value from the parent block + None + }; + + Self { + starting_gas_price, + starting_revision_number, + gas_price_change, + revision_change, + } + } + + /// Create a new Telos block extension for a child block, starting with the last price/revision from this one + pub fn to_child(&self) -> Self { + Self { + starting_gas_price: self.get_last_gas_price(), + starting_revision_number: self.get_last_revision(), + gas_price_change: None, + revision_change: None, + } + } + + /// Get the ending gas price of this block + pub fn get_last_gas_price(&self) -> U256 { + if self.gas_price_change.is_none() { + self.starting_gas_price + } else { + self.gas_price_change.as_ref().unwrap().price + } + } + + /// Get the ending revision number of this block + pub fn get_last_revision(&self) -> u64 { + if self.revision_change.is_none() { + self.starting_revision_number + } else { + self.revision_change.as_ref().unwrap().revision + } + } + + /// Get `TelosTxEnv` at a given transaction index + pub fn tx_env_at(&self, height: u64) -> TelosTxEnv { + let gas_price = if self.gas_price_change.is_some() && self.gas_price_change.as_ref().unwrap().height <= height { + self.gas_price_change.as_ref().unwrap().price + } else { + self.starting_gas_price + }; + + let revision = if self.revision_change.is_some() && self.revision_change.as_ref().unwrap().height <= height { + self.revision_change.as_ref().unwrap().revision + } else { + self.starting_revision_number + }; + + TelosTxEnv { + gas_price, + revision, + } + } +} + +/// Telos transaction environment data +#[cfg_attr(any(test, feature = "arbitrary"), derive(arbitrary::Arbitrary))] +#[derive(Debug, Clone, Default, PartialEq, Eq, Hash, Serialize, Deserialize, Compact)] +pub struct TelosTxEnv { + /// Gas price for this transaction + pub gas_price: U256, + /// Revision number for this transaction + pub revision: u64, +} + +/// Telos gas price +#[cfg_attr(any(test, feature = "arbitrary"), derive(arbitrary::Arbitrary))] +#[derive(Debug, Clone, Default, PartialEq, Eq, Hash, Serialize, Deserialize, Compact)] +pub struct GasPrice { + /// Transaction height + pub height: u64, + /// Value + pub price: U256, +} + +/// Telos revision number +#[cfg_attr(any(test, feature = "arbitrary"), derive(arbitrary::Arbitrary))] +#[derive(Debug, Clone, Default, PartialEq, Eq, Hash, Serialize, Deserialize, Compact)] +pub struct Revision { + /// Transaction height + pub height: u64, + /// Revision + pub revision: u64, +} \ No newline at end of file diff --git a/crates/telos/rpc-engine-api/src/compare.rs b/crates/telos/rpc-engine-api/src/compare.rs index 22d1a04b942c..29d897b8655e 100644 --- a/crates/telos/rpc-engine-api/src/compare.rs +++ b/crates/telos/rpc-engine-api/src/compare.rs @@ -6,7 +6,7 @@ use revm::db::AccountStatus; use revm::{Database, Evm, State, TransitionAccount}; use reth_storage_errors::provider::ProviderError; use crate::structs::{TelosAccountStateTableRow, TelosAccountTableRow}; -use tracing::info; +use tracing::{debug, info}; /// This function compares the state diffs between revm and Telos EVM contract pub fn compare_state_diffs( @@ -27,9 +27,9 @@ where { let block_number = evm.block().number; - info!("{block_number} REVM State diffs: {:#?}", revm_state_diffs); - info!("{block_number} TEVM State diffs account: {:#?}", statediffs_account); - info!("{block_number} TEVM State diffs accountstate: {:#?}", statediffs_accountstate); + debug!("{block_number} REVM State diffs: {:#?}", revm_state_diffs); + debug!("{block_number} TEVM State diffs account: {:#?}", statediffs_account); + debug!("{block_number} TEVM State diffs accountstate: {:#?}", statediffs_accountstate); } let revm_db: &mut &mut State = evm.db_mut(); diff --git a/crates/telos/rpc-engine-api/src/structs.rs b/crates/telos/rpc-engine-api/src/structs.rs index cbd9a4b3a427..1dc9c9cd34f8 100644 --- a/crates/telos/rpc-engine-api/src/structs.rs +++ b/crates/telos/rpc-engine-api/src/structs.rs @@ -49,4 +49,4 @@ pub struct TelosEngineAPIExtraFields { pub new_addresses_using_openwallet: Option>, /// Receipts produced by telos.evm contract pub receipts: Option> -} \ No newline at end of file +} diff --git a/crates/telos/rpc/src/eth/telos_client.rs b/crates/telos/rpc/src/eth/telos_client.rs index 118cbe92f250..0e82849be80e 100644 --- a/crates/telos/rpc/src/eth/telos_client.rs +++ b/crates/telos/rpc/src/eth/telos_client.rs @@ -50,7 +50,7 @@ impl TelosClient { if telos_client_args.telos_endpoint.is_none() || telos_client_args.signer_account.is_none() || telos_client_args.signer_permission.is_none() || telos_client_args.signer_key.is_none() { panic!("Should not construct TelosClient without proper TelosArgs with telos_endpoint and signer args"); } - let api_client = APIClient::::default_provider(telos_client_args.telos_endpoint.unwrap().into()).unwrap(); + let api_client = APIClient::::default_provider(telos_client_args.telos_endpoint.unwrap().into(), Some(3)).unwrap(); let inner = TelosClientInner { api_client, signer_account: name!(&telos_client_args.signer_account.unwrap()), diff --git a/crates/telos/rpc/src/eth/transaction.rs b/crates/telos/rpc/src/eth/transaction.rs index 25bc13e71649..91cb337abb4d 100644 --- a/crates/telos/rpc/src/eth/transaction.rs +++ b/crates/telos/rpc/src/eth/transaction.rs @@ -11,7 +11,7 @@ use reth_rpc_eth_api::{ use reth_rpc_eth_types::{utils::recover_raw_transaction, EthStateCache}; use reth_transaction_pool::{PoolTransaction, TransactionOrigin, TransactionPool}; -use crate::{eth::telos_client::TelosClient}; +use crate::eth::telos_client::TelosClient; use crate::eth::TelosEthApi; impl EthTransactions for TelosEthApi @@ -37,9 +37,14 @@ where // On Telos, transactions are forwarded directly to the native network to be included in a block. if let Some(client) = self.raw_tx_forwarder().as_ref() { tracing::debug!( target: "rpc::eth", "forwarding raw transaction to Telos native"); - let _ = client.send_to_telos(&tx).await.inspect_err(|err| { + let result = client.send_to_telos(&tx).await.inspect_err(|err| { tracing::debug!(target: "rpc::eth", %err, hash=% *pool_transaction.hash(), "failed to forward raw transaction"); }); + + // TODO: Retry here if it's a network error, parse errors from Telos and try to return appropriate error to client + if let Err(err) = result { + return Err(Self::Error::from_eth_err(err)); + } } // submit the transaction to the pool with a `Local` origin diff --git a/start.sh b/start.sh index c3b8f6f1f7c2..3f99a5b1c11e 100755 --- a/start.sh +++ b/start.sh @@ -17,11 +17,18 @@ cd $INSTALL_ROOT [ -z "$DATA_DIR" ] && DATA_DIR=$INSTALL_ROOT/data [ -z "$CHAIN" ] && CHAIN=tevmmainnet [ -z "$RETH_CONFIG" ] && RETH_CONFIG=$DATA_DIR/config.toml -[ -z "$RETH_RPC_ADDRESS" ] && RETH_RPC_ADDRESS= +[ -z "$RETH_RPC_ADDRESS" ] && RETH_RPC_ADDRESS=127.0.0.1 [ -z "$RETH_RPC_PORT" ] && RETH_RPC_PORT=8545 +[ -z "$RETH_WS_ADDRESS" ] && RETH_WS_ADDRESS=127.0.0.1 +[ -z "$RETH_WS_PORT" ] && RETH_WS_PORT=8546 +[ -z "$RETH_AUTH_RPC_ADDRESS" ] && RETH_AUTH_RPC_ADDRESS=127.0.0.1 +[ -z "$RETH_AUTH_RPC_PORT" ] && RETH_AUTH_RPC_PORT=8551 +[ -z "$RETH_IPCPATH" ] && RETH_IPCPATH=$INSTALL_ROOT/reth.ipc +[ -z "$RETH_DISCOVERY_PORT" ] && RETH_DISCOVERY_PORT=30303 [ -z "$LOG_PATH" ] && LOG_PATH=$DATA_DIR/reth.log nohup $RETH_BIN_PATH node \ + --port $RETH_DISCOVERY_PORT \ --log.stdout.filter $LOG_LEVEL \ --datadir $DATA_DIR \ --chain $CHAIN \ @@ -32,6 +39,12 @@ nohup $RETH_BIN_PATH node \ --http.api all \ --ws \ --ws.api all \ + --ws.addr $RETH_WS_ADDRESS \ + --ws.port $RETH_WS_PORT \ + --authrpc.addr $RETH_AUTH_RPC_ADDRESS \ + --authrpc.port $RETH_AUTH_RPC_PORT \ + --ipcpath $RETH_IPCPATH \ + --discovery.port $RETH_DISCOVERY_PORT \ --telos.telos_endpoint $TELOS_ENDPOINT \ --telos.signer_account $TELOS_SIGNER_ACCOUNT \ --telos.signer_permission $TELOS_SIGNER_PERMISSION \