Skip to content
This repository has been archived by the owner on Nov 15, 2023. It is now read-only.

Asset Transaction Payment #488

Merged
merged 75 commits into from
Oct 11, 2021
Merged
Changes from 1 commit
Commits
Show all changes
75 commits
Select commit Hold shift + click to select a range
bee1ce0
use new pallet name based genesis config names
apopiak Jun 10, 2021
c0d13a9
use custom substrate and update polkadot
apopiak Jun 10, 2021
c9266c5
add initial asset-tx-payment pallet
apopiak Jun 10, 2021
7112a86
update cargo.toml
apopiak Jun 10, 2021
9297f6e
add (failing) tests
apopiak Jun 10, 2021
7dd9bea
dispatch Calls instead of using Pallet functions
apopiak Jun 10, 2021
63af3bb
fix fee-refund split
apopiak Jun 10, 2021
f8f838a
add test for transaction payment with tip
apopiak Jun 10, 2021
b31dc8c
update cargo.lock
apopiak Jun 10, 2021
642f8c9
update cargo.lock
apopiak Jun 11, 2021
27c65db
remove mint workaround and use Mutable trait
apopiak Jun 11, 2021
78f1b90
extract fee charging logic into OnChargeAssetTransaction trait
apopiak Jun 14, 2021
3097861
use asset-tx-payment in statemint runtime
apopiak Jun 14, 2021
b9b394e
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Jun 14, 2021
2c10e02
make extrinsics public
apopiak Jun 16, 2021
5f18e5b
make extrinsics public
apopiak Jun 16, 2021
7f36a58
use ChargeAssetIdOf type alias
apopiak Jun 16, 2021
0f29540
update deps
apopiak Jun 16, 2021
4332761
move back to AssetIdOf
apopiak Jun 16, 2021
c77215e
Merge remote-tracking branch 'origin/master' into apopiak/asset-tx-pa…
apopiak Jun 17, 2021
eb9ac14
remove extra rpc_http_threads
apopiak Jun 17, 2021
39e08eb
use different substrate branch
apopiak Jun 17, 2021
1138b90
Update pallets/asset-tx-payment/src/payment.rs
apopiak Jun 25, 2021
34a350d
Update pallets/asset-tx-payment/src/payment.rs
apopiak Jun 25, 2021
b657f59
remove overrides
apopiak Jul 8, 2021
65013c5
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Jul 8, 2021
22816c2
override substrate deps (again)
apopiak Jul 8, 2021
5c30e05
increment spec_version and transaction_version (because we change tra…
apopiak Jul 8, 2021
91afb68
Merge branch 'apopiak/asset-tx-payment' of github.com:paritytech/cumu…
apopiak Jul 8, 2021
0d4acc0
remove direct dependency on pallet-balances from asset-tx-payment
apopiak Jul 8, 2021
a5ecd03
remove Assets pallet visibility workaround
apopiak Jul 8, 2021
530d250
add docs and comments
apopiak Jul 8, 2021
a260e66
remove unused imports
apopiak Jul 8, 2021
2c8ad19
more docs
apopiak Jul 8, 2021
be52a66
add more debug asserts to document assumptions
apopiak Jul 9, 2021
850f999
add test for tx payment from account with only assets
apopiak Jul 9, 2021
1008e67
add test for missing asset case
apopiak Jul 9, 2021
95d4f42
extend test to cover non-sufficient assets
apopiak Jul 9, 2021
4e56a2c
add a test for Pays::No (refunded transaction)
apopiak Jul 9, 2021
86f1d94
add type alias comments
apopiak Jul 9, 2021
5e890f5
add more doc comments
apopiak Jul 9, 2021
1600514
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Sep 21, 2021
d91263b
add asset-tx-payment to statemine and westmint
apopiak Sep 21, 2021
9b7bb39
improve formatting
apopiak Sep 27, 2021
8c5fc82
update license headers
apopiak Sep 27, 2021
8db45aa
add default implementation of HandleCredit for ()
apopiak Sep 27, 2021
8c07bcd
update doc comments and format imports
apopiak Sep 27, 2021
d64f202
adjust Cargo.toml
apopiak Sep 27, 2021
1e650d1
update cargo.lock
apopiak Sep 27, 2021
749cdcf
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Sep 27, 2021
10c3ddf
cargo fmt
apopiak Sep 27, 2021
03f4e12
cargo fmt
apopiak Sep 27, 2021
24dc914
cargo fmt
apopiak Sep 27, 2021
b43da32
cargo +nightly fmt
apopiak Sep 27, 2021
a763672
add type alias for OnChargeTransaction
apopiak Sep 28, 2021
35e4098
cargo +nightly fmt
apopiak Sep 28, 2021
c2fefd5
convert ChargeAssetTxPayment from tuple struct to regular struct
apopiak Sep 28, 2021
d77ff7b
add more comments
apopiak Sep 28, 2021
58c6b7a
formatting
apopiak Sep 28, 2021
6474c34
adjust imports and comment
apopiak Sep 28, 2021
b9382e9
cargo +nightly fmt
apopiak Sep 28, 2021
8638ed8
reformat comment
apopiak Oct 4, 2021
a1d770a
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Oct 5, 2021
7bcac89
use ChargeTransactionPayment's own get_priority + update Substrate
apopiak Oct 5, 2021
acea23e
update Substrate and Polkadot
apopiak Oct 5, 2021
3e7b9c5
Merge branch 'apopiak/transaction-payment-update' into apopiak/asset-…
apopiak Oct 5, 2021
359ee56
cargo fmt
apopiak Oct 5, 2021
1c32551
cargo fmt
apopiak Oct 5, 2021
382120c
add OperationalFeeMultiplier to asset tx payment tests
apopiak Oct 5, 2021
96fd61d
Apply suggestions from code review
apopiak Oct 7, 2021
99349b9
add doc links
apopiak Oct 7, 2021
aef723e
charge a minimum converted asset fee of 1 if the input fee is greater…
apopiak Oct 8, 2021
8fd99f7
cargo +nightly fmt
apopiak Oct 8, 2021
35d13e4
Merge branch 'master' of github.com:paritytech/cumulus into apopiak/a…
apopiak Oct 11, 2021
a413579
bump spec and transaction version
apopiak Oct 11, 2021
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
cargo fmt
  • Loading branch information
apopiak committed Sep 27, 2021
commit 10c3ddfc4f74cdf877569173515e3eeb8cf59114
86 changes: 50 additions & 36 deletions pallets/asset-tx-payment/src/lib.rs
Original file line number Diff line number Diff line change
@@ -59,7 +59,6 @@ use sp_runtime::{
FixedPointOperand,
};


#[cfg(test)]
mod tests;

@@ -135,10 +134,14 @@ pub mod pallet {
/// An asset id of `None` falls back to the underlying transaction payment via the native currency.
#[derive(Encode, Decode, Clone, Eq, PartialEq, TypeInfo)]
#[scale_info(skip_type_params(T))]
pub struct ChargeAssetTxPayment<T: Config>(#[codec(compact)] BalanceOf<T>, Option<ChargeAssetIdOf<T>>);

impl<T: Config> ChargeAssetTxPayment<T> where
T::Call: Dispatchable<Info=DispatchInfo, PostInfo=PostDispatchInfo>,
pub struct ChargeAssetTxPayment<T: Config>(
#[codec(compact)] BalanceOf<T>,
Option<ChargeAssetIdOf<T>>,
);

impl<T: Config> ChargeAssetTxPayment<T>
where
T::Call: Dispatchable<Info = DispatchInfo, PostInfo = PostDispatchInfo>,
AssetBalanceOf<T>: Send + Sync + FixedPointOperand,
BalanceOf<T>: Send + Sync + FixedPointOperand + IsType<ChargeAssetBalanceOf<T>>,
ChargeAssetIdOf<T>: Send + Sync,
@@ -156,13 +159,7 @@ impl<T: Config> ChargeAssetTxPayment<T> where
call: &T::Call,
info: &DispatchInfoOf<T::Call>,
len: usize,
) -> Result<
(
BalanceOf<T>,
InitialPayment<T>,
),
TransactionValidityError,
> {
) -> Result<(BalanceOf<T>, InitialPayment<T>), TransactionValidityError> {
let tip = self.0;
let fee = pallet_transaction_payment::Pallet::<T>::compute_fee(len as u32, info, tip);

@@ -172,10 +169,17 @@ impl<T: Config> ChargeAssetTxPayment<T> where
return Ok((fee, InitialPayment::Nothing));
apopiak marked this conversation as resolved.
Show resolved Hide resolved
}

let maybe_asset_id = self.1;
let maybe_asset_id = self.1;
if let Some(asset_id) = maybe_asset_id {
T::OnChargeAssetTransaction::withdraw_fee(who, call, info, asset_id, fee.into(), tip.into())
.map(|i| (fee, InitialPayment::Asset(i.into())))
T::OnChargeAssetTransaction::withdraw_fee(
who,
call,
info,
asset_id,
fee.into(),
tip.into(),
)
.map(|i| (fee, InitialPayment::Asset(i.into())))
} else {
<<T as pallet_transaction_payment::Config>::OnChargeTransaction as OnChargeTransaction<T>>::withdraw_fee(who, call, info, fee, tip)
.map(|i| (fee, InitialPayment::Native(i)))
@@ -194,17 +198,21 @@ impl<T: Config> ChargeAssetTxPayment<T> where
/// that the transaction which consumes more resources (either length or weight) with the same
/// `fee` ends up having lower priority.
// NOTE: copied from `pallet_transaction_payment`
fn get_priority(len: usize, info: &DispatchInfoOf<T::Call>, final_fee: BalanceOf<T>) -> TransactionPriority {
fn get_priority(
len: usize,
info: &DispatchInfoOf<T::Call>,
final_fee: BalanceOf<T>,
) -> TransactionPriority {
let weight_saturation = T::BlockWeights::get().max_block / info.weight.max(1);
let max_block_length = *T::BlockLength::get().max.get(DispatchClass::Normal);
apopiak marked this conversation as resolved.
Show resolved Hide resolved
let len_saturation = max_block_length as u64 / (len as u64).max(1);
let coefficient: BalanceOf<T> = weight_saturation.min(len_saturation).saturated_into::<BalanceOf<T>>();
let coefficient: BalanceOf<T> =
weight_saturation.min(len_saturation).saturated_into::<BalanceOf<T>>();
final_fee.saturating_mul(coefficient).saturated_into::<TransactionPriority>()
apopiak marked this conversation as resolved.
Show resolved Hide resolved
}
}

impl<T: Config> sp_std::fmt::Debug for ChargeAssetTxPayment<T>
{
impl<T: Config> sp_std::fmt::Debug for ChargeAssetTxPayment<T> {
#[cfg(feature = "std")]
fn fmt(&self, f: &mut sp_std::fmt::Formatter) -> sp_std::fmt::Result {
write!(f, "ChargeAssetTxPayment<{:?}, {:?}>", self.0, self.1.encode())
@@ -215,8 +223,9 @@ impl<T: Config> sp_std::fmt::Debug for ChargeAssetTxPayment<T>
}
}

impl<T: Config> SignedExtension for ChargeAssetTxPayment<T> where
T::Call: Dispatchable<Info=DispatchInfo, PostInfo=PostDispatchInfo>,
impl<T: Config> SignedExtension for ChargeAssetTxPayment<T>
where
T::Call: Dispatchable<Info = DispatchInfo, PostInfo = PostDispatchInfo>,
AssetBalanceOf<T>: Send + Sync + FixedPointOperand,
BalanceOf<T>: Send + Sync + From<u64> + FixedPointOperand + IsType<ChargeAssetBalanceOf<T>>,
ChargeAssetIdOf<T>: Send + Sync,
@@ -234,7 +243,9 @@ impl<T: Config> SignedExtension for ChargeAssetTxPayment<T> where
// imbalance resulting from withdrawing the fee
InitialPayment<T>,
);
fn additional_signed(&self) -> sp_std::result::Result<(), TransactionValidityError> { Ok(()) }
fn additional_signed(&self) -> sp_std::result::Result<(), TransactionValidityError> {
Ok(())
}

fn validate(
&self,
@@ -244,18 +255,15 @@ impl<T: Config> SignedExtension for ChargeAssetTxPayment<T> where
len: usize,
) -> TransactionValidity {
let (fee, _) = self.withdraw_fee(who, call, info, len)?;
Ok(ValidTransaction {
priority: Self::get_priority(len, info, fee),
..Default::default()
})
Ok(ValidTransaction { priority: Self::get_priority(len, info, fee), ..Default::default() })
}

fn pre_dispatch(
self,
who: &Self::AccountId,
call: &Self::Call,
info: &DispatchInfoOf<Self::Call>,
len: usize
len: usize,
) -> Result<Self::Pre, TransactionValidityError> {
let (_fee, initial_payment) = self.withdraw_fee(who, call, info, len)?;
Ok((self.0, who.clone(), initial_payment))
@@ -270,24 +278,30 @@ impl<T: Config> SignedExtension for ChargeAssetTxPayment<T> where
) -> Result<(), TransactionValidityError> {
let (tip, who, initial_payment) = pre;
let actual_fee = pallet_transaction_payment::Pallet::<T>::compute_actual_fee(
len as u32,
info,
post_info,
tip,
len as u32, info, post_info, tip,
);
match initial_payment {
InitialPayment::Native(already_withdrawn) => {
<<T as pallet_transaction_payment::Config>::OnChargeTransaction as OnChargeTransaction<T>>::correct_and_deposit_fee(
&who, info, post_info, actual_fee, tip, already_withdrawn)?;
},
}
InitialPayment::Asset(already_withdrawn) => {
T::OnChargeAssetTransaction::correct_and_deposit_fee(
&who, info, post_info, actual_fee.into(), tip.into(), already_withdrawn.into())?;
},
&who,
info,
post_info,
actual_fee.into(),
tip.into(),
already_withdrawn.into(),
)?;
}
InitialPayment::Nothing => {
debug_assert!(actual_fee.is_zero(), "actual fee should be zero if initial fee was zero.");
debug_assert!(
actual_fee.is_zero(),
"actual fee should be zero if initial fee was zero."
);
debug_assert!(tip.is_zero(), "tip should be zero if initial fee was zero.");
},
}
}

