From ce7cdd1c0c3e955c18beb9e2e306c4cff0ef5cfa Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Thu, 23 Nov 2023 01:52:18 +0400 Subject: [PATCH 1/7] add new runtime apis to validate and do pre dispatch for tx + other refactoring changes --- Cargo.lock | 2 + crates/sp-domains-fraud-proof/Cargo.toml | 2 + crates/sp-domains-fraud-proof/src/tests.rs | 514 +++++++++++++++++- .../src/stateless_runtime.rs | 15 + domains/primitives/runtime/src/lib.rs | 53 +- domains/runtime/evm/src/lib.rs | 163 ++++-- domains/test/runtime/evm/src/lib.rs | 160 ++++-- 7 files changed, 839 insertions(+), 70 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 33365dfeff..4b1faac7ad 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -10984,7 +10984,9 @@ dependencies = [ "domain-block-preprocessor", "domain-runtime-primitives", "domain-test-service", + "fp-rpc", "frame-support", + "frame-system", "futures", "hash-db 0.16.0", "pallet-balances", diff --git a/crates/sp-domains-fraud-proof/Cargo.toml b/crates/sp-domains-fraud-proof/Cargo.toml index 9fc7de8d66..feb4bc223d 100644 --- a/crates/sp-domains-fraud-proof/Cargo.toml +++ b/crates/sp-domains-fraud-proof/Cargo.toml @@ -40,6 +40,8 @@ thiserror = { version = "1.0.48", optional = true } domain-block-builder = { version = "0.1.0", path = "../../domains/client/block-builder" } domain-block-preprocessor = { version = "0.1.0", path = "../../domains/client/block-preprocessor" } domain-test-service = { version = "0.1.0", path = "../../domains/test/service" } +fp-rpc = { version = "3.0.0-dev", git = "https://github.com/subspace/frontier", rev = "56086daa77802eaa285894bfe4b811be66629c89", features = ['default'] } +frame-system = { version = "4.0.0-dev", default-features = false, git = "https://github.com/subspace/polkadot-sdk", rev = "892bf8e938c6bd2b893d3827d1093cd81baa59a1" } futures = "0.3.29" pallet-balances = { version = "4.0.0-dev", git = "https://github.com/subspace/polkadot-sdk", rev = "892bf8e938c6bd2b893d3827d1093cd81baa59a1" } sc-cli = { version = "0.10.0-dev", git = "https://github.com/subspace/polkadot-sdk", rev = "892bf8e938c6bd2b893d3827d1093cd81baa59a1", default-features = false } diff --git a/crates/sp-domains-fraud-proof/src/tests.rs b/crates/sp-domains-fraud-proof/src/tests.rs index a187949acd..6c828451e2 100644 --- a/crates/sp-domains-fraud-proof/src/tests.rs +++ b/crates/sp-domains-fraud-proof/src/tests.rs @@ -1,15 +1,17 @@ -use codec::Encode; +use codec::{Decode, Encode}; use domain_block_preprocessor::stateless_runtime::StatelessRuntime; -use domain_runtime_primitives::{CheckTxValidityError, DomainCoreApi}; +use domain_runtime_primitives::{CheckBundleValidityError, CheckTxValidityError, DomainCoreApi}; use domain_test_service::domain::EvmDomainClient as DomainClient; use domain_test_service::evm_domain_test_runtime::Runtime as TestRuntime; use domain_test_service::EcdsaKeyring::{Alice, Bob, Charlie}; use domain_test_service::Sr25519Keyring::Ferdie; -use domain_test_service::{construct_extrinsic_generic, GENESIS_DOMAIN_ID}; +use domain_test_service::{construct_extrinsic_generic, EvmDomainNode, GENESIS_DOMAIN_ID}; +use fp_rpc::EthereumRuntimeRPCApi; use sc_client_api::{HeaderBackend, ProofProvider, StorageProof}; use sc_service::{BasePath, Role}; use sp_api::{BlockT, ProvideRuntimeApi}; use sp_domains::DomainsApi; +use sp_runtime::generic::UncheckedExtrinsic; use sp_runtime::traits::{BlakeTwo256, Header as HeaderT}; use sp_runtime::transaction_validity::{InvalidTransaction, TransactionValidityError}; use sp_runtime::{OpaqueExtrinsic, Storage}; @@ -17,7 +19,7 @@ use sp_trie::{read_trie_value, LayoutV1}; use std::collections::BTreeMap; use std::sync::Arc; use subspace_runtime_primitives::opaque::Block; -use subspace_test_service::{produce_blocks, MockConsensusNode}; +use subspace_test_service::{produce_block_with, produce_blocks, MockConsensusNode}; use tempfile::TempDir; type HashFor = <::Header as HeaderT>::Hash; @@ -39,6 +41,320 @@ fn generate_storage_proof_for_tx_validity( .unwrap() } +fn get_storage_keys_for_verifying_signed_tx_validity( + domain_node: &EvmDomainNode, + extrinsic: UncheckedExtrinsic< + domain_test_service::evm_domain_test_runtime::Address, + ::RuntimeCall, + domain_test_service::evm_domain_test_runtime::Signature, + domain_test_service::evm_domain_test_runtime::SignedExtra, + >, +) -> Vec> { + let extrinsic_signer = domain_node + .client + .runtime_api() + .extract_signer( + domain_node.client.as_ref().info().best_hash, + vec![extrinsic.clone().into()], + ) + .unwrap() + .first() + .unwrap() + .0 + .clone(); + let extrinsic_era = domain_node + .client + .runtime_api() + .extrinsic_era( + domain_node.client.as_ref().info().best_hash, + &extrinsic.clone().into(), + ) + .unwrap(); + + domain_node + .client + .runtime_api() + .storage_keys_for_verifying_transaction_validity( + domain_node.client.as_ref().info().best_hash, + extrinsic_signer, + domain_node.client.as_ref().info().best_number, + extrinsic_era, + ) + .unwrap() + .unwrap() +} + +#[tokio::test(flavor = "multi_thread")] +async fn check_bundle_validity_runtime_api_should_work() { + let directory = TempDir::new().expect("Must be able to create temporary directory"); + + let mut builder = sc_cli::LoggerBuilder::new(""); + builder.with_colors(false); + let _ = builder.init(); + + let tokio_handle = tokio::runtime::Handle::current(); + + // Start Ferdie + let mut ferdie = MockConsensusNode::run( + tokio_handle.clone(), + Ferdie, + BasePath::new(directory.path().join("ferdie")), + ); + + // Run Alice (a evm domain authority node) + let alice = domain_test_service::DomainNodeBuilder::new( + tokio_handle.clone(), + Alice, + BasePath::new(directory.path().join("alice")), + ) + .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) + .await; + + let mut alice_nonce = 0; + let alice_balance = alice + .client + .runtime_api() + .account_basic( + alice.client.as_ref().info().best_hash, + Alice.to_account_id().into(), + ) + .unwrap() + .balance + .as_u128(); + + let transfer_to_charlie = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: alice_balance, + }, + alice.key, + false, + alice_nonce, + 0, + ); + alice_nonce += 1; + + let transfer_to_charlie_2 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 8, + }, + alice.key, + false, + alice_nonce, + 1, + ); + + let sample_valid_bundle_extrinsics = vec![ + OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(), + OpaqueExtrinsic::from_bytes(&transfer_to_charlie_2.encode()).unwrap(), + ]; + + assert_eq!( + alice + .client + .runtime_api() + .check_bundle_extrinsics_validity( + alice.client.as_ref().info().best_hash, + sample_valid_bundle_extrinsics, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + + let transfer_to_charlie_3_duplicate_nonce_extrinsic = + construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 8, + }, + alice.key, + false, + alice_nonce, + 1, + ); + + // Here tx index: 2 is invalid + let sample_invalid_bundle_with_same_nonce_extrinsic = vec![ + OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(), + OpaqueExtrinsic::from_bytes(&transfer_to_charlie_2.encode()).unwrap(), + OpaqueExtrinsic::from_bytes(&transfer_to_charlie_3_duplicate_nonce_extrinsic.encode()) + .unwrap(), + ]; + + assert_eq!( + alice + .client + .runtime_api() + .check_bundle_extrinsics_validity( + alice.client.as_ref().info().best_hash, + sample_invalid_bundle_with_same_nonce_extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err(CheckBundleValidityError::InvalidTransaction { + tx_index: 2, + tx_validity_error: CheckTxValidityError::InvalidTransaction { + error: TransactionValidityError::Invalid(InvalidTransaction::Stale), + storage_keys: get_storage_keys_for_verifying_signed_tx_validity( + &alice, + transfer_to_charlie_3_duplicate_nonce_extrinsic.clone() + ), + }, + }) + ); + + // transfer most of the alice's balance + let alice_balance = alice + .client + .runtime_api() + .account_basic( + alice.client.as_ref().info().best_hash, + Alice.to_account_id().into(), + ) + .unwrap() + .balance + .as_u128(); + let target_balance = u32::MAX - 100; + let balance_to_transfer = alice_balance - target_balance as u128; + + // reset alice nonce now + alice_nonce = 0; + + let alice_balance_transfer_extrinsic = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: balance_to_transfer, + }, + alice.key, + false, + alice_nonce, + 1, + ); + alice_nonce += 1; + + alice + .send_extrinsic(alice_balance_transfer_extrinsic) + .await + .expect("Failed to send extrinsic"); + + let (slot, bundle) = ferdie.produce_slot_and_wait_for_bundle_submission().await; + assert!(bundle.is_some()); + produce_block_with!(ferdie.produce_block_with_slot(slot), alice) + .await + .unwrap(); + produce_blocks!(ferdie, alice, 10).await.unwrap(); + + let transfer_to_charlie_with_big_tip_1 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 1000000000u32, + ); + alice_nonce += 1; + + let transfer_to_charlie_with_big_tip_2 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 1000000000u32, + ); + + // In single tx context both are valid (both tx are identical except nonce), so no need to test both of them independently. + // only testing one should suffice + assert_eq!( + alice + .client + .runtime_api() + .check_bundle_extrinsics_validity( + alice.client.as_ref().info().best_hash, + vec![transfer_to_charlie_with_big_tip_1.clone().into(),], + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + + // Tx index: 1 is invalid since low balance to pay for the tip + let tx_validity_error = CheckTxValidityError::InvalidTransaction { + error: TransactionValidityError::Invalid(InvalidTransaction::Payment), + storage_keys: get_storage_keys_for_verifying_signed_tx_validity( + &alice, + transfer_to_charlie_with_big_tip_2.clone(), + ), + }; + assert_eq!( + alice + .client + .runtime_api() + .check_bundle_extrinsics_validity( + alice.client.as_ref().info().best_hash, + vec![ + transfer_to_charlie_with_big_tip_1.clone().into(), + transfer_to_charlie_with_big_tip_2.clone().into(), + ], + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err(CheckBundleValidityError::InvalidTransaction { + tx_index: 1, + tx_validity_error: CheckTxValidityError::decode( + &mut tx_validity_error.encode().as_slice() + ) + .unwrap(), + }) + ); + + // A runtime instance preserve changes in between + let runtime_instance = alice.client.runtime_api(); + + // First transaction should be okay + assert_eq!( + runtime_instance + .check_transaction_validity_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_to_charlie_with_big_tip_1.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + + // Second transaction should error out with exact error encountered during check of bundle validity + // to prove that state is preserved between two calls to same runtime instance + assert_eq!( + runtime_instance + .check_transaction_validity_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_to_charlie_with_big_tip_2.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err(tx_validity_error) + ); +} + #[tokio::test(flavor = "multi_thread")] async fn check_tx_validity_runtime_api_should_work() { let directory = TempDir::new().expect("Must be able to create temporary directory"); @@ -111,7 +427,7 @@ async fn check_tx_validity_runtime_api_should_work() { )), ), ( - Some(2), + Some(3), Some(TransactionValidityError::Invalid( InvalidTransaction::Payment, )), @@ -137,7 +453,7 @@ async fn check_tx_validity_runtime_api_should_work() { .runtime_api() .storage_keys_for_verifying_transaction_validity( extrinsic_best_hash, - bob_account_id.clone(), + Some(bob_account_id.clone()), extrinsic_best_number, extrinsic_era, ) @@ -225,6 +541,192 @@ async fn check_tx_validity_runtime_api_should_work() { } } +#[tokio::test(flavor = "multi_thread")] +async fn check_tx_validity_and_do_pre_dispatch_runtime_api_should_work() { + let directory = TempDir::new().expect("Must be able to create temporary directory"); + + let mut builder = sc_cli::LoggerBuilder::new(""); + builder.with_colors(false); + let _ = builder.init(); + + let tokio_handle = tokio::runtime::Handle::current(); + + // Start Ferdie + let mut ferdie = MockConsensusNode::run( + tokio_handle.clone(), + Ferdie, + BasePath::new(directory.path().join("ferdie")), + ); + + // Run Alice (a evm domain authority node) + let alice = domain_test_service::DomainNodeBuilder::new( + tokio_handle.clone(), + Alice, + BasePath::new(directory.path().join("alice")), + ) + .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) + .await; + + // Run Bob (a evm domain full node) + let bob = domain_test_service::DomainNodeBuilder::new( + tokio_handle, + Bob, + BasePath::new(directory.path().join("bob")), + ) + .build_evm_node(Role::Full, GENESIS_DOMAIN_ID, &mut ferdie) + .await; + + // Bob is able to sync blocks. + produce_blocks!(ferdie, alice, 1, bob).await.unwrap(); + + let bob_nonce = bob.account_nonce(); + let transfer_to_charlie = construct_extrinsic_generic::( + &bob.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 8, + }, + bob.key, + false, + bob_nonce, + 1, + ); + + produce_blocks!(ferdie, alice, 2, bob).await.unwrap(); + + let opaque_extrinsic = OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(); + let extrinsic_best_hash = bob.client.as_ref().info().best_hash; + let bob_account_id = Bob.to_account_id().encode(); + let extrinsic_best_number = bob.client.as_ref().info().best_number; + + let extrinsic_era = bob + .client + .runtime_api() + .extrinsic_era(bob.client.as_ref().info().best_hash, &opaque_extrinsic) + .unwrap(); + + let test_table = vec![ + ( + Some(4), + Some(TransactionValidityError::Invalid( + InvalidTransaction::AncientBirthBlock, + )), + ), + ( + Some(3), + Some(TransactionValidityError::Invalid( + InvalidTransaction::Payment, + )), + ), + ( + Some(1), + Some(TransactionValidityError::Invalid( + InvalidTransaction::BadProof, + )), + ), + ( + Some(0), + Some(TransactionValidityError::Invalid( + InvalidTransaction::BadProof, + )), + ), + (None, None), + ]; + + for table_data in test_table { + let mut storage_keys_for_tx_validity = bob + .client + .runtime_api() + .storage_keys_for_verifying_transaction_validity( + extrinsic_best_hash, + Some(bob_account_id.clone()), + extrinsic_best_number, + extrinsic_era, + ) + .unwrap() + .unwrap(); + + let maybe_storage_key_to_remove = table_data.0; + let original_storage_keys_for_tx_validity = storage_keys_for_tx_validity.clone(); + + if let Some(storage_key_to_remove) = maybe_storage_key_to_remove { + storage_keys_for_tx_validity = storage_keys_for_tx_validity + .drain(..) + .enumerate() + .filter(|(i, _v)| *i != storage_key_to_remove) + .map(|(_i, v)| v) + .collect(); + } + + let maybe_error = table_data.1; + + let expected_out = if let Some(err) = maybe_error { + let err_with_storage_keys = CheckTxValidityError::InvalidTransaction { + error: err, + storage_keys: original_storage_keys_for_tx_validity, + }; + Err(err_with_storage_keys) + } else { + Ok(()) + }; + + let storage_proof = generate_storage_proof_for_tx_validity( + bob.client.as_ref().info().best_hash, + bob.client.clone(), + storage_keys_for_tx_validity.clone(), + ); + + let header = bob + .client + .as_ref() + .header(bob.client.as_ref().info().best_hash); + let state_root = header.unwrap().unwrap().state_root; + + let wasm_bundle = ferdie + .client + .runtime_api() + .domain_runtime_code(ferdie.client.as_ref().info().best_hash, GENESIS_DOMAIN_ID) + .unwrap() + .unwrap(); + + let mut domain_stateless_runtime = + StatelessRuntime::::new(ferdie.executor.clone().into(), wasm_bundle.into()); + + let db = storage_proof.into_memory_db::(); + let mut top_storage_map = BTreeMap::new(); + for storage_key in storage_keys_for_tx_validity.iter() { + let storage_value = read_trie_value::, _>( + &db, + &state_root, + storage_key, + None, + None, + ) + .unwrap() + .unwrap(); + top_storage_map.insert(storage_key.to_vec(), storage_value); + } + + let storage = Storage { + top: top_storage_map, + children_default: Default::default(), + }; + + domain_stateless_runtime.set_storage(storage); + + assert_eq!( + domain_stateless_runtime + .check_transaction_validity_and_do_pre_dispatch( + &opaque_extrinsic, + bob.client.as_ref().info().best_number, + bob.client.as_ref().info().best_hash + ) + .unwrap(), + expected_out + ); + } +} + // #[tokio::test(flavor = "multi_thread")] // #[ignore] // async fn execution_proof_creation_and_verification_should_work() { diff --git a/domains/client/block-preprocessor/src/stateless_runtime.rs b/domains/client/block-preprocessor/src/stateless_runtime.rs index 946e511dc0..84ce633834 100644 --- a/domains/client/block-preprocessor/src/stateless_runtime.rs +++ b/domains/client/block-preprocessor/src/stateless_runtime.rs @@ -226,4 +226,19 @@ where block_hash, ) } + + pub fn check_transaction_validity_and_do_pre_dispatch( + &self, + uxt: &::Extrinsic, + block_number: NumberFor, + block_hash: ::Hash, + ) -> Result, ApiError> { + >::check_transaction_validity_and_do_pre_dispatch( + self, + Default::default(), + uxt, + block_number, + block_hash, + ) + } } diff --git a/domains/primitives/runtime/src/lib.rs b/domains/primitives/runtime/src/lib.rs index e3ce4bfa70..c471ebf4ea 100644 --- a/domains/primitives/runtime/src/lib.rs +++ b/domains/primitives/runtime/src/lib.rs @@ -163,8 +163,29 @@ pub enum VerifyTxValidityError { FailedToDecodeAccountId, } +#[derive(Debug, PartialEq, Eq, Encode, Decode, TypeInfo)] +pub enum CheckBundleValidityError { + /// One of the transaction is invalid. + InvalidTransaction { + /// Index of the transaction in bundle's transaction list + tx_index: u32, + /// Tx validity error of the tx referenced by `tx_index` + tx_validity_error: CheckTxValidityError, + }, +} + +impl CheckBundleValidityError { + pub fn from_tx_validity_error(tx_index: u32, tx_validity_error: CheckTxValidityError) -> Self { + Self::InvalidTransaction { + tx_index, + tx_validity_error, + } + } +} + sp_api::decl_runtime_apis! { /// Base API that every domain runtime must implement. + #[api_version(2)] pub trait DomainCoreApi { /// Extracts the optional signer per extrinsic. fn extract_signer( @@ -195,20 +216,50 @@ sp_api::decl_runtime_apis! { /// Returns true if the extrinsic is an inherent extrinsic. fn is_inherent_extrinsic(extrinsic: &::Extrinsic) -> bool; - /// Checks the validity of extrinsic in a bundle. + /// Checks the validity of extrinsic. + #[changed_in(2)] + fn check_transaction_validity( + uxt: &::Extrinsic, + block_number: NumberFor, + block_hash: ::Hash + ) -> Result<(), CheckTxValidityError>; + + /// Checks the validity of extrinsic + /// v2 can check validity of unsigned and self contained extrinsics as well. fn check_transaction_validity( uxt: &::Extrinsic, block_number: NumberFor, block_hash: ::Hash ) -> Result<(), CheckTxValidityError>; + /// Check transaction validity and also perform pre dispatch on transaction. + /// This is useful to maintain side effect in the RuntimeApi instance's storage buffer. + fn check_transaction_validity_and_do_pre_dispatch( + uxt: &::Extrinsic, + block_number: NumberFor, + block_hash: ::Hash + ) -> Result<(), CheckTxValidityError>; + /// Returns the storage keys of states accessed in the API `check_transaction_validity`. + #[changed_in(2)] fn storage_keys_for_verifying_transaction_validity( account_id: opaque::AccountId, block_number: NumberFor, tx_era: Option, ) -> Result>, VerifyTxValidityError>; + /// Returns the storage keys of states accessed in the API `check_transaction_validity` + /// In v2, signer account id is optional to support getting storage keys for unsigned tx as well. + fn storage_keys_for_verifying_transaction_validity( + maybe_account_id: Option, + block_number: NumberFor, + tx_era: Option, + ) -> Result>, VerifyTxValidityError>; + + /// Checks validity of bundle extrinsics. Internally calls `check_transaction_validity_and_do_pre_dispatch` + fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: NumberFor, + block_hash: ::Hash) -> Result<(), CheckBundleValidityError>; + /// Returns extrinsic Era if present fn extrinsic_era( extrinsic: &::Extrinsic diff --git a/domains/runtime/evm/src/lib.rs b/domains/runtime/evm/src/lib.rs index c5c57b2703..fd442a3b71 100644 --- a/domains/runtime/evm/src/lib.rs +++ b/domains/runtime/evm/src/lib.rs @@ -13,7 +13,7 @@ use domain_runtime_primitives::opaque::Header; pub use domain_runtime_primitives::{opaque, Balance, BlockNumber, Hash, Nonce}; use domain_runtime_primitives::{MultiAccountId, TryConvertBack, SLOT_DURATION}; use fp_account::EthereumSignature; -use fp_self_contained::SelfContainedCall; +use fp_self_contained::{CheckedSignature, SelfContainedCall}; use frame_support::dispatch::{DispatchClass, GetDispatchInfo}; use frame_support::inherent::ProvideInherent; use frame_support::traits::{ @@ -46,11 +46,12 @@ use sp_messenger::messages::{ }; use sp_runtime::generic::Era; use sp_runtime::traits::{ - BlakeTwo256, Block as BlockT, Convert, DispatchInfoOf, Dispatchable, IdentifyAccount, - IdentityLookup, PostDispatchInfoOf, UniqueSaturatedInto, Verify, + BlakeTwo256, Block as BlockT, Checkable, Convert, DispatchInfoOf, Dispatchable, + IdentifyAccount, IdentityLookup, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, + ValidateUnsigned, Verify, }; use sp_runtime::transaction_validity::{ - TransactionSource, TransactionValidity, TransactionValidityError, + InvalidTransaction, TransactionSource, TransactionValidity, TransactionValidityError, }; use sp_runtime::{ create_runtime_str, generic, impl_opaque_keys, ApplyExtrinsicResult, ConsensusEngineId, Digest, @@ -185,7 +186,7 @@ pub const VERSION: RuntimeVersion = RuntimeVersion { spec_name: create_runtime_str!("subspace-evm-domain"), impl_name: create_runtime_str!("subspace-evm-domain"), authoring_version: 0, - spec_version: 0, + spec_version: 1, impl_version: 0, apis: RUNTIME_API_VERSIONS, transaction_version: 0, @@ -704,17 +705,20 @@ pub fn extract_signer( } fn storage_keys_for_verifying_tx_validity_inner( - sender: AccountId, + maybe_sender: Option, block_number: BlockNumber, maybe_tx_era: Option, ) -> Vec> { let mut storage_keys = sp_std::vec![ frame_system::BlockHash::::hashed_key_for(BlockNumber::from(0u32)), frame_support::storage::storage_prefix(System::name().as_bytes(), b"Number").to_vec(), - frame_system::Account::::hashed_key_for(sender), pallet_transaction_payment::NextFeeMultiplier::::hashed_key().to_vec() ]; + if let Some(sender) = maybe_sender { + storage_keys.push(frame_system::Account::::hashed_key_for(sender)); + } + match maybe_tx_era { None => storage_keys, Some(era) => { @@ -729,6 +733,52 @@ fn storage_keys_for_verifying_tx_validity_inner( } } +fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { + extrinsic + .0 + .signature + .as_ref() + .map(|(_, _, extra)| extra.4 .0) +} + +fn check_transaction_validity_inner( + lookup: &Lookup, + uxt: &::Extrinsic, + block_number: BlockNumber, + block_hash: ::Hash, +) -> Result, domain_runtime_primitives::CheckTxValidityError> +where + Lookup: sp_runtime::traits::Lookup, +{ + let maybe_signer_info = extract_signer_inner(uxt, lookup); + let maybe_signer = if let Some(signer_info) = maybe_signer_info { + let signer = signer_info.map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { + error: tx_validity_error, + } + })?; + Some(signer) + } else { + None + }; + + let tx_validity = + Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); + + tx_validity + .map(|_| maybe_signer) + .map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys_for_verifying_tx_validity_inner( + maybe_signer, + block_number, + extrinsic_era(uxt), + ), + } + }) +} + #[cfg(feature = "runtime-benchmarks")] mod benches { frame_benchmarking::define_benchmarks!( @@ -908,49 +958,98 @@ impl_runtime_apis! { } } - fn check_transaction_validity( + fn check_transaction_validity(uxt: &::Extrinsic, + block_number: BlockNumber, + block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { + let lookup = frame_system::ChainContext::::default(); + check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) + } + + fn check_transaction_validity_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { let lookup = frame_system::ChainContext::::default(); - let maybe_signer_info = extract_signer_inner(uxt, &lookup); - - if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error - } - })?; - - let tx_validity = - Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); - tx_validity.map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys_for_verifying_tx_validity_inner(signer, block_number, Self::extrinsic_era(uxt)), + let maybe_signer = check_transaction_validity_inner(&lookup, uxt, block_number, block_hash)?; + + let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); + + let xt = uxt.clone().check(&lookup).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys.clone(), + } + })?; + + let dispatch_info = xt.get_dispatch_info(); + + let encoded = uxt.encode(); + let encoded_len = encoded.len(); + + // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) as that will add the side effect of SignedExtension in the storage buffer + // which would help to maintain context across multiple transaction validity check against same + // runtime instance. + match xt.signed { + CheckedSignature::Signed(account_id, extra) => { + extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys, + } + }) + }, + CheckedSignature::Unsigned => { + Runtime::pre_dispatch(&xt.function).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys.clone(), + } + })?; + SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys, + } + }) + }, + CheckedSignature::SelfContained(self_contained_signing_info) => { + xt.function.pre_dispatch_self_contained(&self_contained_signing_info, &dispatch_info, encoded_len).ok_or(TransactionValidityError::Invalid( + InvalidTransaction::BadProof, + )).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + // TODO: Verify the storage keys here + storage_keys, + } + }) } - }) - } else { - Ok(()) - } + } } fn storage_keys_for_verifying_transaction_validity( - who: opaque::AccountId, + maybe_who: Option, block_number: BlockNumber, maybe_tx_era: Option ) -> Result>, domain_runtime_primitives::VerifyTxValidityError> { - let sender = AccountId::decode(&mut who.as_slice()) - .map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)?; - Ok(storage_keys_for_verifying_tx_validity_inner(sender, block_number, maybe_tx_era)) + let maybe_sender = maybe_who.map(|who| AccountId::decode(&mut who.as_slice()).map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)).transpose()?; + Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) + } + + /// This runtime api is equivalent of `execute_block` but only for bundle extrinsic validation. + fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, + block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { + for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { + Self::check_transaction_validity_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; + } + Ok(()) } fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { - extrinsic.0.signature.as_ref().map(|(_, _, extra)| extra.4.0) + extrinsic_era(extrinsic) } fn extrinsic_weight(ext: &::Extrinsic) -> Weight { diff --git a/domains/test/runtime/evm/src/lib.rs b/domains/test/runtime/evm/src/lib.rs index ac057ee8d2..cd555b5997 100644 --- a/domains/test/runtime/evm/src/lib.rs +++ b/domains/test/runtime/evm/src/lib.rs @@ -13,7 +13,7 @@ pub use domain_runtime_primitives::opaque::Header; pub use domain_runtime_primitives::{opaque, Balance, BlockNumber, Hash, Nonce}; use domain_runtime_primitives::{MultiAccountId, TryConvertBack, SLOT_DURATION}; use fp_account::EthereumSignature; -use fp_self_contained::SelfContainedCall; +use fp_self_contained::{CheckedSignature, SelfContainedCall}; use frame_support::dispatch::{DispatchClass, GetDispatchInfo}; use frame_support::inherent::ProvideInherent; use frame_support::traits::{ @@ -46,11 +46,12 @@ use sp_messenger::messages::{ }; use sp_runtime::generic::Era; use sp_runtime::traits::{ - BlakeTwo256, Block as BlockT, Convert, DispatchInfoOf, Dispatchable, IdentifyAccount, - IdentityLookup, PostDispatchInfoOf, UniqueSaturatedInto, Verify, + BlakeTwo256, Block as BlockT, Checkable, Convert, DispatchInfoOf, Dispatchable, + IdentifyAccount, IdentityLookup, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, + ValidateUnsigned, Verify, }; use sp_runtime::transaction_validity::{ - TransactionSource, TransactionValidity, TransactionValidityError, + InvalidTransaction, TransactionSource, TransactionValidity, TransactionValidityError, }; use sp_runtime::{ create_runtime_str, generic, impl_opaque_keys, ApplyExtrinsicResult, ConsensusEngineId, Digest, @@ -703,17 +704,20 @@ pub fn extract_signer( } fn storage_keys_for_verifying_tx_validity_inner( - sender: AccountId, + maybe_sender: Option, block_number: BlockNumber, maybe_tx_era: Option, ) -> Vec> { let mut storage_keys = sp_std::vec![ frame_system::BlockHash::::hashed_key_for(BlockNumber::from(0u32)), frame_support::storage::storage_prefix(System::name().as_bytes(), b"Number").to_vec(), - frame_system::Account::::hashed_key_for(sender), pallet_transaction_payment::NextFeeMultiplier::::hashed_key().to_vec() ]; + if let Some(sender) = maybe_sender { + storage_keys.push(frame_system::Account::::hashed_key_for(sender)); + } + match maybe_tx_era { None => storage_keys, Some(era) => { @@ -728,6 +732,52 @@ fn storage_keys_for_verifying_tx_validity_inner( } } +fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { + extrinsic + .0 + .signature + .as_ref() + .map(|(_, _, extra)| extra.4 .0) +} + +fn check_transaction_validity_inner( + lookup: &Lookup, + uxt: &::Extrinsic, + block_number: BlockNumber, + block_hash: ::Hash, +) -> Result, domain_runtime_primitives::CheckTxValidityError> +where + Lookup: sp_runtime::traits::Lookup, +{ + let maybe_signer_info = extract_signer_inner(uxt, lookup); + let maybe_signer = if let Some(signer_info) = maybe_signer_info { + let signer = signer_info.map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { + error: tx_validity_error, + } + })?; + Some(signer) + } else { + None + }; + + let tx_validity = + Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); + + tx_validity + .map(|_| maybe_signer) + .map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys_for_verifying_tx_validity_inner( + maybe_signer, + block_number, + extrinsic_era(uxt), + ), + } + }) +} + impl_runtime_apis! { impl sp_api::Core for Runtime { fn version() -> RuntimeVersion { @@ -896,49 +946,97 @@ impl_runtime_apis! { } } - fn check_transaction_validity( + fn check_transaction_validity(uxt: &::Extrinsic, + block_number: BlockNumber, + block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { + let lookup = frame_system::ChainContext::::default(); + check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) + } + + fn check_transaction_validity_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { let lookup = frame_system::ChainContext::::default(); - let maybe_signer_info = extract_signer_inner(uxt, &lookup); - - if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error - } - })?; - - let tx_validity = - Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); - tx_validity.map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys_for_verifying_tx_validity_inner(signer, block_number, Self::extrinsic_era(uxt)), + let maybe_signer = check_transaction_validity_inner(&lookup, uxt, block_number, block_hash)?; + + let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); + + let xt = uxt.clone().check(&lookup).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys.clone(), + } + })?; + + let dispatch_info = xt.get_dispatch_info(); + + let encoded = uxt.encode(); + let encoded_len = encoded.len(); + + // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) as that will add the side effect of SignedExtension in the storage buffer + // which would help to maintain context across multiple transaction validity check against same + // runtime instance. + match xt.signed { + CheckedSignature::Signed(account_id, extra) => { + extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys, + } + }) + }, + CheckedSignature::Unsigned => { + Runtime::pre_dispatch(&xt.function).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys: storage_keys.clone(), + } + })?; + SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + storage_keys, + } + }) + }, + CheckedSignature::SelfContained(self_contained_signing_info) => { + xt.function.pre_dispatch_self_contained(&self_contained_signing_info, &dispatch_info, encoded_len).ok_or(TransactionValidityError::Invalid( + InvalidTransaction::BadProof, + )).map(|_| ()).map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: tx_validity_error, + // TODO: Verify the storage keys here + storage_keys, + } + }) } - }) - } else { - Ok(()) - } + } } fn storage_keys_for_verifying_transaction_validity( - who: opaque::AccountId, + maybe_who: Option, block_number: BlockNumber, maybe_tx_era: Option ) -> Result>, domain_runtime_primitives::VerifyTxValidityError> { - let sender = AccountId::decode(&mut who.as_slice()) - .map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)?; - Ok(storage_keys_for_verifying_tx_validity_inner(sender, block_number, maybe_tx_era)) + let maybe_sender = maybe_who.map(|who| AccountId::decode(&mut who.as_slice()).map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)).transpose()?; + Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) + } + + fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, + block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { + for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { + Self::check_transaction_validity_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; + } + Ok(()) } fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { - extrinsic.0.signature.as_ref().map(|(_, _, extra)| extra.4.0) + extrinsic_era(extrinsic) } fn extrinsic_weight(ext: &::Extrinsic) -> Weight { From 11a2f8a2386f778f8515d77fb87986afb0ccafac Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Thu, 23 Nov 2023 01:52:59 +0400 Subject: [PATCH 2/7] modifying domain pre-processor and domain proposer to use new runtime api to check illegaltx --- domains/client/block-preprocessor/src/lib.rs | 33 +++-- .../src/domain_bundle_proposer.rs | 139 +++++++++--------- 2 files changed, 93 insertions(+), 79 deletions(-) diff --git a/domains/client/block-preprocessor/src/lib.rs b/domains/client/block-preprocessor/src/lib.rs index 198e431eb5..1cc28df7b7 100644 --- a/domains/client/block-preprocessor/src/lib.rs +++ b/domains/client/block-preprocessor/src/lib.rs @@ -292,19 +292,6 @@ where ))); } - let is_legal_tx = self - .client - .runtime_api() - .check_transaction_validity(at, &extrinsic, domain_block_number, at)? - .is_ok(); - - if !is_legal_tx { - // TODO: Generate a fraud proof for this invalid bundle - return Ok(BundleValidity::Invalid(InvalidBundleType::IllegalTx( - index as u32, - ))); - } - // Check if this extrinsic is an inherent extrinsic. // If so, this is an invalid bundle since these extrinsics should not be included in the // bundle. Extrinsic is always decodable due to the check above. @@ -314,6 +301,26 @@ where )); } + // Using one instance of runtime_api throughout the loop in order to maintain context + // between them. + // Using `check_transaction_validity_and_do_pre_dispatch` instead of `check_transaction_validity` + // to maintain side-effect in the storage buffer. + let is_legal_tx = runtime_api + .check_transaction_validity_and_do_pre_dispatch( + at, + &extrinsic, + domain_block_number, + at, + )? + .is_ok(); + + if !is_legal_tx { + // TODO: Generate a fraud proof for this invalid bundle + return Ok(BundleValidity::Invalid(InvalidBundleType::IllegalTx( + index as u32, + ))); + } + // TODO: the behavior is changed, as before invalid XDM will be dropped silently, // and the other extrinsic of the bundle will be continue processed, now the whole // bundle is considered as invalid and excluded from further processing. diff --git a/domains/client/domain-operator/src/domain_bundle_proposer.rs b/domains/client/domain-operator/src/domain_bundle_proposer.rs index 971024a1fb..8ce38ea082 100644 --- a/domains/client/domain-operator/src/domain_bundle_proposer.rs +++ b/domains/client/domain-operator/src/domain_bundle_proposer.rs @@ -3,8 +3,8 @@ use codec::Encode; use domain_runtime_primitives::DomainCoreApi; use futures::{select, FutureExt}; use sc_client_api::{AuxStore, BlockBackend}; -use sc_transaction_pool_api::{InPoolTransaction, TransactionSource}; -use sp_api::{HeaderT, NumberFor, ProvideRuntimeApi}; +use sc_transaction_pool_api::InPoolTransaction; +use sp_api::{ApiExt, HeaderT, NumberFor, ProvideRuntimeApi}; use sp_block_builder::BlockBuilder; use sp_blockchain::HeaderBackend; use sp_domains::{ @@ -109,74 +109,81 @@ where let mut extrinsics = Vec::new(); let mut estimated_bundle_weight = Weight::default(); let mut bundle_size = 0u32; - for pending_tx in pending_iterator { - let pending_tx_data = pending_tx.data(); - - let is_within_tx_range = self - .client - .runtime_api() - .is_within_tx_range(parent_hash, pending_tx_data, &bundle_vrf_hash, &tx_range) - .map_err(|err| { - tracing::error!( - ?err, - ?pending_tx_data, - "Error occurred in locating the tx range" - ); - }) - .unwrap_or(false); - if !is_within_tx_range { - continue; - } - // Double check the transaction validity, because the tx pool are re-validate the transaction - // in pool asynchronously so there is race condition that the operator imported a domain block - // and start producing bundle immediately before the re-validation based on the latest block - // is finished, cause the bundle contains illegal tx accidentally and being considered as invalid - // bundle and slashing on the honest operator. - // - // NOTE: this check need to be kept aligned with how the illegal tx is detected in the block-preprocessor - // thus when https://github.com/subspace/subspace/issues/2184 is implemented, we also need to - // perfome the check for bundle as a whole instead of checking tx one by one. - let transaction_validity = self - .client - .runtime_api() - .validate_transaction( - parent_hash, - TransactionSource::External, - pending_tx_data.clone(), - parent_hash, - ) - .map_err(|error| { - sp_blockchain::Error::Application(Box::from(format!( - "Error getting transaction validity: {error}" - ))) - })?; - if transaction_validity.is_err() { - continue; - } + // Seperate code block to make sure that runtime api instance is dropped after validation is done. + { + // We are using one runtime api instance here to maintain storage changes in the instance's internal buffer + // between runtime calls done in this loop. + let runtime_api_instance = self.client.runtime_api(); + for pending_tx in pending_iterator { + let pending_tx_data = pending_tx.data(); - let tx_weight = self - .client - .runtime_api() - .extrinsic_weight(parent_hash, pending_tx_data) - .map_err(|error| { - sp_blockchain::Error::Application(Box::from(format!( - "Error getting extrinsic weight: {error}" - ))) - })?; - let next_estimated_bundle_weight = estimated_bundle_weight.saturating_add(tx_weight); - if next_estimated_bundle_weight.any_gt(domain_block_limit.max_block_weight) { - break; - } + let is_within_tx_range = runtime_api_instance + .is_within_tx_range(parent_hash, pending_tx_data, &bundle_vrf_hash, &tx_range) + .map_err(|err| { + tracing::error!( + ?err, + ?pending_tx_data, + "Error occurred in locating the tx range" + ); + }) + .unwrap_or(false); + if !is_within_tx_range { + continue; + } - let next_bundle_size = bundle_size + pending_tx_data.encoded_size() as u32; - if next_bundle_size > domain_block_limit.max_block_size { - break; - } + let tx_weight = runtime_api_instance + .extrinsic_weight(parent_hash, pending_tx_data) + .map_err(|error| { + sp_blockchain::Error::Application(Box::from(format!( + "Error getting extrinsic weight: {error}" + ))) + })?; + let next_estimated_bundle_weight = + estimated_bundle_weight.saturating_add(tx_weight); + if next_estimated_bundle_weight.any_gt(domain_block_limit.max_block_weight) { + break; + } - estimated_bundle_weight = next_estimated_bundle_weight; - bundle_size = next_bundle_size; - extrinsics.push(pending_tx_data.clone()); + let next_bundle_size = bundle_size + pending_tx_data.encoded_size() as u32; + if next_bundle_size > domain_block_limit.max_block_size { + break; + } + + estimated_bundle_weight = next_estimated_bundle_weight; + bundle_size = next_bundle_size; + + // Double check the transaction validity, because the tx pool are re-validate the transaction + // in pool asynchronously so there is race condition that the operator imported a domain block + // and start producing bundle immediately before the re-validation based on the latest block + // is finished, cause the bundle contains illegal tx accidentally and being considered as invalid + // bundle and slashing on the honest operator. + // + // This check is done in similar fashion to block builder's build block. + // This check needs to be the last check as otherwise if the tx won't be part of bundle due to + // some other checks, its side effect will still be part of RuntimeApiImpl's changes buffer. + let transaction_validity_result = + runtime_api_instance.execute_in_transaction(|api| { + let transaction_validity_result = api + .check_transaction_validity_and_do_pre_dispatch( + parent_hash, + pending_tx_data, + parent_number, + parent_hash, + ); + // Only commit, if there are no errors (both ApiError and CheckTxValidityError) + if let Ok(Ok(_)) = transaction_validity_result { + sp_api::TransactionOutcome::Commit(transaction_validity_result) + } else { + sp_api::TransactionOutcome::Rollback(transaction_validity_result) + } + })?; + if transaction_validity_result.is_err() { + continue; + } + + extrinsics.push(pending_tx_data.clone()); + } } let extrinsics_root = HeaderHashingFor::::ordered_trie_root( From 69b02cd7ea558d787cd3610c151ad58cd50f2b75 Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Tue, 28 Nov 2023 13:22:41 +0400 Subject: [PATCH 3/7] apply review suggestions --- crates/sp-domains-fraud-proof/src/tests.rs | 267 +++++++++++++++--- crates/sp-domains/src/lib.rs | 6 +- domains/client/block-preprocessor/src/lib.rs | 27 +- .../src/stateless_runtime.rs | 6 +- .../src/domain_bundle_proposer.rs | 27 +- domains/primitives/runtime/src/lib.rs | 8 +- domains/runtime/evm/src/lib.rs | 34 ++- domains/test/runtime/evm/src/lib.rs | 32 ++- 8 files changed, 348 insertions(+), 59 deletions(-) diff --git a/crates/sp-domains-fraud-proof/src/tests.rs b/crates/sp-domains-fraud-proof/src/tests.rs index 6c828451e2..2df989525c 100644 --- a/crates/sp-domains-fraud-proof/src/tests.rs +++ b/crates/sp-domains-fraud-proof/src/tests.rs @@ -9,7 +9,7 @@ use domain_test_service::{construct_extrinsic_generic, EvmDomainNode, GENESIS_DO use fp_rpc::EthereumRuntimeRPCApi; use sc_client_api::{HeaderBackend, ProofProvider, StorageProof}; use sc_service::{BasePath, Role}; -use sp_api::{BlockT, ProvideRuntimeApi}; +use sp_api::{ApiExt, BlockT, ProvideRuntimeApi, TransactionOutcome}; use sp_domains::DomainsApi; use sp_runtime::generic::UncheckedExtrinsic; use sp_runtime::traits::{BlakeTwo256, Header as HeaderT}; @@ -84,6 +84,235 @@ fn get_storage_keys_for_verifying_signed_tx_validity( .unwrap() } +#[tokio::test(flavor = "multi_thread")] +async fn runtime_instance_assumptions_are_correct() { + let directory = TempDir::new().expect("Must be able to create temporary directory"); + + let mut builder = sc_cli::LoggerBuilder::new(""); + builder.with_colors(false); + let _ = builder.init(); + + let tokio_handle = tokio::runtime::Handle::current(); + + // Start Ferdie + let mut ferdie = MockConsensusNode::run( + tokio_handle.clone(), + Ferdie, + BasePath::new(directory.path().join("ferdie")), + ); + + // Run Alice (a evm domain authority node) + let alice = domain_test_service::DomainNodeBuilder::new( + tokio_handle.clone(), + Alice, + BasePath::new(directory.path().join("alice")), + ) + .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) + .await; + + // transfer most of the alice's balance + let alice_balance = alice + .client + .runtime_api() + .account_basic( + alice.client.as_ref().info().best_hash, + Alice.to_account_id().into(), + ) + .unwrap() + .balance + .as_u128(); + let target_balance = u32::MAX - 100; + let balance_to_transfer = alice_balance - target_balance as u128; + + let alice_balance_transfer_extrinsic = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: balance_to_transfer, + }, + alice.key, + false, + 0, + 1, + ); + + alice + .send_extrinsic(alice_balance_transfer_extrinsic) + .await + .expect("Failed to send extrinsic"); + + let (slot, bundle) = ferdie.produce_slot_and_wait_for_bundle_submission().await; + assert!(bundle.is_some()); + produce_block_with!(ferdie.produce_block_with_slot(slot), alice) + .await + .unwrap(); + produce_blocks!(ferdie, alice, 10).await.unwrap(); + + let mut alice_nonce = 1; + + let transfer_to_charlie_with_big_tip_1 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 2000000000u32, + ); + alice_nonce += 1; + + let transfer_to_charlie_with_big_tip_2 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 2000000000u32, + ); + + // A runtime instance preserve changes in between + let runtime_instance = alice.client.runtime_api(); + + let tx_validity_error = CheckTxValidityError::InvalidTransaction { + error: TransactionValidityError::Invalid(InvalidTransaction::Payment), + storage_keys: get_storage_keys_for_verifying_signed_tx_validity( + &alice, + transfer_to_charlie_with_big_tip_2.clone(), + ), + }; + + // First transaction should be okay + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_to_charlie_with_big_tip_1.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + + // Second transaction should error out with exact error encountered during check of bundle validity + // to prove that state is preserved between two calls to same runtime instance + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_to_charlie_with_big_tip_2.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err(CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()).unwrap()) + ); + + let test_commit_mode = vec![true, false]; + + for commit_mode in test_commit_mode { + // A runtime instance can rollback changes safely. + let runtime_instance = alice.client.runtime_api(); + + alice_nonce = 1; + + let transfer_with_big_tip_1 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 4000000000u32 / 3, + ); + alice_nonce += 1; + + let transfer_with_big_tip_2 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 4000000000u32 / 3, + ); + + if commit_mode { + alice_nonce += 1; + } + + let transfer_with_big_tip_3 = construct_extrinsic_generic::( + &alice.client, + pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }, + alice.key, + false, + alice_nonce, + 4000000000u32 / 3, + ); + + assert!(runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_with_big_tip_1.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash + ) + .unwrap() + .is_ok()); + + assert!(runtime_instance + .execute_in_transaction(|api| { + if commit_mode { + TransactionOutcome::Commit(api.check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_with_big_tip_2.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + )) + } else { + TransactionOutcome::Rollback(api.check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_with_big_tip_2.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + )) + } + }) + .is_ok()); + + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &transfer_with_big_tip_3.clone().into(), + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash + ) + .unwrap(), + if commit_mode { + Err( + CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()) + .unwrap(), + ) + } else { + Ok(()) + } + ) + } +} + #[tokio::test(flavor = "multi_thread")] async fn check_bundle_validity_runtime_api_should_work() { let directory = TempDir::new().expect("Must be able to create temporary directory"); @@ -261,7 +490,7 @@ async fn check_bundle_validity_runtime_api_should_work() { alice.key, false, alice_nonce, - 1000000000u32, + 2000000000u32, ); alice_nonce += 1; @@ -274,7 +503,7 @@ async fn check_bundle_validity_runtime_api_should_work() { alice.key, false, alice_nonce, - 1000000000u32, + 2000000000u32, ); // In single tx context both are valid (both tx are identical except nonce), so no need to test both of them independently. @@ -323,36 +552,6 @@ async fn check_bundle_validity_runtime_api_should_work() { .unwrap(), }) ); - - // A runtime instance preserve changes in between - let runtime_instance = alice.client.runtime_api(); - - // First transaction should be okay - assert_eq!( - runtime_instance - .check_transaction_validity_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, - &transfer_to_charlie_with_big_tip_1.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, - ) - .unwrap(), - Ok(()) - ); - - // Second transaction should error out with exact error encountered during check of bundle validity - // to prove that state is preserved between two calls to same runtime instance - assert_eq!( - runtime_instance - .check_transaction_validity_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, - &transfer_to_charlie_with_big_tip_2.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, - ) - .unwrap(), - Err(tx_validity_error) - ); } #[tokio::test(flavor = "multi_thread")] @@ -716,7 +915,7 @@ async fn check_tx_validity_and_do_pre_dispatch_runtime_api_should_work() { assert_eq!( domain_stateless_runtime - .check_transaction_validity_and_do_pre_dispatch( + .check_transaction_and_do_pre_dispatch( &opaque_extrinsic, bob.client.as_ref().info().best_number, bob.client.as_ref().info().best_hash diff --git a/crates/sp-domains/src/lib.rs b/crates/sp-domains/src/lib.rs index 55d3c482f1..f166cbdba9 100644 --- a/crates/sp-domains/src/lib.rs +++ b/crates/sp-domains/src/lib.rs @@ -812,9 +812,9 @@ impl InvalidBundleType { match self { Self::UndecodableTx(_) => 1, Self::OutOfRangeTx(_) => 2, - Self::IllegalTx(_) => 3, - Self::InvalidXDM(_) => 4, - Self::InherentExtrinsic(_) => 5, + Self::InherentExtrinsic(_) => 3, + Self::IllegalTx(_) => 4, + Self::InvalidXDM(_) => 5, } } diff --git a/domains/client/block-preprocessor/src/lib.rs b/domains/client/block-preprocessor/src/lib.rs index 1cc28df7b7..302d52ef53 100644 --- a/domains/client/block-preprocessor/src/lib.rs +++ b/domains/client/block-preprocessor/src/lib.rs @@ -20,7 +20,7 @@ use codec::{Decode, Encode}; use domain_runtime_primitives::opaque::AccountId; use domain_runtime_primitives::DomainCoreApi; use sc_client_api::BlockBackend; -use sp_api::{HashT, ProvideRuntimeApi}; +use sp_api::{ApiExt, HashT, ProvideRuntimeApi}; use sp_blockchain::HeaderBackend; use sp_core::H256; use sp_domains::extrinsics::deduplicate_and_shuffle_extrinsics; @@ -268,6 +268,14 @@ where // // NOTE: for each extrinsic the checking order must follow `InvalidBundleType::checking_order` let runtime_api = self.client.runtime_api(); + let api_version = runtime_api + .api_version::>(at) + .map_err(sp_blockchain::Error::RuntimeApiError)? + .ok_or_else(|| { + sp_blockchain::Error::Application( + format!("Could not find `DomainCoreApi` api for block `{:?}`.", at).into(), + ) + })?; for (index, opaque_extrinsic) in bundle.extrinsics.iter().enumerate() { let Ok(extrinsic) = <::Extrinsic>::decode(&mut opaque_extrinsic.encode().as_slice()) @@ -303,16 +311,25 @@ where // Using one instance of runtime_api throughout the loop in order to maintain context // between them. - // Using `check_transaction_validity_and_do_pre_dispatch` instead of `check_transaction_validity` + // Using `check_transaction_and_do_pre_dispatch` instead of `check_transaction_validity` // to maintain side-effect in the storage buffer. - let is_legal_tx = runtime_api - .check_transaction_validity_and_do_pre_dispatch( + let is_legal_tx = if api_version == 2 { + runtime_api.check_transaction_and_do_pre_dispatch( at, &extrinsic, domain_block_number, at, )? - .is_ok(); + } else { + #[allow(deprecated)] // old check_transaction_validity + runtime_api.check_transaction_validity_before_version_2( + at, + &extrinsic, + domain_block_number, + at, + )? + } + .is_ok(); if !is_legal_tx { // TODO: Generate a fraud proof for this invalid bundle diff --git a/domains/client/block-preprocessor/src/stateless_runtime.rs b/domains/client/block-preprocessor/src/stateless_runtime.rs index 84ce633834..1e5f2eed63 100644 --- a/domains/client/block-preprocessor/src/stateless_runtime.rs +++ b/domains/client/block-preprocessor/src/stateless_runtime.rs @@ -212,6 +212,7 @@ where ) } + /// This is stateful runtime api call and require setting of storage keys. pub fn check_transaction_validity( &self, uxt: &::Extrinsic, @@ -227,13 +228,14 @@ where ) } - pub fn check_transaction_validity_and_do_pre_dispatch( + /// This is stateful runtime api call and require setting of storage keys. + pub fn check_transaction_and_do_pre_dispatch( &self, uxt: &::Extrinsic, block_number: NumberFor, block_hash: ::Hash, ) -> Result, ApiError> { - >::check_transaction_validity_and_do_pre_dispatch( + >::check_transaction_and_do_pre_dispatch( self, Default::default(), uxt, diff --git a/domains/client/domain-operator/src/domain_bundle_proposer.rs b/domains/client/domain-operator/src/domain_bundle_proposer.rs index 8ce38ea082..777ad89f22 100644 --- a/domains/client/domain-operator/src/domain_bundle_proposer.rs +++ b/domains/client/domain-operator/src/domain_bundle_proposer.rs @@ -115,6 +115,18 @@ where // We are using one runtime api instance here to maintain storage changes in the instance's internal buffer // between runtime calls done in this loop. let runtime_api_instance = self.client.runtime_api(); + let api_version = runtime_api_instance + .api_version::>(parent_hash) + .map_err(sp_blockchain::Error::RuntimeApiError)? + .ok_or_else(|| { + sp_blockchain::Error::Application( + format!( + "Could not find `DomainCoreApi` api for block `{:?}`.", + parent_hash + ) + .into(), + ) + })?; for pending_tx in pending_iterator { let pending_tx_data = pending_tx.data(); @@ -164,13 +176,22 @@ where // some other checks, its side effect will still be part of RuntimeApiImpl's changes buffer. let transaction_validity_result = runtime_api_instance.execute_in_transaction(|api| { - let transaction_validity_result = api - .check_transaction_validity_and_do_pre_dispatch( + let transaction_validity_result = if api_version == 2 { + api.check_transaction_and_do_pre_dispatch( parent_hash, pending_tx_data, parent_number, parent_hash, - ); + ) + } else { + #[allow(deprecated)] // old check_transaction_validity + api.check_transaction_validity_before_version_2( + parent_hash, + pending_tx_data, + parent_number, + parent_hash, + ) + }; // Only commit, if there are no errors (both ApiError and CheckTxValidityError) if let Ok(Ok(_)) = transaction_validity_result { sp_api::TransactionOutcome::Commit(transaction_validity_result) diff --git a/domains/primitives/runtime/src/lib.rs b/domains/primitives/runtime/src/lib.rs index c471ebf4ea..c1e72b6e50 100644 --- a/domains/primitives/runtime/src/lib.rs +++ b/domains/primitives/runtime/src/lib.rs @@ -232,9 +232,10 @@ sp_api::decl_runtime_apis! { block_hash: ::Hash ) -> Result<(), CheckTxValidityError>; - /// Check transaction validity and also perform pre dispatch on transaction. + /// New in v2. + /// Checks transaction and also perform pre dispatch on transaction. /// This is useful to maintain side effect in the RuntimeApi instance's storage buffer. - fn check_transaction_validity_and_do_pre_dispatch( + fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: NumberFor, block_hash: ::Hash @@ -256,7 +257,8 @@ sp_api::decl_runtime_apis! { tx_era: Option, ) -> Result>, VerifyTxValidityError>; - /// Checks validity of bundle extrinsics. Internally calls `check_transaction_validity_and_do_pre_dispatch` + /// New in v2. + /// Checks validity of bundle extrinsics. Internally calls `check_transaction_and_do_pre_dispatch` fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: NumberFor, block_hash: ::Hash) -> Result<(), CheckBundleValidityError>; diff --git a/domains/runtime/evm/src/lib.rs b/domains/runtime/evm/src/lib.rs index fd442a3b71..47f33ebdf1 100644 --- a/domains/runtime/evm/src/lib.rs +++ b/domains/runtime/evm/src/lib.rs @@ -47,7 +47,7 @@ use sp_messenger::messages::{ use sp_runtime::generic::Era; use sp_runtime::traits::{ BlakeTwo256, Block as BlockT, Checkable, Convert, DispatchInfoOf, Dispatchable, - IdentifyAccount, IdentityLookup, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, + IdentifyAccount, IdentityLookup, One, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, ValidateUnsigned, Verify, }; use sp_runtime::transaction_validity::{ @@ -965,14 +965,31 @@ impl_runtime_apis! { check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) } - fn check_transaction_validity_and_do_pre_dispatch( + fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { - let lookup = frame_system::ChainContext::::default(); + let lookup = frame_system::ChainContext::::default(); + + let maybe_signer_info = extract_signer_inner(uxt, &lookup); + let maybe_signer = if let Some(signer_info) = maybe_signer_info { + let signer = signer_info.map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { + error: tx_validity_error, + } + })?; + Some(signer) + } else { + None + }; - let maybe_signer = check_transaction_validity_inner(&lookup, uxt, block_number, block_hash)?; + // Initializing block related storage required for validation + System::initialize( + &(System::block_number() + BlockNumber::one()), + &block_hash, + &Default::default(), + ); let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); @@ -985,6 +1002,13 @@ impl_runtime_apis! { let dispatch_info = xt.get_dispatch_info(); + if dispatch_info.class == DispatchClass::Mandatory { + return Err( domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: InvalidTransaction::BadMandatory.into(), + storage_keys: storage_keys.clone(), + }) + } + let encoded = uxt.encode(); let encoded_len = encoded.len(); @@ -1041,7 +1065,7 @@ impl_runtime_apis! { fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { - Self::check_transaction_validity_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; + Self::check_transaction_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; } Ok(()) } diff --git a/domains/test/runtime/evm/src/lib.rs b/domains/test/runtime/evm/src/lib.rs index cd555b5997..e611dae8db 100644 --- a/domains/test/runtime/evm/src/lib.rs +++ b/domains/test/runtime/evm/src/lib.rs @@ -47,7 +47,7 @@ use sp_messenger::messages::{ use sp_runtime::generic::Era; use sp_runtime::traits::{ BlakeTwo256, Block as BlockT, Checkable, Convert, DispatchInfoOf, Dispatchable, - IdentifyAccount, IdentityLookup, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, + IdentifyAccount, IdentityLookup, One, PostDispatchInfoOf, SignedExtension, UniqueSaturatedInto, ValidateUnsigned, Verify, }; use sp_runtime::transaction_validity::{ @@ -953,14 +953,31 @@ impl_runtime_apis! { check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) } - fn check_transaction_validity_and_do_pre_dispatch( + fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { let lookup = frame_system::ChainContext::::default(); - let maybe_signer = check_transaction_validity_inner(&lookup, uxt, block_number, block_hash)?; + let maybe_signer_info = extract_signer_inner(uxt, &lookup); + let maybe_signer = if let Some(signer_info) = maybe_signer_info { + let signer = signer_info.map_err(|tx_validity_error| { + domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { + error: tx_validity_error, + } + })?; + Some(signer) + } else { + None + }; + + // Initializing block related storage required for validation + System::initialize( + &(System::block_number() + BlockNumber::one()), + &block_hash, + &Default::default(), + ); let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); @@ -973,6 +990,13 @@ impl_runtime_apis! { let dispatch_info = xt.get_dispatch_info(); + if dispatch_info.class == DispatchClass::Mandatory { + return Err( domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { + error: InvalidTransaction::BadMandatory.into(), + storage_keys: storage_keys.clone(), + }) + } + let encoded = uxt.encode(); let encoded_len = encoded.len(); @@ -1028,7 +1052,7 @@ impl_runtime_apis! { fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { - Self::check_transaction_validity_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; + Self::check_transaction_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; } Ok(()) } From d5f0a5dccb920ac9211529ac89bc74e709c3262f Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Fri, 1 Dec 2023 13:48:54 +0400 Subject: [PATCH 4/7] remove check_bundle_validity runtime api to simplify the implementation --- crates/sp-domains-fraud-proof/src/tests.rs | 157 +++++++++++++-------- domains/primitives/runtime/src/lib.rs | 25 ---- domains/runtime/evm/src/lib.rs | 9 -- domains/test/runtime/evm/src/lib.rs | 8 -- 4 files changed, 97 insertions(+), 102 deletions(-) diff --git a/crates/sp-domains-fraud-proof/src/tests.rs b/crates/sp-domains-fraud-proof/src/tests.rs index 2df989525c..09be993ea2 100644 --- a/crates/sp-domains-fraud-proof/src/tests.rs +++ b/crates/sp-domains-fraud-proof/src/tests.rs @@ -1,6 +1,6 @@ use codec::{Decode, Encode}; use domain_block_preprocessor::stateless_runtime::StatelessRuntime; -use domain_runtime_primitives::{CheckBundleValidityError, CheckTxValidityError, DomainCoreApi}; +use domain_runtime_primitives::{CheckTxValidityError, DomainCoreApi}; use domain_test_service::domain::EvmDomainClient as DomainClient; use domain_test_service::evm_domain_test_runtime::Runtime as TestRuntime; use domain_test_service::EcdsaKeyring::{Alice, Bob, Charlie}; @@ -381,19 +381,22 @@ async fn check_bundle_validity_runtime_api_should_work() { OpaqueExtrinsic::from_bytes(&transfer_to_charlie_2.encode()).unwrap(), ]; - assert_eq!( - alice - .client - .runtime_api() - .check_bundle_extrinsics_validity( - alice.client.as_ref().info().best_hash, - sample_valid_bundle_extrinsics, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, - ) - .unwrap(), - Ok(()) - ); + { + let runtime_instance = alice.client.runtime_api(); + for extrinsic in sample_valid_bundle_extrinsics { + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + &extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + } + } let transfer_to_charlie_3_duplicate_nonce_extrinsic = construct_extrinsic_generic::( @@ -416,28 +419,45 @@ async fn check_bundle_validity_runtime_api_should_work() { .unwrap(), ]; - assert_eq!( - alice - .client - .runtime_api() - .check_bundle_extrinsics_validity( - alice.client.as_ref().info().best_hash, - sample_invalid_bundle_with_same_nonce_extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, - ) - .unwrap(), - Err(CheckBundleValidityError::InvalidTransaction { - tx_index: 2, - tx_validity_error: CheckTxValidityError::InvalidTransaction { - error: TransactionValidityError::Invalid(InvalidTransaction::Stale), - storage_keys: get_storage_keys_for_verifying_signed_tx_validity( - &alice, - transfer_to_charlie_3_duplicate_nonce_extrinsic.clone() - ), - }, - }) - ); + { + let runtime_instance = alice.client.runtime_api(); + for (i, extrinsic) in sample_invalid_bundle_with_same_nonce_extrinsic + .iter() + .enumerate() + { + if i != 2 { + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + } else { + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err(CheckTxValidityError::InvalidTransaction { + error: TransactionValidityError::Invalid(InvalidTransaction::Stale), + storage_keys: get_storage_keys_for_verifying_signed_tx_validity( + &alice, + transfer_to_charlie_3_duplicate_nonce_extrinsic.clone() + ), + }) + ); + } + } + } // transfer most of the alice's balance let alice_balance = alice @@ -512,9 +532,9 @@ async fn check_bundle_validity_runtime_api_should_work() { alice .client .runtime_api() - .check_bundle_extrinsics_validity( + .check_transaction_and_do_pre_dispatch( alice.client.as_ref().info().best_hash, - vec![transfer_to_charlie_with_big_tip_1.clone().into(),], + &OpaqueExtrinsic::from_bytes(&transfer_to_charlie_with_big_tip_1.encode()).unwrap(), alice.client.as_ref().info().best_number, alice.client.as_ref().info().best_hash, ) @@ -530,28 +550,45 @@ async fn check_bundle_validity_runtime_api_should_work() { transfer_to_charlie_with_big_tip_2.clone(), ), }; - assert_eq!( - alice - .client - .runtime_api() - .check_bundle_extrinsics_validity( - alice.client.as_ref().info().best_hash, - vec![ - transfer_to_charlie_with_big_tip_1.clone().into(), - transfer_to_charlie_with_big_tip_2.clone().into(), - ], - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, - ) - .unwrap(), - Err(CheckBundleValidityError::InvalidTransaction { - tx_index: 1, - tx_validity_error: CheckTxValidityError::decode( - &mut tx_validity_error.encode().as_slice() - ) - .unwrap(), - }) - ); + + let transfer_to_charlie_with_big_tip = vec![ + transfer_to_charlie_with_big_tip_1.clone().into(), + transfer_to_charlie_with_big_tip_2.clone().into(), + ]; + + { + let runtime_instance = alice.client.runtime_api(); + for (i, extrinsic) in transfer_to_charlie_with_big_tip.iter().enumerate() { + if i == 1 { + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Err( + CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()) + .unwrap() + ) + ); + } else { + assert_eq!( + runtime_instance + .check_transaction_and_do_pre_dispatch( + alice.client.as_ref().info().best_hash, + extrinsic, + alice.client.as_ref().info().best_number, + alice.client.as_ref().info().best_hash, + ) + .unwrap(), + Ok(()) + ); + } + } + } } #[tokio::test(flavor = "multi_thread")] diff --git a/domains/primitives/runtime/src/lib.rs b/domains/primitives/runtime/src/lib.rs index c1e72b6e50..d7c007dab5 100644 --- a/domains/primitives/runtime/src/lib.rs +++ b/domains/primitives/runtime/src/lib.rs @@ -163,26 +163,6 @@ pub enum VerifyTxValidityError { FailedToDecodeAccountId, } -#[derive(Debug, PartialEq, Eq, Encode, Decode, TypeInfo)] -pub enum CheckBundleValidityError { - /// One of the transaction is invalid. - InvalidTransaction { - /// Index of the transaction in bundle's transaction list - tx_index: u32, - /// Tx validity error of the tx referenced by `tx_index` - tx_validity_error: CheckTxValidityError, - }, -} - -impl CheckBundleValidityError { - pub fn from_tx_validity_error(tx_index: u32, tx_validity_error: CheckTxValidityError) -> Self { - Self::InvalidTransaction { - tx_index, - tx_validity_error, - } - } -} - sp_api::decl_runtime_apis! { /// Base API that every domain runtime must implement. #[api_version(2)] @@ -257,11 +237,6 @@ sp_api::decl_runtime_apis! { tx_era: Option, ) -> Result>, VerifyTxValidityError>; - /// New in v2. - /// Checks validity of bundle extrinsics. Internally calls `check_transaction_and_do_pre_dispatch` - fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: NumberFor, - block_hash: ::Hash) -> Result<(), CheckBundleValidityError>; - /// Returns extrinsic Era if present fn extrinsic_era( extrinsic: &::Extrinsic diff --git a/domains/runtime/evm/src/lib.rs b/domains/runtime/evm/src/lib.rs index 47f33ebdf1..3c9c904f56 100644 --- a/domains/runtime/evm/src/lib.rs +++ b/domains/runtime/evm/src/lib.rs @@ -1061,15 +1061,6 @@ impl_runtime_apis! { Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) } - /// This runtime api is equivalent of `execute_block` but only for bundle extrinsic validation. - fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, - block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { - for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { - Self::check_transaction_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; - } - Ok(()) - } - fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { diff --git a/domains/test/runtime/evm/src/lib.rs b/domains/test/runtime/evm/src/lib.rs index e611dae8db..26f7c59aac 100644 --- a/domains/test/runtime/evm/src/lib.rs +++ b/domains/test/runtime/evm/src/lib.rs @@ -1049,14 +1049,6 @@ impl_runtime_apis! { Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) } - fn check_bundle_extrinsics_validity(bundle_extrinsics: Vec<::Extrinsic>, block_number: BlockNumber, - block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckBundleValidityError> { - for (extrinsic_index, extrinsic) in bundle_extrinsics.iter().enumerate() { - Self::check_transaction_and_do_pre_dispatch(extrinsic, block_number, block_hash).map_err(|tx_validity_error| domain_runtime_primitives::CheckBundleValidityError::from_tx_validity_error(extrinsic_index as u32, tx_validity_error))?; - } - Ok(()) - } - fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { From e022889e38660dcee2ad1f8be0a79a747e4c4099 Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Sat, 9 Dec 2023 23:09:39 +0400 Subject: [PATCH 5/7] addressing feedback - backward incompatible apis + test simplification --- crates/sp-domains-fraud-proof/src/tests.rs | 820 +++--------------- domains/client/block-preprocessor/src/lib.rs | 30 +- .../src/stateless_runtime.rs | 21 +- .../src/domain_bundle_proposer.rs | 27 +- domains/client/domain-operator/src/tests.rs | 2 +- domains/primitives/runtime/src/lib.rs | 70 +- domains/runtime/evm/src/lib.rs | 152 +--- domains/test/runtime/evm/src/lib.rs | 147 +--- domains/test/service/src/domain.rs | 17 +- domains/test/service/src/lib.rs | 4 +- 10 files changed, 175 insertions(+), 1115 deletions(-) diff --git a/crates/sp-domains-fraud-proof/src/tests.rs b/crates/sp-domains-fraud-proof/src/tests.rs index 09be993ea2..5bc1db18b4 100644 --- a/crates/sp-domains-fraud-proof/src/tests.rs +++ b/crates/sp-domains-fraud-proof/src/tests.rs @@ -1,91 +1,18 @@ -use codec::{Decode, Encode}; -use domain_block_preprocessor::stateless_runtime::StatelessRuntime; -use domain_runtime_primitives::{CheckTxValidityError, DomainCoreApi}; -use domain_test_service::domain::EvmDomainClient as DomainClient; -use domain_test_service::evm_domain_test_runtime::Runtime as TestRuntime; -use domain_test_service::EcdsaKeyring::{Alice, Bob, Charlie}; +use codec::Encode; +use domain_runtime_primitives::DomainCoreApi; +use domain_test_service::EcdsaKeyring::{Alice, Charlie}; use domain_test_service::Sr25519Keyring::Ferdie; -use domain_test_service::{construct_extrinsic_generic, EvmDomainNode, GENESIS_DOMAIN_ID}; -use fp_rpc::EthereumRuntimeRPCApi; -use sc_client_api::{HeaderBackend, ProofProvider, StorageProof}; +use domain_test_service::GENESIS_DOMAIN_ID; +use sc_client_api::HeaderBackend; use sc_service::{BasePath, Role}; -use sp_api::{ApiExt, BlockT, ProvideRuntimeApi, TransactionOutcome}; -use sp_domains::DomainsApi; -use sp_runtime::generic::UncheckedExtrinsic; -use sp_runtime::traits::{BlakeTwo256, Header as HeaderT}; +use sp_api::{ApiExt, ProvideRuntimeApi, TransactionOutcome}; use sp_runtime::transaction_validity::{InvalidTransaction, TransactionValidityError}; -use sp_runtime::{OpaqueExtrinsic, Storage}; -use sp_trie::{read_trie_value, LayoutV1}; -use std::collections::BTreeMap; -use std::sync::Arc; -use subspace_runtime_primitives::opaque::Block; -use subspace_test_service::{produce_block_with, produce_blocks, MockConsensusNode}; +use sp_runtime::OpaqueExtrinsic; +use subspace_test_service::MockConsensusNode; use tempfile::TempDir; -type HashFor = <::Header as HeaderT>::Hash; - -fn generate_storage_proof_for_tx_validity( - block_hash: HashFor, - client: Arc, - keys: Vec>, -) -> StorageProof { - client - .read_proof( - block_hash, - &mut keys - .iter() - .map(|k| k.as_slice()) - .collect::>() - .into_iter(), - ) - .unwrap() -} - -fn get_storage_keys_for_verifying_signed_tx_validity( - domain_node: &EvmDomainNode, - extrinsic: UncheckedExtrinsic< - domain_test_service::evm_domain_test_runtime::Address, - ::RuntimeCall, - domain_test_service::evm_domain_test_runtime::Signature, - domain_test_service::evm_domain_test_runtime::SignedExtra, - >, -) -> Vec> { - let extrinsic_signer = domain_node - .client - .runtime_api() - .extract_signer( - domain_node.client.as_ref().info().best_hash, - vec![extrinsic.clone().into()], - ) - .unwrap() - .first() - .unwrap() - .0 - .clone(); - let extrinsic_era = domain_node - .client - .runtime_api() - .extrinsic_era( - domain_node.client.as_ref().info().best_hash, - &extrinsic.clone().into(), - ) - .unwrap(); - - domain_node - .client - .runtime_api() - .storage_keys_for_verifying_transaction_validity( - domain_node.client.as_ref().info().best_hash, - extrinsic_signer, - domain_node.client.as_ref().info().best_number, - extrinsic_era, - ) - .unwrap() - .unwrap() -} - #[tokio::test(flavor = "multi_thread")] -async fn runtime_instance_assumptions_are_correct() { +async fn runtime_instance_storage_assumptions_are_correct() { let directory = TempDir::new().expect("Must be able to create temporary directory"); let mut builder = sc_cli::LoggerBuilder::new(""); @@ -102,7 +29,7 @@ async fn runtime_instance_assumptions_are_correct() { ); // Run Alice (a evm domain authority node) - let alice = domain_test_service::DomainNodeBuilder::new( + let mut alice = domain_test_service::DomainNodeBuilder::new( tokio_handle.clone(), Alice, BasePath::new(directory.path().join("alice")), @@ -110,90 +37,44 @@ async fn runtime_instance_assumptions_are_correct() { .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) .await; - // transfer most of the alice's balance - let alice_balance = alice - .client - .runtime_api() - .account_basic( - alice.client.as_ref().info().best_hash, - Alice.to_account_id().into(), - ) - .unwrap() - .balance - .as_u128(); - let target_balance = u32::MAX - 100; - let balance_to_transfer = alice_balance - target_balance as u128; - - let alice_balance_transfer_extrinsic = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: balance_to_transfer, - }, - alice.key, - false, - 0, - 1, - ); + let best_hash = alice.client.as_ref().info().best_hash; + let best_number = alice.client.as_ref().info().best_number; - alice - .send_extrinsic(alice_balance_transfer_extrinsic) - .await - .expect("Failed to send extrinsic"); - - let (slot, bundle) = ferdie.produce_slot_and_wait_for_bundle_submission().await; - assert!(bundle.is_some()); - produce_block_with!(ferdie.produce_block_with_slot(slot), alice) - .await - .unwrap(); - produce_blocks!(ferdie, alice, 10).await.unwrap(); + // transfer most of the alice's balance + let alice_balance = alice.free_balance(Alice.to_account_id()); - let mut alice_nonce = 1; + let mut alice_nonce = alice.account_nonce(); - let transfer_to_charlie_with_big_tip_1 = construct_extrinsic_generic::( - &alice.client, + let transfer_to_charlie_with_big_tip_1 = alice.construct_extrinsic_with_tip( + alice_nonce, + (alice_balance / 3) * 2, pallet_balances::Call::transfer_allow_death { dest: Charlie.to_account_id(), value: 1, }, - alice.key, - false, - alice_nonce, - 2000000000u32, ); alice_nonce += 1; - let transfer_to_charlie_with_big_tip_2 = construct_extrinsic_generic::( - &alice.client, + let transfer_to_charlie_with_big_tip_2 = alice.construct_extrinsic_with_tip( + alice_nonce, + (alice_balance / 3) * 2, pallet_balances::Call::transfer_allow_death { dest: Charlie.to_account_id(), value: 1, }, - alice.key, - false, - alice_nonce, - 2000000000u32, ); // A runtime instance preserve changes in between let runtime_instance = alice.client.runtime_api(); - let tx_validity_error = CheckTxValidityError::InvalidTransaction { - error: TransactionValidityError::Invalid(InvalidTransaction::Payment), - storage_keys: get_storage_keys_for_verifying_signed_tx_validity( - &alice, - transfer_to_charlie_with_big_tip_2.clone(), - ), - }; - // First transaction should be okay assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_to_charlie_with_big_tip_1.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), Ok(()) @@ -204,70 +85,82 @@ async fn runtime_instance_assumptions_are_correct() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_to_charlie_with_big_tip_2.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), - Err(CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()).unwrap()) + Err(TransactionValidityError::Invalid( + InvalidTransaction::Payment + )) + ); + + // If second tx is executed in dedicated runtime instance it would error with `InvalidTransaction::Future` + // as the nonce at the block for this account is less than the tx's nonce. And there is no overlay + // preserved between runtime instance. + assert_eq!( + alice + .client + .runtime_api() + .check_transaction_and_do_pre_dispatch( + best_hash, + &transfer_to_charlie_with_big_tip_2.clone().into(), + best_number, + best_hash, + ) + .unwrap(), + Err(TransactionValidityError::Invalid( + InvalidTransaction::Future + )) ); let test_commit_mode = vec![true, false]; for commit_mode in test_commit_mode { - // A runtime instance can rollback changes safely. - let runtime_instance = alice.client.runtime_api(); + alice_nonce = alice.account_nonce(); - alice_nonce = 1; - - let transfer_with_big_tip_1 = construct_extrinsic_generic::( - &alice.client, + let transfer_with_big_tip_1 = alice.construct_extrinsic_with_tip( + alice_nonce, + alice_balance / 3, pallet_balances::Call::transfer_allow_death { dest: Charlie.to_account_id(), value: 1, }, - alice.key, - false, - alice_nonce, - 4000000000u32 / 3, ); alice_nonce += 1; - let transfer_with_big_tip_2 = construct_extrinsic_generic::( - &alice.client, + let transfer_with_big_tip_2 = alice.construct_extrinsic_with_tip( + alice_nonce, + alice_balance / 3, pallet_balances::Call::transfer_allow_death { dest: Charlie.to_account_id(), value: 1, }, - alice.key, - false, - alice_nonce, - 4000000000u32 / 3, ); if commit_mode { alice_nonce += 1; } - let transfer_with_big_tip_3 = construct_extrinsic_generic::( - &alice.client, + let transfer_with_big_tip_3 = alice.construct_extrinsic_with_tip( + alice_nonce, + alice_balance / 3, pallet_balances::Call::transfer_allow_death { dest: Charlie.to_account_id(), value: 1, }, - alice.key, - false, - alice_nonce, - 4000000000u32 / 3, ); + // A runtime instance can rollback changes safely. + let runtime_instance = alice.client.runtime_api(); + assert!(runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_with_big_tip_1.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash + best_number, + best_hash ) .unwrap() .is_ok()); @@ -276,17 +169,17 @@ async fn runtime_instance_assumptions_are_correct() { .execute_in_transaction(|api| { if commit_mode { TransactionOutcome::Commit(api.check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_with_big_tip_2.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, )) } else { TransactionOutcome::Rollback(api.check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_with_big_tip_2.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, )) } }) @@ -295,17 +188,16 @@ async fn runtime_instance_assumptions_are_correct() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &transfer_with_big_tip_3.clone().into(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash + best_number, + best_hash ) .unwrap(), if commit_mode { - Err( - CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()) - .unwrap(), - ) + Err(TransactionValidityError::Invalid( + InvalidTransaction::Payment, + )) } else { Ok(()) } @@ -314,7 +206,7 @@ async fn runtime_instance_assumptions_are_correct() { } #[tokio::test(flavor = "multi_thread")] -async fn check_bundle_validity_runtime_api_should_work() { +async fn check_tx_validity_and_pre_dispatch_runtime_api_should_work() { let directory = TempDir::new().expect("Must be able to create temporary directory"); let mut builder = sc_cli::LoggerBuilder::new(""); @@ -331,7 +223,7 @@ async fn check_bundle_validity_runtime_api_should_work() { ); // Run Alice (a evm domain authority node) - let alice = domain_test_service::DomainNodeBuilder::new( + let mut alice = domain_test_service::DomainNodeBuilder::new( tokio_handle.clone(), Alice, BasePath::new(directory.path().join("alice")), @@ -339,42 +231,20 @@ async fn check_bundle_validity_runtime_api_should_work() { .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) .await; - let mut alice_nonce = 0; - let alice_balance = alice - .client - .runtime_api() - .account_basic( - alice.client.as_ref().info().best_hash, - Alice.to_account_id().into(), - ) - .unwrap() - .balance - .as_u128(); + let mut alice_nonce = alice.account_nonce(); + let best_hash = alice.client.as_ref().info().best_hash; + let best_number = alice.client.as_ref().info().best_number; + let dummy_runtime_call = pallet_balances::Call::transfer_allow_death { + dest: Charlie.to_account_id(), + value: 1, + }; - let transfer_to_charlie = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: alice_balance, - }, - alice.key, - false, - alice_nonce, - 0, - ); + let transfer_to_charlie = + alice.construct_extrinsic_with_tip(alice_nonce, 0, dummy_runtime_call.clone()); alice_nonce += 1; - let transfer_to_charlie_2 = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 8, - }, - alice.key, - false, - alice_nonce, - 1, - ); + let transfer_to_charlie_2 = + alice.construct_extrinsic_with_tip(alice_nonce, 0, dummy_runtime_call.clone()); let sample_valid_bundle_extrinsics = vec![ OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(), @@ -387,10 +257,10 @@ async fn check_bundle_validity_runtime_api_should_work() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), Ok(()) @@ -399,17 +269,7 @@ async fn check_bundle_validity_runtime_api_should_work() { } let transfer_to_charlie_3_duplicate_nonce_extrinsic = - construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 8, - }, - alice.key, - false, - alice_nonce, - 1, - ); + alice.construct_extrinsic_with_tip(alice_nonce, 0, dummy_runtime_call.clone()); // Here tx index: 2 is invalid let sample_invalid_bundle_with_same_nonce_extrinsic = vec![ @@ -429,10 +289,10 @@ async fn check_bundle_validity_runtime_api_should_work() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), Ok(()) @@ -441,89 +301,36 @@ async fn check_bundle_validity_runtime_api_should_work() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), - Err(CheckTxValidityError::InvalidTransaction { - error: TransactionValidityError::Invalid(InvalidTransaction::Stale), - storage_keys: get_storage_keys_for_verifying_signed_tx_validity( - &alice, - transfer_to_charlie_3_duplicate_nonce_extrinsic.clone() - ), - }) + Err(TransactionValidityError::Invalid(InvalidTransaction::Stale)) ); } } } - // transfer most of the alice's balance - let alice_balance = alice - .client - .runtime_api() - .account_basic( - alice.client.as_ref().info().best_hash, - Alice.to_account_id().into(), - ) - .unwrap() - .balance - .as_u128(); - let target_balance = u32::MAX - 100; - let balance_to_transfer = alice_balance - target_balance as u128; + // Resetting the nonce + alice_nonce = alice.account_nonce(); - // reset alice nonce now - alice_nonce = 0; + // transfer most of the alice's balance + let alice_balance = alice.free_balance(Alice.to_account_id()); - let alice_balance_transfer_extrinsic = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: balance_to_transfer, - }, - alice.key, - false, + let transfer_to_charlie_with_big_tip_1 = alice.construct_extrinsic_with_tip( alice_nonce, - 1, + (alice_balance / 3) * 2, + dummy_runtime_call.clone(), ); - alice_nonce += 1; - alice - .send_extrinsic(alice_balance_transfer_extrinsic) - .await - .expect("Failed to send extrinsic"); - - let (slot, bundle) = ferdie.produce_slot_and_wait_for_bundle_submission().await; - assert!(bundle.is_some()); - produce_block_with!(ferdie.produce_block_with_slot(slot), alice) - .await - .unwrap(); - produce_blocks!(ferdie, alice, 10).await.unwrap(); - - let transfer_to_charlie_with_big_tip_1 = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 1, - }, - alice.key, - false, - alice_nonce, - 2000000000u32, - ); alice_nonce += 1; - let transfer_to_charlie_with_big_tip_2 = construct_extrinsic_generic::( - &alice.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 1, - }, - alice.key, - false, + let transfer_to_charlie_with_big_tip_2 = alice.construct_extrinsic_with_tip( alice_nonce, - 2000000000u32, + (alice_balance / 3) * 2, + dummy_runtime_call.clone(), ); // In single tx context both are valid (both tx are identical except nonce), so no need to test both of them independently. @@ -533,24 +340,16 @@ async fn check_bundle_validity_runtime_api_should_work() { .client .runtime_api() .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, &OpaqueExtrinsic::from_bytes(&transfer_to_charlie_with_big_tip_1.encode()).unwrap(), - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), Ok(()) ); // Tx index: 1 is invalid since low balance to pay for the tip - let tx_validity_error = CheckTxValidityError::InvalidTransaction { - error: TransactionValidityError::Invalid(InvalidTransaction::Payment), - storage_keys: get_storage_keys_for_verifying_signed_tx_validity( - &alice, - transfer_to_charlie_with_big_tip_2.clone(), - ), - }; - let transfer_to_charlie_with_big_tip = vec![ transfer_to_charlie_with_big_tip_1.clone().into(), transfer_to_charlie_with_big_tip_2.clone().into(), @@ -563,25 +362,24 @@ async fn check_bundle_validity_runtime_api_should_work() { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), - Err( - CheckTxValidityError::decode(&mut tx_validity_error.encode().as_slice()) - .unwrap() - ) + Err(TransactionValidityError::Invalid( + InvalidTransaction::Payment + )) ); } else { assert_eq!( runtime_instance .check_transaction_and_do_pre_dispatch( - alice.client.as_ref().info().best_hash, + best_hash, extrinsic, - alice.client.as_ref().info().best_number, - alice.client.as_ref().info().best_hash, + best_number, + best_hash, ) .unwrap(), Ok(()) @@ -591,378 +389,6 @@ async fn check_bundle_validity_runtime_api_should_work() { } } -#[tokio::test(flavor = "multi_thread")] -async fn check_tx_validity_runtime_api_should_work() { - let directory = TempDir::new().expect("Must be able to create temporary directory"); - - let mut builder = sc_cli::LoggerBuilder::new(""); - builder.with_colors(false); - let _ = builder.init(); - - let tokio_handle = tokio::runtime::Handle::current(); - - // Start Ferdie - let mut ferdie = MockConsensusNode::run( - tokio_handle.clone(), - Ferdie, - BasePath::new(directory.path().join("ferdie")), - ); - - // Run Alice (a evm domain authority node) - let alice = domain_test_service::DomainNodeBuilder::new( - tokio_handle.clone(), - Alice, - BasePath::new(directory.path().join("alice")), - ) - .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) - .await; - - // Run Bob (a evm domain full node) - let bob = domain_test_service::DomainNodeBuilder::new( - tokio_handle, - Bob, - BasePath::new(directory.path().join("bob")), - ) - .build_evm_node(Role::Full, GENESIS_DOMAIN_ID, &mut ferdie) - .await; - - // Bob is able to sync blocks. - produce_blocks!(ferdie, alice, 1, bob).await.unwrap(); - - let bob_nonce = bob.account_nonce(); - let transfer_to_charlie = construct_extrinsic_generic::( - &bob.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 8, - }, - bob.key, - false, - bob_nonce + 1, - 1, - ); - - produce_blocks!(ferdie, alice, 2, bob).await.unwrap(); - - let opaque_extrinsic = OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(); - let extrinsic_best_hash = bob.client.as_ref().info().best_hash; - let bob_account_id = Bob.to_account_id().encode(); - let extrinsic_best_number = bob.client.as_ref().info().best_number; - - let extrinsic_era = bob - .client - .runtime_api() - .extrinsic_era(bob.client.as_ref().info().best_hash, &opaque_extrinsic) - .unwrap(); - - let test_table = vec![ - ( - Some(4), - Some(TransactionValidityError::Invalid( - InvalidTransaction::AncientBirthBlock, - )), - ), - ( - Some(3), - Some(TransactionValidityError::Invalid( - InvalidTransaction::Payment, - )), - ), - ( - Some(1), - Some(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )), - ), - ( - Some(0), - Some(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )), - ), - (None, None), - ]; - - for table_data in test_table { - let mut storage_keys_for_tx_validity = bob - .client - .runtime_api() - .storage_keys_for_verifying_transaction_validity( - extrinsic_best_hash, - Some(bob_account_id.clone()), - extrinsic_best_number, - extrinsic_era, - ) - .unwrap() - .unwrap(); - - let maybe_storage_key_to_remove = table_data.0; - let original_storage_keys_for_tx_validity = storage_keys_for_tx_validity.clone(); - - if let Some(storage_key_to_remove) = maybe_storage_key_to_remove { - storage_keys_for_tx_validity = storage_keys_for_tx_validity - .drain(..) - .enumerate() - .filter(|(i, _v)| *i != storage_key_to_remove) - .map(|(_i, v)| v) - .collect(); - } - - let maybe_error = table_data.1; - - let expected_out = if let Some(err) = maybe_error { - let err_with_storage_keys = CheckTxValidityError::InvalidTransaction { - error: err, - storage_keys: original_storage_keys_for_tx_validity, - }; - Err(err_with_storage_keys) - } else { - Ok(()) - }; - - let storage_proof = generate_storage_proof_for_tx_validity( - bob.client.as_ref().info().best_hash, - bob.client.clone(), - storage_keys_for_tx_validity.clone(), - ); - - let header = bob - .client - .as_ref() - .header(bob.client.as_ref().info().best_hash); - let state_root = header.unwrap().unwrap().state_root; - - let wasm_bundle = ferdie - .client - .runtime_api() - .domain_runtime_code(ferdie.client.as_ref().info().best_hash, GENESIS_DOMAIN_ID) - .unwrap() - .unwrap(); - - let mut domain_stateless_runtime = - StatelessRuntime::::new(ferdie.executor.clone().into(), wasm_bundle.into()); - - let db = storage_proof.into_memory_db::(); - let mut top_storage_map = BTreeMap::new(); - for storage_key in storage_keys_for_tx_validity.iter() { - let storage_value = read_trie_value::, _>( - &db, - &state_root, - storage_key, - None, - None, - ) - .unwrap() - .unwrap(); - top_storage_map.insert(storage_key.to_vec(), storage_value); - } - - let storage = Storage { - top: top_storage_map, - children_default: Default::default(), - }; - - domain_stateless_runtime.set_storage(storage); - - assert_eq!( - domain_stateless_runtime - .check_transaction_validity( - &opaque_extrinsic, - bob.client.as_ref().info().best_number, - bob.client.as_ref().info().best_hash - ) - .unwrap(), - expected_out - ); - } -} - -#[tokio::test(flavor = "multi_thread")] -async fn check_tx_validity_and_do_pre_dispatch_runtime_api_should_work() { - let directory = TempDir::new().expect("Must be able to create temporary directory"); - - let mut builder = sc_cli::LoggerBuilder::new(""); - builder.with_colors(false); - let _ = builder.init(); - - let tokio_handle = tokio::runtime::Handle::current(); - - // Start Ferdie - let mut ferdie = MockConsensusNode::run( - tokio_handle.clone(), - Ferdie, - BasePath::new(directory.path().join("ferdie")), - ); - - // Run Alice (a evm domain authority node) - let alice = domain_test_service::DomainNodeBuilder::new( - tokio_handle.clone(), - Alice, - BasePath::new(directory.path().join("alice")), - ) - .build_evm_node(Role::Authority, GENESIS_DOMAIN_ID, &mut ferdie) - .await; - - // Run Bob (a evm domain full node) - let bob = domain_test_service::DomainNodeBuilder::new( - tokio_handle, - Bob, - BasePath::new(directory.path().join("bob")), - ) - .build_evm_node(Role::Full, GENESIS_DOMAIN_ID, &mut ferdie) - .await; - - // Bob is able to sync blocks. - produce_blocks!(ferdie, alice, 1, bob).await.unwrap(); - - let bob_nonce = bob.account_nonce(); - let transfer_to_charlie = construct_extrinsic_generic::( - &bob.client, - pallet_balances::Call::transfer_allow_death { - dest: Charlie.to_account_id(), - value: 8, - }, - bob.key, - false, - bob_nonce, - 1, - ); - - produce_blocks!(ferdie, alice, 2, bob).await.unwrap(); - - let opaque_extrinsic = OpaqueExtrinsic::from_bytes(&transfer_to_charlie.encode()).unwrap(); - let extrinsic_best_hash = bob.client.as_ref().info().best_hash; - let bob_account_id = Bob.to_account_id().encode(); - let extrinsic_best_number = bob.client.as_ref().info().best_number; - - let extrinsic_era = bob - .client - .runtime_api() - .extrinsic_era(bob.client.as_ref().info().best_hash, &opaque_extrinsic) - .unwrap(); - - let test_table = vec![ - ( - Some(4), - Some(TransactionValidityError::Invalid( - InvalidTransaction::AncientBirthBlock, - )), - ), - ( - Some(3), - Some(TransactionValidityError::Invalid( - InvalidTransaction::Payment, - )), - ), - ( - Some(1), - Some(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )), - ), - ( - Some(0), - Some(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )), - ), - (None, None), - ]; - - for table_data in test_table { - let mut storage_keys_for_tx_validity = bob - .client - .runtime_api() - .storage_keys_for_verifying_transaction_validity( - extrinsic_best_hash, - Some(bob_account_id.clone()), - extrinsic_best_number, - extrinsic_era, - ) - .unwrap() - .unwrap(); - - let maybe_storage_key_to_remove = table_data.0; - let original_storage_keys_for_tx_validity = storage_keys_for_tx_validity.clone(); - - if let Some(storage_key_to_remove) = maybe_storage_key_to_remove { - storage_keys_for_tx_validity = storage_keys_for_tx_validity - .drain(..) - .enumerate() - .filter(|(i, _v)| *i != storage_key_to_remove) - .map(|(_i, v)| v) - .collect(); - } - - let maybe_error = table_data.1; - - let expected_out = if let Some(err) = maybe_error { - let err_with_storage_keys = CheckTxValidityError::InvalidTransaction { - error: err, - storage_keys: original_storage_keys_for_tx_validity, - }; - Err(err_with_storage_keys) - } else { - Ok(()) - }; - - let storage_proof = generate_storage_proof_for_tx_validity( - bob.client.as_ref().info().best_hash, - bob.client.clone(), - storage_keys_for_tx_validity.clone(), - ); - - let header = bob - .client - .as_ref() - .header(bob.client.as_ref().info().best_hash); - let state_root = header.unwrap().unwrap().state_root; - - let wasm_bundle = ferdie - .client - .runtime_api() - .domain_runtime_code(ferdie.client.as_ref().info().best_hash, GENESIS_DOMAIN_ID) - .unwrap() - .unwrap(); - - let mut domain_stateless_runtime = - StatelessRuntime::::new(ferdie.executor.clone().into(), wasm_bundle.into()); - - let db = storage_proof.into_memory_db::(); - let mut top_storage_map = BTreeMap::new(); - for storage_key in storage_keys_for_tx_validity.iter() { - let storage_value = read_trie_value::, _>( - &db, - &state_root, - storage_key, - None, - None, - ) - .unwrap() - .unwrap(); - top_storage_map.insert(storage_key.to_vec(), storage_value); - } - - let storage = Storage { - top: top_storage_map, - children_default: Default::default(), - }; - - domain_stateless_runtime.set_storage(storage); - - assert_eq!( - domain_stateless_runtime - .check_transaction_and_do_pre_dispatch( - &opaque_extrinsic, - bob.client.as_ref().info().best_number, - bob.client.as_ref().info().best_hash - ) - .unwrap(), - expected_out - ); - } -} - // #[tokio::test(flavor = "multi_thread")] // #[ignore] // async fn execution_proof_creation_and_verification_should_work() { diff --git a/domains/client/block-preprocessor/src/lib.rs b/domains/client/block-preprocessor/src/lib.rs index 302d52ef53..43618f3394 100644 --- a/domains/client/block-preprocessor/src/lib.rs +++ b/domains/client/block-preprocessor/src/lib.rs @@ -20,7 +20,7 @@ use codec::{Decode, Encode}; use domain_runtime_primitives::opaque::AccountId; use domain_runtime_primitives::DomainCoreApi; use sc_client_api::BlockBackend; -use sp_api::{ApiExt, HashT, ProvideRuntimeApi}; +use sp_api::{HashT, ProvideRuntimeApi}; use sp_blockchain::HeaderBackend; use sp_core::H256; use sp_domains::extrinsics::deduplicate_and_shuffle_extrinsics; @@ -268,14 +268,6 @@ where // // NOTE: for each extrinsic the checking order must follow `InvalidBundleType::checking_order` let runtime_api = self.client.runtime_api(); - let api_version = runtime_api - .api_version::>(at) - .map_err(sp_blockchain::Error::RuntimeApiError)? - .ok_or_else(|| { - sp_blockchain::Error::Application( - format!("Could not find `DomainCoreApi` api for block `{:?}`.", at).into(), - ) - })?; for (index, opaque_extrinsic) in bundle.extrinsics.iter().enumerate() { let Ok(extrinsic) = <::Extrinsic>::decode(&mut opaque_extrinsic.encode().as_slice()) @@ -313,23 +305,9 @@ where // between them. // Using `check_transaction_and_do_pre_dispatch` instead of `check_transaction_validity` // to maintain side-effect in the storage buffer. - let is_legal_tx = if api_version == 2 { - runtime_api.check_transaction_and_do_pre_dispatch( - at, - &extrinsic, - domain_block_number, - at, - )? - } else { - #[allow(deprecated)] // old check_transaction_validity - runtime_api.check_transaction_validity_before_version_2( - at, - &extrinsic, - domain_block_number, - at, - )? - } - .is_ok(); + let is_legal_tx = runtime_api + .check_transaction_and_do_pre_dispatch(at, &extrinsic, domain_block_number, at)? + .is_ok(); if !is_legal_tx { // TODO: Generate a fraud proof for this invalid bundle diff --git a/domains/client/block-preprocessor/src/stateless_runtime.rs b/domains/client/block-preprocessor/src/stateless_runtime.rs index 1e5f2eed63..9f02bbc0b3 100644 --- a/domains/client/block-preprocessor/src/stateless_runtime.rs +++ b/domains/client/block-preprocessor/src/stateless_runtime.rs @@ -1,12 +1,13 @@ use codec::{Codec, Encode}; use domain_runtime_primitives::opaque::AccountId; -use domain_runtime_primitives::{CheckTxValidityError, DomainCoreApi}; +use domain_runtime_primitives::DomainCoreApi; use sc_executor::RuntimeVersionOf; use sp_api::{ApiError, BlockT, Core, Hasher, RuntimeVersion}; use sp_core::traits::{CallContext, CodeExecutor, FetchRuntimeCode, RuntimeCode}; use sp_messenger::messages::ExtractedStateRootsFromProof; use sp_messenger::MessengerApi; use sp_runtime::traits::NumberFor; +use sp_runtime::transaction_validity::TransactionValidityError; use sp_runtime::Storage; use sp_state_machine::BasicExternalities; use std::borrow::Cow; @@ -212,29 +213,13 @@ where ) } - /// This is stateful runtime api call and require setting of storage keys. - pub fn check_transaction_validity( - &self, - uxt: &::Extrinsic, - block_number: NumberFor, - block_hash: ::Hash, - ) -> Result, ApiError> { - >::check_transaction_validity( - self, - Default::default(), - uxt, - block_number, - block_hash, - ) - } - /// This is stateful runtime api call and require setting of storage keys. pub fn check_transaction_and_do_pre_dispatch( &self, uxt: &::Extrinsic, block_number: NumberFor, block_hash: ::Hash, - ) -> Result, ApiError> { + ) -> Result, ApiError> { >::check_transaction_and_do_pre_dispatch( self, Default::default(), diff --git a/domains/client/domain-operator/src/domain_bundle_proposer.rs b/domains/client/domain-operator/src/domain_bundle_proposer.rs index 777ad89f22..2de5f4f094 100644 --- a/domains/client/domain-operator/src/domain_bundle_proposer.rs +++ b/domains/client/domain-operator/src/domain_bundle_proposer.rs @@ -115,18 +115,6 @@ where // We are using one runtime api instance here to maintain storage changes in the instance's internal buffer // between runtime calls done in this loop. let runtime_api_instance = self.client.runtime_api(); - let api_version = runtime_api_instance - .api_version::>(parent_hash) - .map_err(sp_blockchain::Error::RuntimeApiError)? - .ok_or_else(|| { - sp_blockchain::Error::Application( - format!( - "Could not find `DomainCoreApi` api for block `{:?}`.", - parent_hash - ) - .into(), - ) - })?; for pending_tx in pending_iterator { let pending_tx_data = pending_tx.data(); @@ -176,22 +164,13 @@ where // some other checks, its side effect will still be part of RuntimeApiImpl's changes buffer. let transaction_validity_result = runtime_api_instance.execute_in_transaction(|api| { - let transaction_validity_result = if api_version == 2 { - api.check_transaction_and_do_pre_dispatch( + let transaction_validity_result = api + .check_transaction_and_do_pre_dispatch( parent_hash, pending_tx_data, parent_number, parent_hash, - ) - } else { - #[allow(deprecated)] // old check_transaction_validity - api.check_transaction_validity_before_version_2( - parent_hash, - pending_tx_data, - parent_number, - parent_hash, - ) - }; + ); // Only commit, if there are no errors (both ApiError and CheckTxValidityError) if let Ok(Ok(_)) = transaction_validity_result { sp_api::TransactionOutcome::Commit(transaction_validity_result) diff --git a/domains/client/domain-operator/src/tests.rs b/domains/client/domain-operator/src/tests.rs index 41d97cd0c0..dc968d41e4 100644 --- a/domains/client/domain-operator/src/tests.rs +++ b/domains/client/domain-operator/src/tests.rs @@ -2769,7 +2769,7 @@ async fn test_domain_transaction_fee_and_operator_reward() { // Transaction fee (including the tip) is deducted from alice's account let alice_free_balance_changes = pre_alice_free_balance - alice.free_balance(Alice.to_account_id()); - assert!(alice_free_balance_changes >= tip as u128); + assert!(alice_free_balance_changes >= tip); // All the transaction fee is collected as operator reward assert_eq!(alice_free_balance_changes, receipt.total_rewards); diff --git a/domains/primitives/runtime/src/lib.rs b/domains/primitives/runtime/src/lib.rs index d7c007dab5..4d76358c20 100644 --- a/domains/primitives/runtime/src/lib.rs +++ b/domains/primitives/runtime/src/lib.rs @@ -20,9 +20,7 @@ use parity_scale_codec::{Decode, Encode}; use scale_info::TypeInfo; use sp_runtime::generic::{Era, UncheckedExtrinsic}; -use sp_runtime::traits::{ - Block as BlockT, Convert, IdentifyAccount, LookupError, NumberFor, Verify, -}; +use sp_runtime::traits::{Block as BlockT, Convert, IdentifyAccount, NumberFor, Verify}; use sp_runtime::transaction_validity::TransactionValidityError; use sp_runtime::{Digest, MultiAddress, MultiSignature}; use sp_std::vec::Vec; @@ -136,36 +134,8 @@ pub mod opaque { pub type AccountId = Vec; } -#[derive(Debug, PartialEq, Eq, Encode, Decode, TypeInfo)] -pub enum CheckTxValidityError { - /// Can not find the sender from address. - Lookup, - /// Unable to extract signer from tx - UnableToExtractSigner { error: TransactionValidityError }, - /// Transaction is invalid. - InvalidTransaction { - /// Concrete transaction validity error type. - error: TransactionValidityError, - /// Storage keys of state accessed in the validation. - storage_keys: Vec>, - }, -} - -impl From for CheckTxValidityError { - fn from(_lookup_error: LookupError) -> Self { - Self::Lookup - } -} - -#[derive(Debug, PartialEq, Eq, Encode, Decode, TypeInfo)] -pub enum VerifyTxValidityError { - /// Failed to decode the opaque account id into the runtime account type. - FailedToDecodeAccountId, -} - sp_api::decl_runtime_apis! { /// Base API that every domain runtime must implement. - #[api_version(2)] pub trait DomainCoreApi { /// Extracts the optional signer per extrinsic. fn extract_signer( @@ -196,46 +166,12 @@ sp_api::decl_runtime_apis! { /// Returns true if the extrinsic is an inherent extrinsic. fn is_inherent_extrinsic(extrinsic: &::Extrinsic) -> bool; - /// Checks the validity of extrinsic. - #[changed_in(2)] - fn check_transaction_validity( - uxt: &::Extrinsic, - block_number: NumberFor, - block_hash: ::Hash - ) -> Result<(), CheckTxValidityError>; - - /// Checks the validity of extrinsic - /// v2 can check validity of unsigned and self contained extrinsics as well. - fn check_transaction_validity( - uxt: &::Extrinsic, - block_number: NumberFor, - block_hash: ::Hash - ) -> Result<(), CheckTxValidityError>; - - /// New in v2. - /// Checks transaction and also perform pre dispatch on transaction. - /// This is useful to maintain side effect in the RuntimeApi instance's storage buffer. + /// Checks the validity of extrinsic + do pre_dispatch as well. fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: NumberFor, block_hash: ::Hash - ) -> Result<(), CheckTxValidityError>; - - /// Returns the storage keys of states accessed in the API `check_transaction_validity`. - #[changed_in(2)] - fn storage_keys_for_verifying_transaction_validity( - account_id: opaque::AccountId, - block_number: NumberFor, - tx_era: Option, - ) -> Result>, VerifyTxValidityError>; - - /// Returns the storage keys of states accessed in the API `check_transaction_validity` - /// In v2, signer account id is optional to support getting storage keys for unsigned tx as well. - fn storage_keys_for_verifying_transaction_validity( - maybe_account_id: Option, - block_number: NumberFor, - tx_era: Option, - ) -> Result>, VerifyTxValidityError>; + ) -> Result<(), TransactionValidityError>; /// Returns extrinsic Era if present fn extrinsic_era( diff --git a/domains/runtime/evm/src/lib.rs b/domains/runtime/evm/src/lib.rs index 3c9c904f56..779c2384b1 100644 --- a/domains/runtime/evm/src/lib.rs +++ b/domains/runtime/evm/src/lib.rs @@ -18,7 +18,7 @@ use frame_support::dispatch::{DispatchClass, GetDispatchInfo}; use frame_support::inherent::ProvideInherent; use frame_support::traits::{ ConstU16, ConstU32, ConstU64, Currency, Everything, FindAuthor, Imbalance, OnFinalize, - OnUnbalanced, PalletInfoAccess, + OnUnbalanced, }; use frame_support::weights::constants::{ BlockExecutionWeight, ExtrinsicBaseWeight, ParityDbWeight, WEIGHT_REF_TIME_PER_MILLIS, @@ -56,7 +56,7 @@ use sp_runtime::transaction_validity::{ use sp_runtime::{ create_runtime_str, generic, impl_opaque_keys, ApplyExtrinsicResult, ConsensusEngineId, Digest, }; -pub use sp_runtime::{MultiAddress, Perbill, Permill, SaturatedConversion}; +pub use sp_runtime::{MultiAddress, Perbill, Permill}; use sp_std::marker::PhantomData; use sp_std::prelude::*; #[cfg(feature = "std")] @@ -186,7 +186,7 @@ pub const VERSION: RuntimeVersion = RuntimeVersion { spec_name: create_runtime_str!("subspace-evm-domain"), impl_name: create_runtime_str!("subspace-evm-domain"), authoring_version: 0, - spec_version: 1, + spec_version: 0, impl_version: 0, apis: RUNTIME_API_VERSIONS, transaction_version: 0, @@ -704,35 +704,6 @@ pub fn extract_signer( .collect() } -fn storage_keys_for_verifying_tx_validity_inner( - maybe_sender: Option, - block_number: BlockNumber, - maybe_tx_era: Option, -) -> Vec> { - let mut storage_keys = sp_std::vec![ - frame_system::BlockHash::::hashed_key_for(BlockNumber::from(0u32)), - frame_support::storage::storage_prefix(System::name().as_bytes(), b"Number").to_vec(), - pallet_transaction_payment::NextFeeMultiplier::::hashed_key().to_vec() - ]; - - if let Some(sender) = maybe_sender { - storage_keys.push(frame_system::Account::::hashed_key_for(sender)); - } - - match maybe_tx_era { - None => storage_keys, - Some(era) => { - let birth_number = era - .birth(block_number.saturated_into::()) - .saturated_into::(); - storage_keys.push(frame_system::BlockHash::::hashed_key_for( - birth_number, - )); - storage_keys - } - } -} - fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { extrinsic .0 @@ -741,44 +712,6 @@ fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { .map(|(_, _, extra)| extra.4 .0) } -fn check_transaction_validity_inner( - lookup: &Lookup, - uxt: &::Extrinsic, - block_number: BlockNumber, - block_hash: ::Hash, -) -> Result, domain_runtime_primitives::CheckTxValidityError> -where - Lookup: sp_runtime::traits::Lookup, -{ - let maybe_signer_info = extract_signer_inner(uxt, lookup); - let maybe_signer = if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error, - } - })?; - Some(signer) - } else { - None - }; - - let tx_validity = - Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); - - tx_validity - .map(|_| maybe_signer) - .map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys_for_verifying_tx_validity_inner( - maybe_signer, - block_number, - extrinsic_era(uxt), - ), - } - }) -} - #[cfg(feature = "runtime-benchmarks")] mod benches { frame_benchmarking::define_benchmarks!( @@ -958,55 +891,26 @@ impl_runtime_apis! { } } - fn check_transaction_validity(uxt: &::Extrinsic, - block_number: BlockNumber, - block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { - let lookup = frame_system::ChainContext::::default(); - check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) - } - fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash - ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { - let lookup = frame_system::ChainContext::::default(); - - let maybe_signer_info = extract_signer_inner(uxt, &lookup); - let maybe_signer = if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error, - } - })?; - Some(signer) - } else { - None - }; + ) -> Result<(), TransactionValidityError> { + let lookup = frame_system::ChainContext::::default(); // Initializing block related storage required for validation System::initialize( - &(System::block_number() + BlockNumber::one()), + &(block_number + BlockNumber::one()), &block_hash, &Default::default(), ); - let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); - - let xt = uxt.clone().check(&lookup).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys.clone(), - } - })?; + let xt = uxt.clone().check(&lookup)?; let dispatch_info = xt.get_dispatch_info(); if dispatch_info.class == DispatchClass::Mandatory { - return Err( domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: InvalidTransaction::BadMandatory.into(), - storage_keys: storage_keys.clone(), - }) + return Err(InvalidTransaction::BadMandatory.into()); } let encoded = uxt.encode(); @@ -1017,50 +921,20 @@ impl_runtime_apis! { // runtime instance. match xt.signed { CheckedSignature::Signed(account_id, extra) => { - extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys, - } - }) + extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()) }, CheckedSignature::Unsigned => { - Runtime::pre_dispatch(&xt.function).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys.clone(), - } - })?; - SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys, - } - }) + Runtime::pre_dispatch(&xt.function).map(|_| ())?; + SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()) }, CheckedSignature::SelfContained(self_contained_signing_info) => { xt.function.pre_dispatch_self_contained(&self_contained_signing_info, &dispatch_info, encoded_len).ok_or(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - // TODO: Verify the storage keys here - storage_keys, - } - }) + InvalidTransaction::Call, + )).map(|_| ()) } } } - fn storage_keys_for_verifying_transaction_validity( - maybe_who: Option, - block_number: BlockNumber, - maybe_tx_era: Option - ) -> Result>, domain_runtime_primitives::VerifyTxValidityError> { - let maybe_sender = maybe_who.map(|who| AccountId::decode(&mut who.as_slice()).map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)).transpose()?; - Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) - } - fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { diff --git a/domains/test/runtime/evm/src/lib.rs b/domains/test/runtime/evm/src/lib.rs index 26f7c59aac..e6cab20a52 100644 --- a/domains/test/runtime/evm/src/lib.rs +++ b/domains/test/runtime/evm/src/lib.rs @@ -18,7 +18,7 @@ use frame_support::dispatch::{DispatchClass, GetDispatchInfo}; use frame_support::inherent::ProvideInherent; use frame_support::traits::{ ConstU16, ConstU32, ConstU64, Currency, Everything, FindAuthor, Imbalance, OnFinalize, - OnUnbalanced, PalletInfoAccess, + OnUnbalanced, }; use frame_support::weights::constants::{ BlockExecutionWeight, ExtrinsicBaseWeight, ParityDbWeight, WEIGHT_REF_TIME_PER_MILLIS, @@ -55,7 +55,6 @@ use sp_runtime::transaction_validity::{ }; use sp_runtime::{ create_runtime_str, generic, impl_opaque_keys, ApplyExtrinsicResult, ConsensusEngineId, Digest, - SaturatedConversion, }; pub use sp_runtime::{MultiAddress, Perbill, Permill}; use sp_std::marker::PhantomData; @@ -703,35 +702,6 @@ pub fn extract_signer( .collect() } -fn storage_keys_for_verifying_tx_validity_inner( - maybe_sender: Option, - block_number: BlockNumber, - maybe_tx_era: Option, -) -> Vec> { - let mut storage_keys = sp_std::vec![ - frame_system::BlockHash::::hashed_key_for(BlockNumber::from(0u32)), - frame_support::storage::storage_prefix(System::name().as_bytes(), b"Number").to_vec(), - pallet_transaction_payment::NextFeeMultiplier::::hashed_key().to_vec() - ]; - - if let Some(sender) = maybe_sender { - storage_keys.push(frame_system::Account::::hashed_key_for(sender)); - } - - match maybe_tx_era { - None => storage_keys, - Some(era) => { - let birth_number = era - .birth(block_number.saturated_into::()) - .saturated_into::(); - storage_keys.push(frame_system::BlockHash::::hashed_key_for( - birth_number, - )); - storage_keys - } - } -} - fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { extrinsic .0 @@ -740,44 +710,6 @@ fn extrinsic_era(extrinsic: &::Extrinsic) -> Option { .map(|(_, _, extra)| extra.4 .0) } -fn check_transaction_validity_inner( - lookup: &Lookup, - uxt: &::Extrinsic, - block_number: BlockNumber, - block_hash: ::Hash, -) -> Result, domain_runtime_primitives::CheckTxValidityError> -where - Lookup: sp_runtime::traits::Lookup, -{ - let maybe_signer_info = extract_signer_inner(uxt, lookup); - let maybe_signer = if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error, - } - })?; - Some(signer) - } else { - None - }; - - let tx_validity = - Executive::validate_transaction(TransactionSource::External, uxt.clone(), block_hash); - - tx_validity - .map(|_| maybe_signer) - .map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys_for_verifying_tx_validity_inner( - maybe_signer, - block_number, - extrinsic_era(uxt), - ), - } - }) -} - impl_runtime_apis! { impl sp_api::Core for Runtime { fn version() -> RuntimeVersion { @@ -946,55 +878,26 @@ impl_runtime_apis! { } } - fn check_transaction_validity(uxt: &::Extrinsic, - block_number: BlockNumber, - block_hash: ::Hash) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { - let lookup = frame_system::ChainContext::::default(); - check_transaction_validity_inner(&lookup, uxt, block_number, block_hash).map(|_| ()) - } - fn check_transaction_and_do_pre_dispatch( uxt: &::Extrinsic, block_number: BlockNumber, block_hash: ::Hash - ) -> Result<(), domain_runtime_primitives::CheckTxValidityError> { + ) -> Result<(), TransactionValidityError> { let lookup = frame_system::ChainContext::::default(); - let maybe_signer_info = extract_signer_inner(uxt, &lookup); - let maybe_signer = if let Some(signer_info) = maybe_signer_info { - let signer = signer_info.map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::UnableToExtractSigner { - error: tx_validity_error, - } - })?; - Some(signer) - } else { - None - }; - // Initializing block related storage required for validation System::initialize( - &(System::block_number() + BlockNumber::one()), + &(block_number + BlockNumber::one()), &block_hash, &Default::default(), ); - let storage_keys = storage_keys_for_verifying_tx_validity_inner(maybe_signer, block_number, Self::extrinsic_era(uxt)); - - let xt = uxt.clone().check(&lookup).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys.clone(), - } - })?; + let xt = uxt.clone().check(&lookup)?; let dispatch_info = xt.get_dispatch_info(); if dispatch_info.class == DispatchClass::Mandatory { - return Err( domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: InvalidTransaction::BadMandatory.into(), - storage_keys: storage_keys.clone(), - }) + return Err(InvalidTransaction::BadMandatory.into()); } let encoded = uxt.encode(); @@ -1005,50 +908,20 @@ impl_runtime_apis! { // runtime instance. match xt.signed { CheckedSignature::Signed(account_id, extra) => { - extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys, - } - }) + extra.pre_dispatch(&account_id, &xt.function, &dispatch_info, encoded_len).map(|_| ()) }, CheckedSignature::Unsigned => { - Runtime::pre_dispatch(&xt.function).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys: storage_keys.clone(), - } - })?; - SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - storage_keys, - } - }) + Runtime::pre_dispatch(&xt.function).map(|_| ())?; + SignedExtra::pre_dispatch_unsigned(&xt.function, &dispatch_info, encoded_len).map(|_| ()) }, CheckedSignature::SelfContained(self_contained_signing_info) => { xt.function.pre_dispatch_self_contained(&self_contained_signing_info, &dispatch_info, encoded_len).ok_or(TransactionValidityError::Invalid( - InvalidTransaction::BadProof, - )).map(|_| ()).map_err(|tx_validity_error| { - domain_runtime_primitives::CheckTxValidityError::InvalidTransaction { - error: tx_validity_error, - // TODO: Verify the storage keys here - storage_keys, - } - }) + InvalidTransaction::Call, + )).map(|_| ()) } } } - fn storage_keys_for_verifying_transaction_validity( - maybe_who: Option, - block_number: BlockNumber, - maybe_tx_era: Option - ) -> Result>, domain_runtime_primitives::VerifyTxValidityError> { - let maybe_sender = maybe_who.map(|who| AccountId::decode(&mut who.as_slice()).map_err(|_| domain_runtime_primitives::VerifyTxValidityError::FailedToDecodeAccountId)).transpose()?; - Ok(storage_keys_for_verifying_tx_validity_inner(maybe_sender, block_number, maybe_tx_era)) - } - fn extrinsic_era( extrinsic: &::Extrinsic ) -> Option { diff --git a/domains/test/service/src/domain.rs b/domains/test/service/src/domain.rs index eed107a19d..d73f4c40ab 100644 --- a/domains/test/service/src/domain.rs +++ b/domains/test/service/src/domain.rs @@ -2,7 +2,9 @@ #![warn(missing_docs)] use crate::chain_spec::create_domain_spec; -use crate::{construct_extrinsic_generic, node_config, EcdsaKeyring, UncheckedExtrinsicFor}; +use crate::{ + construct_extrinsic_generic, node_config, BalanceOf, EcdsaKeyring, UncheckedExtrinsicFor, +}; use domain_client_operator::{BootstrapResult, Bootstrapper, OperatorStreams}; use domain_runtime_primitives::opaque::Block; use domain_runtime_primitives::{Balance, DomainCoreApi}; @@ -338,7 +340,7 @@ where self.key, false, self.account_nonce(), - 0, + 0.into(), ); self.rpc_handlers.send_transaction(extrinsic.into()).await } @@ -349,14 +351,21 @@ where nonce: u32, function: impl Into<::RuntimeCall>, ) -> UncheckedExtrinsicFor { - construct_extrinsic_generic::(&self.client, function, self.key, false, nonce, 0) + construct_extrinsic_generic::( + &self.client, + function, + self.key, + false, + nonce, + 0.into(), + ) } /// Construct an extrinsic with the given transaction tip. pub fn construct_extrinsic_with_tip( &mut self, nonce: u32, - tip: u32, + tip: BalanceOf, function: impl Into<::RuntimeCall>, ) -> UncheckedExtrinsicFor { construct_extrinsic_generic::( diff --git a/domains/test/service/src/lib.rs b/domains/test/service/src/lib.rs index 249713bfff..72a2e4d74b 100644 --- a/domains/test/service/src/lib.rs +++ b/domains/test/service/src/lib.rs @@ -177,7 +177,7 @@ pub fn construct_extrinsic_generic( caller: EcdsaKeyring, immortal: bool, nonce: u32, - tip: u32, + tip: BalanceOf, ) -> UncheckedExtrinsicFor where Runtime: frame_system::Config + pallet_transaction_payment::Config + Send + Sync, @@ -207,7 +207,7 @@ where }), frame_system::CheckNonce::::from(nonce.into()), frame_system::CheckWeight::::new(), - pallet_transaction_payment::ChargeTransactionPayment::::from(tip.into()), + pallet_transaction_payment::ChargeTransactionPayment::::from(tip), ); let raw_payload = generic::SignedPayload::< ::RuntimeCall, From 6dd44494e1e64a014fcea9e3b5903845c27192f8 Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Sun, 10 Dec 2023 09:29:20 +0400 Subject: [PATCH 6/7] remove fp-rpc from dev dependency as it is not used --- Cargo.lock | 1 - crates/sp-domains-fraud-proof/Cargo.toml | 1 - 2 files changed, 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 4b1faac7ad..8251c60afd 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -10984,7 +10984,6 @@ dependencies = [ "domain-block-preprocessor", "domain-runtime-primitives", "domain-test-service", - "fp-rpc", "frame-support", "frame-system", "futures", diff --git a/crates/sp-domains-fraud-proof/Cargo.toml b/crates/sp-domains-fraud-proof/Cargo.toml index feb4bc223d..54dd4efc5b 100644 --- a/crates/sp-domains-fraud-proof/Cargo.toml +++ b/crates/sp-domains-fraud-proof/Cargo.toml @@ -40,7 +40,6 @@ thiserror = { version = "1.0.48", optional = true } domain-block-builder = { version = "0.1.0", path = "../../domains/client/block-builder" } domain-block-preprocessor = { version = "0.1.0", path = "../../domains/client/block-preprocessor" } domain-test-service = { version = "0.1.0", path = "../../domains/test/service" } -fp-rpc = { version = "3.0.0-dev", git = "https://github.com/subspace/frontier", rev = "56086daa77802eaa285894bfe4b811be66629c89", features = ['default'] } frame-system = { version = "4.0.0-dev", default-features = false, git = "https://github.com/subspace/polkadot-sdk", rev = "892bf8e938c6bd2b893d3827d1093cd81baa59a1" } futures = "0.3.29" pallet-balances = { version = "4.0.0-dev", git = "https://github.com/subspace/polkadot-sdk", rev = "892bf8e938c6bd2b893d3827d1093cd81baa59a1" } From 66178a40d8b3d90c82cbd4702226431672d9241f Mon Sep 17 00:00:00 2001 From: Parth Desai Date: Tue, 12 Dec 2023 20:25:24 +0400 Subject: [PATCH 7/7] addressing minor nits --- crates/sp-domains-fraud-proof/src/tests.rs | 2 +- domains/client/block-preprocessor/src/lib.rs | 1 - domains/runtime/evm/src/lib.rs | 6 +++--- domains/test/runtime/evm/src/lib.rs | 6 +++--- 4 files changed, 7 insertions(+), 8 deletions(-) diff --git a/crates/sp-domains-fraud-proof/src/tests.rs b/crates/sp-domains-fraud-proof/src/tests.rs index 5bc1db18b4..946d01a2d5 100644 --- a/crates/sp-domains-fraud-proof/src/tests.rs +++ b/crates/sp-domains-fraud-proof/src/tests.rs @@ -12,7 +12,7 @@ use subspace_test_service::MockConsensusNode; use tempfile::TempDir; #[tokio::test(flavor = "multi_thread")] -async fn runtime_instance_storage_assumptions_are_correct() { +async fn storage_change_of_the_same_runtime_instance_should_perserved_cross_runtime_calls() { let directory = TempDir::new().expect("Must be able to create temporary directory"); let mut builder = sc_cli::LoggerBuilder::new(""); diff --git a/domains/client/block-preprocessor/src/lib.rs b/domains/client/block-preprocessor/src/lib.rs index 43618f3394..f5b255dac8 100644 --- a/domains/client/block-preprocessor/src/lib.rs +++ b/domains/client/block-preprocessor/src/lib.rs @@ -310,7 +310,6 @@ where .is_ok(); if !is_legal_tx { - // TODO: Generate a fraud proof for this invalid bundle return Ok(BundleValidity::Invalid(InvalidBundleType::IllegalTx( index as u32, ))); diff --git a/domains/runtime/evm/src/lib.rs b/domains/runtime/evm/src/lib.rs index 779c2384b1..312c722bdb 100644 --- a/domains/runtime/evm/src/lib.rs +++ b/domains/runtime/evm/src/lib.rs @@ -913,10 +913,10 @@ impl_runtime_apis! { return Err(InvalidTransaction::BadMandatory.into()); } - let encoded = uxt.encode(); - let encoded_len = encoded.len(); + let encoded_len = uxt.encoded_size(); - // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) as that will add the side effect of SignedExtension in the storage buffer + // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) + // as that will add the side effect of SignedExtension in the storage buffer // which would help to maintain context across multiple transaction validity check against same // runtime instance. match xt.signed { diff --git a/domains/test/runtime/evm/src/lib.rs b/domains/test/runtime/evm/src/lib.rs index e6cab20a52..00b925993b 100644 --- a/domains/test/runtime/evm/src/lib.rs +++ b/domains/test/runtime/evm/src/lib.rs @@ -900,10 +900,10 @@ impl_runtime_apis! { return Err(InvalidTransaction::BadMandatory.into()); } - let encoded = uxt.encode(); - let encoded_len = encoded.len(); + let encoded_len = uxt.encoded_size(); - // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) as that will add the side effect of SignedExtension in the storage buffer + // We invoke `pre_dispatch` in addition to `validate_transaction`(even though the validation is almost same) + // as that will add the side effect of SignedExtension in the storage buffer // which would help to maintain context across multiple transaction validity check against same // runtime instance. match xt.signed {