datahaven/operator/runtime/mainnet/tests/lib.rs
Gonza Montiel 782321e5d0
feat: Implement dynamic fee adjustment (#251)
#  Implement Dynamic Fee Adjustment Mechanism

## Overview
Implements a dynamic fee adjustment mechanism, replacing the constant
fee multiplier with an adaptive multiplier that responds to network
congestion, following Moonbeam's pattern.

## Changes
- Replaces `ConstFeeMultiplier` with `TargetedFeeAdjustment` across all
runtime configurations (mainnet, stagenet, testnet)
- Implements an EIP-1559-like slow-adjusting fee mechanism that prevents
DoS attacks by adjusting fees based on block fullness
- **Configurable Parameters**: 
  - Target block fullness: 35%
- Adjustment variable: 4/1000 (responds in ~1 hour at extreme
congestion)
  - Two modes:
    -  `SlowAdjustingFeeUpdate` for mainnet and testnet.
    - `FastAdjustingFeeUpdate` for stagenet.
- Adds tests coverage for different fee scenarios

## Technical Details

The fee adjustment algorithm works as follows:
```
diff = (previous_block_weight - target) / maximum_block_weight
next_multiplier = prev_multiplier * (1 + (v * diff) + ((v * diff)^2 / 2))
assert(next_multiplier > min)
```
**Where:**
- `v` = AdjustmentVariable
- `target` = TargetBlockFullness  
- `min` = MinimumMultiplier

`SlowAdjustingFeeUpdate` sets a minimum multiplier of `1x` for a
conservative fee adjustment, while `FastAdjustingFeeUpdate` sets it to
`0.1x`, which is mainly used for dev networks / testing.
2025-10-28 10:06:45 +00:00

63 lines
1.9 KiB
Rust

//! Integration tests for DataHaven mainnet runtime
pub mod common;
mod fee_adjustment;
pub mod governance;
mod migrations;
mod native_token_transfer;
mod proxy;
mod safe_mode_tx_pause;
use common::*;
use datahaven_mainnet_runtime::{
currency::HAVE, Balances, Runtime, System, UncheckedExtrinsic, VERSION,
};
use sp_core::H160;
use sp_runtime::transaction_validity::{
InvalidTransaction, TransactionSource, TransactionValidityError,
};
use sp_transaction_pool::runtime_api::runtime_decl_for_tagged_transaction_queue::TaggedTransactionQueueV3;
// Runtime Tests
#[test]
fn test_runtime_version_and_metadata() {
ExtBuilder::default().build().execute_with(|| {
assert!(!VERSION.spec_name.is_empty());
assert!(VERSION.spec_version > 0);
assert_eq!(System::block_number(), 1);
});
}
#[test]
fn test_balances_functionality() {
ExtBuilder::default()
.with_balances(vec![(account_id(ALICE), 2_000_000 * HAVE)])
.build()
.execute_with(|| {
assert_eq!(Balances::free_balance(&account_id(ALICE)), 2_000_000 * HAVE);
});
}
#[test]
fn validate_transaction_fails_on_filtered_call() {
ExtBuilder::default().build().execute_with(|| {
let xt = UncheckedExtrinsic::new_bare(
pallet_evm::Call::<Runtime>::call {
source: H160::default(),
target: H160::default(),
input: Vec::new(),
value: sp_core::U256::zero(),
gas_limit: 21000,
max_fee_per_gas: sp_core::U256::zero(),
max_priority_fee_per_gas: Some(sp_core::U256::zero()),
nonce: None,
access_list: Vec::new(),
}
.into(),
);
assert_eq!(
Runtime::validate_transaction(TransactionSource::External, xt, Default::default(),),
Err(TransactionValidityError::Invalid(InvalidTransaction::Call)),
);
});
}