Skip to content

Commit

Permalink
Limit recursion when checking for balance transfers or contract creation
Browse files Browse the repository at this point in the history
  • Loading branch information
teor2345 committed Jan 17, 2025
1 parent 850ac44 commit 37b3ed3
Show file tree
Hide file tree
Showing 2 changed files with 38 additions and 12 deletions.
25 changes: 19 additions & 6 deletions crates/subspace-runtime/src/signed_extensions.rs
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ use sp_runtime::transaction_validity::{
};
use sp_std::prelude::*;

const MAX_BALANCE_RECURSION_DEPTH: u16 = 5;

/// Disable balance transfers, if configured in the runtime.
#[derive(Debug, Encode, Decode, Clone, Eq, PartialEq, Default, TypeInfo)]
pub struct DisablePallets;
Expand All @@ -30,7 +32,9 @@ impl SignedExtension for DisablePallets {
_len: usize,
) -> TransactionValidity {
// Disable normal balance transfers.
if !RuntimeConfigs::enable_balance_transfers() && contains_balance_transfer(call) {
if !RuntimeConfigs::enable_balance_transfers()
&& contains_balance_transfer(call, MAX_BALANCE_RECURSION_DEPTH)
{
InvalidTransaction::Call.into()
} else {
Ok(ValidTransaction::default())
Expand Down Expand Up @@ -61,7 +65,14 @@ impl SignedExtension for DisablePallets {
}
}

fn contains_balance_transfer(call: &RuntimeCall) -> bool {
fn contains_balance_transfer(call: &RuntimeCall, mut recursion_depth_left: u16) -> bool {
if recursion_depth_left == 0 {
// If the recursion depth is reached, we assume the call contains a balance transfer.
return true;
}

recursion_depth_left -= 1;

match call {
RuntimeCall::Balances(
pallet_balances::Call::transfer_allow_death { .. }
Expand All @@ -71,12 +82,14 @@ fn contains_balance_transfer(call: &RuntimeCall) -> bool {
RuntimeCall::Utility(utility_call) => match utility_call {
pallet_utility::Call::batch { calls }
| pallet_utility::Call::batch_all { calls }
| pallet_utility::Call::force_batch { calls } => {
calls.iter().any(contains_balance_transfer)
}
| pallet_utility::Call::force_batch { calls } => calls
.iter()
.any(|call| contains_balance_transfer(call, recursion_depth_left)),
pallet_utility::Call::as_derivative { call, .. }
| pallet_utility::Call::dispatch_as { call, .. }
| pallet_utility::Call::with_weight { call, .. } => contains_balance_transfer(call),
| pallet_utility::Call::with_weight { call, .. } => {
contains_balance_transfer(call, recursion_depth_left)
}
pallet_utility::Call::__Ignore(..) => false,
},
_ => false,
Expand Down
25 changes: 19 additions & 6 deletions domains/runtime/evm/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -148,12 +148,15 @@ pub type Executive = domain_pallet_executive::Executive<
AllPalletsWithSystem,
>;

const MAX_CONTRACT_RECURSION_DEPTH: u16 = 5;

/// Rejects contracts that can't be created under the current allow list.
/// Returns false if the call is a contract call, and the account is *not* allowed to call it.
/// Otherwise, returns true.
pub fn is_create_contract_allowed(call: &RuntimeCall, signer: &AccountId) -> bool {
if is_create_contract(call)
&& !pallet_evm_nonce_tracker::Pallet::<Runtime>::is_allowed_to_create_contracts(signer)
// Only enter recursive code if this account can't create contracts
if !pallet_evm_nonce_tracker::Pallet::<Runtime>::is_allowed_to_create_contracts(signer)
&& is_create_contract(call, MAX_CONTRACT_RECURSION_DEPTH)
{
return false;
}
Expand All @@ -163,7 +166,14 @@ pub fn is_create_contract_allowed(call: &RuntimeCall, signer: &AccountId) -> boo
}

/// Returns true if the call is a contract creation call.
pub fn is_create_contract(call: &RuntimeCall) -> bool {
pub fn is_create_contract(call: &RuntimeCall, mut recursion_depth_left: u16) -> bool {
if recursion_depth_left == 0 {
// If the recursion depth is reached, we assume the call contains a contract.
return true;
}

recursion_depth_left -= 1;

match call {
RuntimeCall::EVM(pallet_evm::Call::create { .. })
| RuntimeCall::EVM(pallet_evm::Call::create2 { .. }) => true,
Expand All @@ -179,14 +189,17 @@ pub fn is_create_contract(call: &RuntimeCall) -> bool {
transaction: EthereumTransaction::EIP1559(transaction),
..
}) => transaction.action == TransactionAction::Create,
// TODO: does this need a recursion limit?
RuntimeCall::Utility(utility_call) => match utility_call {
pallet_utility::Call::batch { calls }
| pallet_utility::Call::batch_all { calls }
| pallet_utility::Call::force_batch { calls } => calls.iter().any(is_create_contract),
| pallet_utility::Call::force_batch { calls } => calls
.iter()
.any(|call| is_create_contract(call, recursion_depth_left)),
pallet_utility::Call::as_derivative { call, .. }
| pallet_utility::Call::dispatch_as { call, .. }
| pallet_utility::Call::with_weight { call, .. } => is_create_contract(call),
| pallet_utility::Call::with_weight { call, .. } => {
is_create_contract(call, recursion_depth_left)
}
pallet_utility::Call::__Ignore(..) => false,
},
_ => false,
Expand Down

0 comments on commit 37b3ed3

Please sign in to comment.