use crate::{limits::BlockWeights, Config, Pallet, LOG_TARGET};
use codec::{Decode, Encode};
use frame_support::{
dispatch::{DispatchInfo, PostDispatchInfo},
traits::Get,
};
use scale_info::TypeInfo;
use sp_runtime::{
traits::{
DispatchInfoOf, Dispatchable, PostDispatchInfoOf, TransactionExtension, ValidateResult,
},
transaction_validity::{InvalidTransaction, TransactionValidityError, ValidTransaction},
DispatchResult,
};
use sp_weights::Weight;
#[derive(Encode, Decode, Clone, Eq, PartialEq, Default, TypeInfo)]
#[scale_info(skip_type_params(T))]
pub struct CheckWeight<T: Config + Send + Sync>(core::marker::PhantomData<T>);
impl<T: Config + Send + Sync> CheckWeight<T>
where
T::RuntimeCall: Dispatchable<Info = DispatchInfo, PostInfo = PostDispatchInfo>,
{
fn check_extrinsic_weight(
info: &DispatchInfoOf<T::RuntimeCall>,
) -> Result<(), TransactionValidityError> {
let max = T::BlockWeights::get().get(info.class).max_extrinsic;
match max {
Some(max) if info.total_weight().any_gt(max) => {
log::debug!(
target: LOG_TARGET,
"Extrinsic {} is greater than the max extrinsic {}",
info.total_weight(),
max,
);
Err(InvalidTransaction::ExhaustsResources.into())
},
_ => Ok(()),
}
}
fn check_block_length(
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
) -> Result<u32, TransactionValidityError> {
let length_limit = T::BlockLength::get();
let current_len = Pallet::<T>::all_extrinsics_len();
let added_len = len as u32;
let next_len = current_len.saturating_add(added_len);
if next_len > *length_limit.max.get(info.class) {
log::debug!(
target: LOG_TARGET,
"Exceeded block length limit: {} > {}",
next_len,
length_limit.max.get(info.class),
);
Err(InvalidTransaction::ExhaustsResources.into())
} else {
Ok(next_len)
}
}
pub fn new() -> Self {
Self(Default::default())
}
pub fn do_validate(
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
) -> Result<(ValidTransaction, u32), TransactionValidityError> {
let next_len = Self::check_block_length(info, len)?;
Self::check_extrinsic_weight(info)?;
Ok((Default::default(), next_len))
}
pub fn do_prepare(
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
next_len: u32,
) -> Result<(), TransactionValidityError> {
let all_weight = Pallet::<T>::block_weight();
let maximum_weight = T::BlockWeights::get();
let next_weight =
calculate_consumed_weight::<T::RuntimeCall>(&maximum_weight, all_weight, info, len)?;
crate::AllExtrinsicsLen::<T>::put(next_len);
crate::BlockWeight::<T>::put(next_weight);
Ok(())
}
pub fn do_post_dispatch(
info: &DispatchInfoOf<T::RuntimeCall>,
post_info: &PostDispatchInfoOf<T::RuntimeCall>,
) -> Result<(), TransactionValidityError> {
let unspent = post_info.calc_unspent(info);
if unspent.any_gt(Weight::zero()) {
crate::BlockWeight::<T>::mutate(|current_weight| {
current_weight.reduce(unspent, info.class);
})
}
log::trace!(
target: LOG_TARGET,
"Used block weight: {:?}",
crate::BlockWeight::<T>::get(),
);
log::trace!(
target: LOG_TARGET,
"Used block length: {:?}",
Pallet::<T>::all_extrinsics_len(),
);
Ok(())
}
}
pub fn calculate_consumed_weight<Call>(
maximum_weight: &BlockWeights,
mut all_weight: crate::ConsumedWeight,
info: &DispatchInfoOf<Call>,
len: usize,
) -> Result<crate::ConsumedWeight, TransactionValidityError>
where
Call: Dispatchable<Info = DispatchInfo, PostInfo = PostDispatchInfo>,
{
let extrinsic_weight = info
.total_weight()
.saturating_add(maximum_weight.get(info.class).base_extrinsic)
.saturating_add(Weight::from_parts(0, len as u64));
let limit_per_class = maximum_weight.get(info.class);
if limit_per_class.max_total.is_none() && limit_per_class.reserved.is_none() {
all_weight.accrue(extrinsic_weight, info.class)
} else {
all_weight.checked_accrue(extrinsic_weight, info.class).map_err(|_| {
log::debug!(
target: LOG_TARGET,
"All weight checked add overflow.",
);
InvalidTransaction::ExhaustsResources
})?;
}
let per_class = *all_weight.get(info.class);
match limit_per_class.max_total {
Some(max) if per_class.any_gt(max) => {
log::debug!(
target: LOG_TARGET,
"Exceeded the per-class allowance.",
);
return Err(InvalidTransaction::ExhaustsResources.into());
},
_ => {},
}
if all_weight.total().any_gt(maximum_weight.max_block) {
match limit_per_class.reserved {
Some(reserved) if per_class.any_gt(reserved) => {
log::debug!(
target: LOG_TARGET,
"Total block weight is exceeded.",
);
return Err(InvalidTransaction::ExhaustsResources.into());
},
_ => {},
}
}
Ok(all_weight)
}
impl<T: Config + Send + Sync> TransactionExtension<T::RuntimeCall> for CheckWeight<T>
where
T::RuntimeCall: Dispatchable<Info = DispatchInfo, PostInfo = PostDispatchInfo>,
{
const IDENTIFIER: &'static str = "CheckWeight";
type Implicit = ();
type Pre = ();
type Val = u32; fn weight(&self, _: &T::RuntimeCall) -> Weight {
<T::ExtensionsWeightInfo as super::WeightInfo>::check_weight()
}
fn validate(
&self,
origin: T::RuntimeOrigin,
_call: &T::RuntimeCall,
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
_self_implicit: Self::Implicit,
_inherited_implication: &impl Encode,
) -> ValidateResult<Self::Val, T::RuntimeCall> {
let (validity, next_len) = Self::do_validate(info, len)?;
Ok((validity, next_len, origin))
}
fn prepare(
self,
val: Self::Val,
_origin: &T::RuntimeOrigin,
_call: &T::RuntimeCall,
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
) -> Result<Self::Pre, TransactionValidityError> {
Self::do_prepare(info, len, val)
}
fn post_dispatch_details(
_pre: Self::Pre,
info: &DispatchInfoOf<T::RuntimeCall>,
post_info: &PostDispatchInfoOf<T::RuntimeCall>,
_len: usize,
_result: &DispatchResult,
) -> Result<Weight, TransactionValidityError> {
Self::do_post_dispatch(info, post_info)?;
Ok(Weight::zero())
}
fn bare_validate(
_call: &T::RuntimeCall,
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
) -> frame_support::pallet_prelude::TransactionValidity {
Ok(Self::do_validate(info, len)?.0)
}
fn bare_validate_and_prepare(
_call: &T::RuntimeCall,
info: &DispatchInfoOf<T::RuntimeCall>,
len: usize,
) -> Result<(), TransactionValidityError> {
let (_, next_len) = Self::do_validate(info, len)?;
Self::do_prepare(info, len, next_len)
}
fn bare_post_dispatch(
info: &DispatchInfoOf<T::RuntimeCall>,
post_info: &mut PostDispatchInfoOf<T::RuntimeCall>,
_len: usize,
_result: &DispatchResult,
) -> Result<(), TransactionValidityError> {
Self::do_post_dispatch(info, post_info)
}
}
impl<T: Config + Send + Sync> core::fmt::Debug for CheckWeight<T> {
#[cfg(feature = "std")]
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
write!(f, "CheckWeight")
}
#[cfg(not(feature = "std"))]
fn fmt(&self, _: &mut core::fmt::Formatter) -> core::fmt::Result {
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{
mock::{new_test_ext, System, Test, CALL},
AllExtrinsicsLen, BlockWeight, DispatchClass,
};
use core::marker::PhantomData;
use frame_support::{assert_err, assert_ok, dispatch::Pays, weights::Weight};
use sp_runtime::traits::DispatchTransaction;
fn block_weights() -> crate::limits::BlockWeights {
<Test as crate::Config>::BlockWeights::get()
}
fn normal_weight_limit() -> Weight {
block_weights()
.get(DispatchClass::Normal)
.max_total
.unwrap_or_else(|| block_weights().max_block)
}
fn block_weight_limit() -> Weight {
block_weights().max_block
}
fn normal_length_limit() -> u32 {
*<Test as Config>::BlockLength::get().max.get(DispatchClass::Normal)
}
#[test]
fn mandatory_extrinsic_doesnt_care_about_limits() {
fn check(call: impl FnOnce(&DispatchInfo, usize)) {
new_test_ext().execute_with(|| {
let max = DispatchInfo {
call_weight: Weight::MAX,
class: DispatchClass::Mandatory,
..Default::default()
};
let len = 0_usize;
call(&max, len);
});
}
check(|max, len| {
let next_len = CheckWeight::<Test>::check_block_length(max, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(max, len, next_len));
assert_eq!(System::block_weight().total(), Weight::MAX);
assert!(System::block_weight().total().ref_time() > block_weight_limit().ref_time());
});
check(|max, len| {
assert_ok!(CheckWeight::<Test>::do_validate(max, len));
});
}
#[test]
fn normal_extrinsic_limited_by_maximum_extrinsic_weight() {
new_test_ext().execute_with(|| {
let max = DispatchInfo {
call_weight: block_weights().get(DispatchClass::Normal).max_extrinsic.unwrap() +
Weight::from_parts(1, 0),
class: DispatchClass::Normal,
..Default::default()
};
let len = 0_usize;
assert_err!(
CheckWeight::<Test>::do_validate(&max, len),
InvalidTransaction::ExhaustsResources
);
});
}
#[test]
fn operational_extrinsic_limited_by_operational_space_limit() {
new_test_ext().execute_with(|| {
let weights = block_weights();
let operational_limit = weights
.get(DispatchClass::Operational)
.max_total
.unwrap_or_else(|| weights.max_block);
let base_weight = weights.get(DispatchClass::Operational).base_extrinsic;
let call_weight = operational_limit - base_weight;
let okay = DispatchInfo {
call_weight,
class: DispatchClass::Operational,
..Default::default()
};
let max = DispatchInfo {
call_weight: call_weight + Weight::from_parts(1, 0),
class: DispatchClass::Operational,
..Default::default()
};
let len = 0_usize;
assert_eq!(CheckWeight::<Test>::do_validate(&okay, len), Ok(Default::default()));
assert_err!(
CheckWeight::<Test>::do_validate(&max, len),
InvalidTransaction::ExhaustsResources
);
});
}
#[test]
fn register_extra_weight_unchecked_doesnt_care_about_limits() {
new_test_ext().execute_with(|| {
System::register_extra_weight_unchecked(Weight::MAX, DispatchClass::Normal);
assert_eq!(System::block_weight().total(), Weight::MAX);
assert!(System::block_weight().total().ref_time() > block_weight_limit().ref_time());
});
}
#[test]
fn full_block_with_normal_and_operational() {
new_test_ext().execute_with(|| {
let max_normal =
DispatchInfo { call_weight: Weight::from_parts(753, 0), ..Default::default() };
let rest_operational = DispatchInfo {
call_weight: Weight::from_parts(246, 0),
class: DispatchClass::Operational,
..Default::default()
};
let len = 0_usize;
let next_len = CheckWeight::<Test>::check_block_length(&max_normal, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&max_normal, len, next_len));
assert_eq!(System::block_weight().total(), Weight::from_parts(768, 0));
let next_len = CheckWeight::<Test>::check_block_length(&rest_operational, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&rest_operational, len, next_len));
assert_eq!(block_weight_limit(), Weight::from_parts(1024, u64::MAX));
assert_eq!(System::block_weight().total(), block_weight_limit().set_proof_size(0));
assert_eq!(CheckWeight::<Test>::check_extrinsic_weight(&rest_operational), Ok(()));
});
}
#[test]
fn dispatch_order_does_not_effect_weight_logic() {
new_test_ext().execute_with(|| {
let max_normal =
DispatchInfo { call_weight: Weight::from_parts(753, 0), ..Default::default() };
let rest_operational = DispatchInfo {
call_weight: Weight::from_parts(246, 0),
class: DispatchClass::Operational,
..Default::default()
};
let len = 0_usize;
let next_len = CheckWeight::<Test>::check_block_length(&rest_operational, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&rest_operational, len, next_len));
assert_eq!(System::block_weight().total(), Weight::from_parts(266, 0));
let next_len = CheckWeight::<Test>::check_block_length(&max_normal, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&max_normal, len, next_len));
assert_eq!(block_weight_limit(), Weight::from_parts(1024, u64::MAX));
assert_eq!(System::block_weight().total(), block_weight_limit().set_proof_size(0));
});
}
#[test]
fn operational_works_on_full_block() {
new_test_ext().execute_with(|| {
System::register_extra_weight_unchecked(Weight::MAX, DispatchClass::Mandatory);
let dispatch_normal = DispatchInfo {
call_weight: Weight::from_parts(251, 0),
class: DispatchClass::Normal,
..Default::default()
};
let dispatch_operational = DispatchInfo {
call_weight: Weight::from_parts(246, 0),
class: DispatchClass::Operational,
..Default::default()
};
let len = 0_usize;
let next_len = CheckWeight::<Test>::check_block_length(&dispatch_normal, len).unwrap();
assert_err!(
CheckWeight::<Test>::do_prepare(&dispatch_normal, len, next_len),
InvalidTransaction::ExhaustsResources
);
let next_len =
CheckWeight::<Test>::check_block_length(&dispatch_operational, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&dispatch_operational, len, next_len));
assert_err!(
CheckWeight::<Test>::do_prepare(&dispatch_operational, len, next_len),
InvalidTransaction::ExhaustsResources
);
assert_eq!(CheckWeight::<Test>::check_extrinsic_weight(&dispatch_operational), Ok(()));
});
}
#[test]
fn signed_ext_check_weight_works_operational_tx() {
new_test_ext().execute_with(|| {
let normal =
DispatchInfo { call_weight: Weight::from_parts(100, 0), ..Default::default() };
let op = DispatchInfo {
call_weight: Weight::from_parts(100, 0),
extension_weight: Weight::zero(),
class: DispatchClass::Operational,
pays_fee: Pays::Yes,
};
let len = 0_usize;
let normal_limit = normal_weight_limit();
BlockWeight::<Test>::mutate(|current_weight| {
current_weight.set(normal_limit, DispatchClass::Normal)
});
assert_eq!(
CheckWeight::<Test>(PhantomData)
.validate_and_prepare(Some(1).into(), CALL, &normal, len)
.unwrap_err(),
InvalidTransaction::ExhaustsResources.into()
);
assert_ok!(CheckWeight::<Test>(PhantomData).validate_and_prepare(
Some(1).into(),
CALL,
&op,
len
));
let len = 100_usize;
AllExtrinsicsLen::<Test>::put(normal_length_limit());
assert_eq!(
CheckWeight::<Test>(PhantomData)
.validate_and_prepare(Some(1).into(), CALL, &normal, len)
.unwrap_err(),
InvalidTransaction::ExhaustsResources.into()
);
assert_ok!(CheckWeight::<Test>(PhantomData).validate_and_prepare(
Some(1).into(),
CALL,
&op,
len
));
})
}
#[test]
fn signed_ext_check_weight_block_size_works() {
new_test_ext().execute_with(|| {
let normal = DispatchInfo::default();
let normal_limit = normal_weight_limit().ref_time() as usize;
let reset_check_weight = |tx, s, f| {
AllExtrinsicsLen::<Test>::put(0);
let r = CheckWeight::<Test>(PhantomData).validate_and_prepare(
Some(1).into(),
CALL,
tx,
s,
);
if f {
assert!(r.is_err())
} else {
assert!(r.is_ok())
}
};
reset_check_weight(&normal, normal_limit - 1, false);
reset_check_weight(&normal, normal_limit, false);
reset_check_weight(&normal, normal_limit + 1, true);
let op = DispatchInfo {
call_weight: Weight::zero(),
extension_weight: Weight::zero(),
class: DispatchClass::Operational,
pays_fee: Pays::Yes,
};
reset_check_weight(&op, normal_limit, false);
reset_check_weight(&op, normal_limit + 100, false);
reset_check_weight(&op, 1024, false);
reset_check_weight(&op, 1025, true);
})
}
#[test]
fn signed_ext_check_weight_works_normal_tx() {
new_test_ext().execute_with(|| {
let normal_limit = normal_weight_limit();
let small =
DispatchInfo { call_weight: Weight::from_parts(100, 0), ..Default::default() };
let base_extrinsic = block_weights().get(DispatchClass::Normal).base_extrinsic;
let medium =
DispatchInfo { call_weight: normal_limit - base_extrinsic, ..Default::default() };
let big = DispatchInfo {
call_weight: normal_limit - base_extrinsic + Weight::from_parts(1, 0),
..Default::default()
};
let len = 0_usize;
let reset_check_weight = |i, f, s| {
BlockWeight::<Test>::mutate(|current_weight| {
current_weight.set(s, DispatchClass::Normal)
});
let r = CheckWeight::<Test>(PhantomData).validate_and_prepare(
Some(1).into(),
CALL,
i,
len,
);
if f {
assert!(r.is_err())
} else {
assert!(r.is_ok())
}
};
reset_check_weight(&small, false, Weight::zero());
reset_check_weight(&medium, false, Weight::zero());
reset_check_weight(&big, true, Weight::from_parts(1, 0));
})
}
#[test]
fn signed_ext_check_weight_refund_works() {
new_test_ext().execute_with(|| {
let info =
DispatchInfo { call_weight: Weight::from_parts(512, 0), ..Default::default() };
let post_info = PostDispatchInfo {
actual_weight: Some(Weight::from_parts(128, 0)),
pays_fee: Default::default(),
};
let len = 0_usize;
let base_extrinsic = block_weights().get(DispatchClass::Normal).base_extrinsic;
BlockWeight::<Test>::mutate(|current_weight| {
current_weight.set(Weight::zero(), DispatchClass::Mandatory);
current_weight
.set(Weight::from_parts(256, 0) - base_extrinsic, DispatchClass::Normal);
});
let pre = CheckWeight::<Test>(PhantomData)
.validate_and_prepare(Some(1).into(), CALL, &info, len)
.unwrap()
.0;
assert_eq!(
BlockWeight::<Test>::get().total(),
info.total_weight() + Weight::from_parts(256, 0)
);
assert_ok!(CheckWeight::<Test>::post_dispatch_details(
pre,
&info,
&post_info,
len,
&Ok(())
));
assert_eq!(
BlockWeight::<Test>::get().total(),
post_info.actual_weight.unwrap() + Weight::from_parts(256, 0)
);
})
}
#[test]
fn signed_ext_check_weight_actual_weight_higher_than_max_is_capped() {
new_test_ext().execute_with(|| {
let info =
DispatchInfo { call_weight: Weight::from_parts(512, 0), ..Default::default() };
let post_info = PostDispatchInfo {
actual_weight: Some(Weight::from_parts(700, 0)),
pays_fee: Default::default(),
};
let len = 0_usize;
BlockWeight::<Test>::mutate(|current_weight| {
current_weight.set(Weight::zero(), DispatchClass::Mandatory);
current_weight.set(Weight::from_parts(128, 0), DispatchClass::Normal);
});
let pre = CheckWeight::<Test>(PhantomData)
.validate_and_prepare(Some(1).into(), CALL, &info, len)
.unwrap()
.0;
assert_eq!(
BlockWeight::<Test>::get().total(),
info.total_weight() +
Weight::from_parts(128, 0) +
block_weights().get(DispatchClass::Normal).base_extrinsic,
);
assert_ok!(CheckWeight::<Test>::post_dispatch_details(
pre,
&info,
&post_info,
len,
&Ok(())
));
assert_eq!(
BlockWeight::<Test>::get().total(),
info.total_weight() +
Weight::from_parts(128, 0) +
block_weights().get(DispatchClass::Normal).base_extrinsic,
);
})
}
#[test]
fn zero_weight_extrinsic_still_has_base_weight() {
new_test_ext().execute_with(|| {
let weights = block_weights();
let free = DispatchInfo { call_weight: Weight::zero(), ..Default::default() };
let len = 0_usize;
assert_eq!(System::block_weight().total(), weights.base_block);
assert_ok!(CheckWeight::<Test>(PhantomData).validate_and_prepare(
Some(1).into(),
CALL,
&free,
len
));
assert_eq!(
System::block_weight().total(),
weights.get(DispatchClass::Normal).base_extrinsic + weights.base_block
);
})
}
#[test]
fn normal_and_mandatory_tracked_separately() {
new_test_ext().execute_with(|| {
let max_normal =
DispatchInfo { call_weight: Weight::from_parts(753, 0), ..Default::default() };
let mandatory = DispatchInfo {
call_weight: Weight::from_parts(1019, 0),
class: DispatchClass::Mandatory,
..Default::default()
};
let len = 0_usize;
let next_len = CheckWeight::<Test>::check_block_length(&max_normal, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&max_normal, len, next_len));
assert_eq!(System::block_weight().total(), Weight::from_parts(768, 0));
let next_len = CheckWeight::<Test>::check_block_length(&mandatory, len).unwrap();
assert_ok!(CheckWeight::<Test>::do_prepare(&mandatory, len, next_len));
assert_eq!(block_weight_limit(), Weight::from_parts(1024, u64::MAX));
assert_eq!(System::block_weight().total(), Weight::from_parts(1024 + 768, 0));
assert_eq!(CheckWeight::<Test>::check_extrinsic_weight(&mandatory), Ok(()));
});
}
#[test]
fn no_max_total_should_still_be_limited_by_max_block() {
let maximum_weight = BlockWeights::builder()
.base_block(Weight::zero())
.for_class(DispatchClass::non_mandatory(), |w| {
w.base_extrinsic = Weight::zero();
w.max_total = Some(Weight::from_parts(20, u64::MAX));
})
.for_class(DispatchClass::Mandatory, |w| {
w.base_extrinsic = Weight::zero();
w.reserved = Some(Weight::from_parts(5, u64::MAX));
w.max_total = None;
})
.build_or_panic();
let all_weight = crate::ConsumedWeight::new(|class| match class {
DispatchClass::Normal => Weight::from_parts(10, 0),
DispatchClass::Operational => Weight::from_parts(10, 0),
DispatchClass::Mandatory => Weight::zero(),
});
assert_eq!(maximum_weight.max_block, all_weight.total().set_proof_size(u64::MAX));
let mandatory1 = DispatchInfo {
call_weight: Weight::from_parts(5, 0),
class: DispatchClass::Mandatory,
..Default::default()
};
let mandatory2 = DispatchInfo {
call_weight: Weight::from_parts(6, 0),
class: DispatchClass::Mandatory,
..Default::default()
};
assert_ok!(calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&mandatory1,
0
));
assert_err!(
calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight,
&mandatory2,
0
),
InvalidTransaction::ExhaustsResources
);
}
#[test]
fn proof_size_includes_length() {
let maximum_weight = BlockWeights::builder()
.base_block(Weight::zero())
.for_class(DispatchClass::non_mandatory(), |w| {
w.base_extrinsic = Weight::zero();
w.max_total = Some(Weight::from_parts(20, 1000));
})
.for_class(DispatchClass::Mandatory, |w| {
w.base_extrinsic = Weight::zero();
w.max_total = Some(Weight::from_parts(20, 1000));
})
.build_or_panic();
let all_weight = crate::ConsumedWeight::new(|class| match class {
DispatchClass::Normal => Weight::from_parts(5, 0),
DispatchClass::Operational => Weight::from_parts(5, 0),
DispatchClass::Mandatory => Weight::from_parts(0, 0),
});
let normal = DispatchInfo {
call_weight: Weight::from_parts(5, 0),
class: DispatchClass::Normal,
..Default::default()
};
let mandatory = DispatchInfo {
call_weight: Weight::from_parts(5, 0),
class: DispatchClass::Mandatory,
..Default::default()
};
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&normal,
0,
)
.unwrap();
assert_eq!(consumed.total().saturating_sub(all_weight.total()), normal.total_weight());
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&mandatory,
0,
)
.unwrap();
assert_eq!(consumed.total().saturating_sub(all_weight.total()), mandatory.total_weight());
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&normal,
100,
)
.unwrap();
assert_eq!(
consumed.total().saturating_sub(all_weight.total()),
normal.total_weight().add_proof_size(100)
);
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&mandatory,
100,
)
.unwrap();
assert_eq!(
consumed.total().saturating_sub(all_weight.total()),
mandatory.total_weight().add_proof_size(100)
);
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&normal,
2000,
);
assert_eq!(consumed, Err(InvalidTransaction::ExhaustsResources.into()));
let consumed = calculate_consumed_weight::<<Test as Config>::RuntimeCall>(
&maximum_weight,
all_weight.clone(),
&mandatory,
2000,
);
assert_eq!(consumed, Err(InvalidTransaction::ExhaustsResources.into()));
}
}