Ok(())
19 changes: 8 additions & 11 deletions pallets/asset-tx-payment/src/payment.rs
Original file line number Diff line number Diff line change
@@ -14,15 +14,14 @@
// limitations under the License.

///! Traits and default implementation for paying transaction fees in assets.
apopiak marked this conversation as resolved.
Show resolved Hide resolved

use super::*;
use crate::Config;

use codec::FullCodec;
use frame_support::{
traits::fungibles::{Balanced, Inspect, CreditOf},
unsigned::TransactionValidityError,
traits::fungibles::{Balanced, CreditOf, Inspect},
traits::tokens::BalanceConversion,
unsigned::TransactionValidityError,
};
use scale_info::TypeInfo;
use sp_runtime::{
@@ -42,13 +41,7 @@ pub trait OnChargeAssetTransaction<T: Config> {
+ Default
+ TypeInfo;
/// The type used to identify the assets used for transaction payment.
type AssetId: FullCodec
+ Copy
+ MaybeSerializeDeserialize
+ Debug
+ Default
+ Eq
+ TypeInfo;
type AssetId: FullCodec + Copy + MaybeSerializeDeserialize + Debug + Default + Eq + TypeInfo;
/// The type used to store the intermediate values between pre- and post-dispatch.
type LiquidityInfo;

@@ -125,7 +118,11 @@ where
) -> Result<Self::LiquidityInfo, TransactionValidityError> {
let converted_fee = CON::to_asset_balance(fee, asset_id)
apopiak marked this conversation as resolved.
Show resolved Hide resolved
.map_err(|_| TransactionValidityError::from(InvalidTransaction::Payment))?;
let can_withdraw = <T::Fungibles as Inspect<T::AccountId>>::can_withdraw(asset_id.into(), who, converted_fee);
let can_withdraw = <T::Fungibles as Inspect<T::AccountId>>::can_withdraw(
asset_id.into(),
who,
converted_fee,
);
if !matches!(can_withdraw, WithdrawConsequence::Success) {
return Err(InvalidTransaction::Payment.into());
}
Loading