mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 15:47:58 +00:00
109287f4ca
* Put `GetWeight` where it belongs * add GetWeight to v2 * Re-export unchanged trait --------- Co-authored-by: Just van Stam <just.van.stam@gmail.com> Co-authored-by: Keith Yeung <kungfukeith11@gmail.com>
260 lines
8.9 KiB
Rust
260 lines
8.9 KiB
Rust
// Copyright (C) Parity Technologies (UK) Ltd.
|
|
// This file is part of Polkadot.
|
|
|
|
// Polkadot is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
|
|
// Polkadot is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with Polkadot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
use frame_support::{
|
|
dispatch::GetDispatchInfo,
|
|
traits::{tokens::currency::Currency as CurrencyT, Get, OnUnbalanced as OnUnbalancedT},
|
|
weights::{
|
|
constants::{WEIGHT_PROOF_SIZE_PER_MB, WEIGHT_REF_TIME_PER_SECOND},
|
|
WeightToFee as WeightToFeeT,
|
|
},
|
|
};
|
|
use parity_scale_codec::Decode;
|
|
use sp_runtime::traits::{SaturatedConversion, Saturating, Zero};
|
|
use sp_std::{marker::PhantomData, result::Result};
|
|
use xcm::latest::{prelude::*, Weight};
|
|
use xcm_executor::{
|
|
traits::{WeightBounds, WeightTrader},
|
|
Assets,
|
|
};
|
|
|
|
pub struct FixedWeightBounds<T, C, M>(PhantomData<(T, C, M)>);
|
|
impl<T: Get<Weight>, C: Decode + GetDispatchInfo, M: Get<u32>> WeightBounds<C>
|
|
for FixedWeightBounds<T, C, M>
|
|
{
|
|
fn weight(message: &mut Xcm<C>) -> Result<Weight, ()> {
|
|
log::trace!(target: "xcm::weight", "FixedWeightBounds message: {:?}", message);
|
|
let mut instructions_left = M::get();
|
|
Self::weight_with_limit(message, &mut instructions_left)
|
|
}
|
|
fn instr_weight(instruction: &Instruction<C>) -> Result<Weight, ()> {
|
|
Self::instr_weight_with_limit(instruction, &mut u32::max_value())
|
|
}
|
|
}
|
|
|
|
impl<T: Get<Weight>, C: Decode + GetDispatchInfo, M> FixedWeightBounds<T, C, M> {
|
|
fn weight_with_limit(message: &Xcm<C>, instrs_limit: &mut u32) -> Result<Weight, ()> {
|
|
let mut r: Weight = Weight::zero();
|
|
*instrs_limit = instrs_limit.checked_sub(message.0.len() as u32).ok_or(())?;
|
|
for m in message.0.iter() {
|
|
r = r.checked_add(&Self::instr_weight_with_limit(m, instrs_limit)?).ok_or(())?;
|
|
}
|
|
Ok(r)
|
|
}
|
|
fn instr_weight_with_limit(
|
|
instruction: &Instruction<C>,
|
|
instrs_limit: &mut u32,
|
|
) -> Result<Weight, ()> {
|
|
let instr_weight = match instruction {
|
|
Transact { require_weight_at_most, .. } => *require_weight_at_most,
|
|
SetErrorHandler(xcm) | SetAppendix(xcm) => Self::weight_with_limit(xcm, instrs_limit)?,
|
|
_ => Weight::zero(),
|
|
};
|
|
T::get().checked_add(&instr_weight).ok_or(())
|
|
}
|
|
}
|
|
|
|
pub struct WeightInfoBounds<W, C, M>(PhantomData<(W, C, M)>);
|
|
impl<W, C, M> WeightBounds<C> for WeightInfoBounds<W, C, M>
|
|
where
|
|
W: XcmWeightInfo<C>,
|
|
C: Decode + GetDispatchInfo,
|
|
M: Get<u32>,
|
|
Instruction<C>: xcm::latest::GetWeight<W>,
|
|
{
|
|
fn weight(message: &mut Xcm<C>) -> Result<Weight, ()> {
|
|
log::trace!(target: "xcm::weight", "WeightInfoBounds message: {:?}", message);
|
|
let mut instructions_left = M::get();
|
|
Self::weight_with_limit(message, &mut instructions_left)
|
|
}
|
|
fn instr_weight(instruction: &Instruction<C>) -> Result<Weight, ()> {
|
|
Self::instr_weight_with_limit(instruction, &mut u32::max_value())
|
|
}
|
|
}
|
|
|
|
impl<W, C, M> WeightInfoBounds<W, C, M>
|
|
where
|
|
W: XcmWeightInfo<C>,
|
|
C: Decode + GetDispatchInfo,
|
|
M: Get<u32>,
|
|
Instruction<C>: xcm::latest::GetWeight<W>,
|
|
{
|
|
fn weight_with_limit(message: &Xcm<C>, instrs_limit: &mut u32) -> Result<Weight, ()> {
|
|
let mut r: Weight = Weight::zero();
|
|
*instrs_limit = instrs_limit.checked_sub(message.0.len() as u32).ok_or(())?;
|
|
for m in message.0.iter() {
|
|
r = r.checked_add(&Self::instr_weight_with_limit(m, instrs_limit)?).ok_or(())?;
|
|
}
|
|
Ok(r)
|
|
}
|
|
fn instr_weight_with_limit(
|
|
instruction: &Instruction<C>,
|
|
instrs_limit: &mut u32,
|
|
) -> Result<Weight, ()> {
|
|
let instr_weight = match instruction {
|
|
Transact { require_weight_at_most, .. } => *require_weight_at_most,
|
|
SetErrorHandler(xcm) | SetAppendix(xcm) => Self::weight_with_limit(xcm, instrs_limit)?,
|
|
_ => Weight::zero(),
|
|
};
|
|
instruction.weight().checked_add(&instr_weight).ok_or(())
|
|
}
|
|
}
|
|
|
|
/// Function trait for handling some revenue. Similar to a negative imbalance (credit) handler, but
|
|
/// for a `MultiAsset`. Sensible implementations will deposit the asset in some known treasury or
|
|
/// block-author account.
|
|
pub trait TakeRevenue {
|
|
/// Do something with the given `revenue`, which is a single non-wildcard `MultiAsset`.
|
|
fn take_revenue(revenue: MultiAsset);
|
|
}
|
|
|
|
/// Null implementation just burns the revenue.
|
|
impl TakeRevenue for () {
|
|
fn take_revenue(_revenue: MultiAsset) {}
|
|
}
|
|
|
|
/// Simple fee calculator that requires payment in a single fungible at a fixed rate.
|
|
///
|
|
/// The constant `Get` type parameter should be the fungible ID, the amount of it required for one
|
|
/// second of weight and the amount required for 1 MB of proof.
|
|
pub struct FixedRateOfFungible<T: Get<(AssetId, u128, u128)>, R: TakeRevenue>(
|
|
Weight,
|
|
u128,
|
|
PhantomData<(T, R)>,
|
|
);
|
|
impl<T: Get<(AssetId, u128, u128)>, R: TakeRevenue> WeightTrader for FixedRateOfFungible<T, R> {
|
|
fn new() -> Self {
|
|
Self(Weight::zero(), 0, PhantomData)
|
|
}
|
|
|
|
fn buy_weight(
|
|
&mut self,
|
|
weight: Weight,
|
|
payment: Assets,
|
|
context: &XcmContext,
|
|
) -> Result<Assets, XcmError> {
|
|
log::trace!(
|
|
target: "xcm::weight",
|
|
"FixedRateOfFungible::buy_weight weight: {:?}, payment: {:?}, context: {:?}",
|
|
weight, payment, context,
|
|
);
|
|
let (id, units_per_second, units_per_mb) = T::get();
|
|
let amount = (units_per_second * (weight.ref_time() as u128) /
|
|
(WEIGHT_REF_TIME_PER_SECOND as u128)) +
|
|
(units_per_mb * (weight.proof_size() as u128) / (WEIGHT_PROOF_SIZE_PER_MB as u128));
|
|
if amount == 0 {
|
|
return Ok(payment)
|
|
}
|
|
let unused =
|
|
payment.checked_sub((id, amount).into()).map_err(|_| XcmError::TooExpensive)?;
|
|
self.0 = self.0.saturating_add(weight);
|
|
self.1 = self.1.saturating_add(amount);
|
|
Ok(unused)
|
|
}
|
|
|
|
fn refund_weight(&mut self, weight: Weight, context: &XcmContext) -> Option<MultiAsset> {
|
|
log::trace!(target: "xcm::weight", "FixedRateOfFungible::refund_weight weight: {:?}, context: {:?}", weight, context);
|
|
let (id, units_per_second, units_per_mb) = T::get();
|
|
let weight = weight.min(self.0);
|
|
let amount = (units_per_second * (weight.ref_time() as u128) /
|
|
(WEIGHT_REF_TIME_PER_SECOND as u128)) +
|
|
(units_per_mb * (weight.proof_size() as u128) / (WEIGHT_PROOF_SIZE_PER_MB as u128));
|
|
self.0 -= weight;
|
|
self.1 = self.1.saturating_sub(amount);
|
|
if amount > 0 {
|
|
Some((id, amount).into())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<T: Get<(AssetId, u128, u128)>, R: TakeRevenue> Drop for FixedRateOfFungible<T, R> {
|
|
fn drop(&mut self) {
|
|
if self.1 > 0 {
|
|
R::take_revenue((T::get().0, self.1).into());
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Weight trader which uses the configured `WeightToFee` to set the right price for weight and then
|
|
/// places any weight bought into the right account.
|
|
pub struct UsingComponents<
|
|
WeightToFee: WeightToFeeT<Balance = Currency::Balance>,
|
|
AssetId: Get<MultiLocation>,
|
|
AccountId,
|
|
Currency: CurrencyT<AccountId>,
|
|
OnUnbalanced: OnUnbalancedT<Currency::NegativeImbalance>,
|
|
>(
|
|
Weight,
|
|
Currency::Balance,
|
|
PhantomData<(WeightToFee, AssetId, AccountId, Currency, OnUnbalanced)>,
|
|
);
|
|
impl<
|
|
WeightToFee: WeightToFeeT<Balance = Currency::Balance>,
|
|
AssetId: Get<MultiLocation>,
|
|
AccountId,
|
|
Currency: CurrencyT<AccountId>,
|
|
OnUnbalanced: OnUnbalancedT<Currency::NegativeImbalance>,
|
|
> WeightTrader for UsingComponents<WeightToFee, AssetId, AccountId, Currency, OnUnbalanced>
|
|
{
|
|
fn new() -> Self {
|
|
Self(Weight::zero(), Zero::zero(), PhantomData)
|
|
}
|
|
|
|
fn buy_weight(
|
|
&mut self,
|
|
weight: Weight,
|
|
payment: Assets,
|
|
context: &XcmContext,
|
|
) -> Result<Assets, XcmError> {
|
|
log::trace!(target: "xcm::weight", "UsingComponents::buy_weight weight: {:?}, payment: {:?}, context: {:?}", weight, payment, context);
|
|
let amount = WeightToFee::weight_to_fee(&weight);
|
|
let u128_amount: u128 = amount.try_into().map_err(|_| XcmError::Overflow)?;
|
|
let required = (Concrete(AssetId::get()), u128_amount).into();
|
|
let unused = payment.checked_sub(required).map_err(|_| XcmError::TooExpensive)?;
|
|
self.0 = self.0.saturating_add(weight);
|
|
self.1 = self.1.saturating_add(amount);
|
|
Ok(unused)
|
|
}
|
|
|
|
fn refund_weight(&mut self, weight: Weight, context: &XcmContext) -> Option<MultiAsset> {
|
|
log::trace!(target: "xcm::weight", "UsingComponents::refund_weight weight: {:?}, context: {:?}", weight, context);
|
|
let weight = weight.min(self.0);
|
|
let amount = WeightToFee::weight_to_fee(&weight);
|
|
self.0 -= weight;
|
|
self.1 = self.1.saturating_sub(amount);
|
|
let amount: u128 = amount.saturated_into();
|
|
if amount > 0 {
|
|
Some((AssetId::get(), amount).into())
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
}
|
|
impl<
|
|
WeightToFee: WeightToFeeT<Balance = Currency::Balance>,
|
|
AssetId: Get<MultiLocation>,
|
|
AccountId,
|
|
Currency: CurrencyT<AccountId>,
|
|
OnUnbalanced: OnUnbalancedT<Currency::NegativeImbalance>,
|
|
> Drop for UsingComponents<WeightToFee, AssetId, AccountId, Currency, OnUnbalanced>
|
|
{
|
|
fn drop(&mut self) {
|
|
OnUnbalanced::on_unbalanced(Currency::issue(self.1));
|
|
}
|
|
}
|