mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-30 15:11:02 +00:00
efe290490a
* Refactor ParentIsDefault to ParentIsAllZeroes * Remove Default bound on all AccountId types under the xcm directory * Change to ParentIs<A: Get<AccountId>, AccountId> * Provide a better account for ParentIs * Fixes * Fixes * Fixes * Fixes * Update xcm/xcm-builder/src/currency_adapter.rs Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com> * Use preset account ID value for parent MultiLocations Co-authored-by: Bastian Köcher <bkchr@users.noreply.github.com>
176 lines
6.6 KiB
Rust
176 lines
6.6 KiB
Rust
// Copyright 2020 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/>.
|
|
|
|
//! Adapters to work with `frame_support::traits::Currency` through XCM.
|
|
|
|
use frame_support::traits::{ExistenceRequirement::AllowDeath, Get, WithdrawReasons};
|
|
use sp_runtime::traits::{CheckedSub, SaturatedConversion};
|
|
use sp_std::{convert::TryInto, marker::PhantomData, result};
|
|
use xcm::latest::{Error as XcmError, MultiAsset, MultiLocation, Result};
|
|
use xcm_executor::{
|
|
traits::{Convert, MatchesFungible, TransactAsset},
|
|
Assets,
|
|
};
|
|
|
|
/// Asset transaction errors.
|
|
enum Error {
|
|
/// Asset not found.
|
|
AssetNotFound,
|
|
/// `MultiLocation` to `AccountId` conversion failed.
|
|
AccountIdConversionFailed,
|
|
/// `u128` amount to currency `Balance` conversion failed.
|
|
AmountToBalanceConversionFailed,
|
|
}
|
|
|
|
impl From<Error> for XcmError {
|
|
fn from(e: Error) -> Self {
|
|
use XcmError::FailedToTransactAsset;
|
|
match e {
|
|
Error::AssetNotFound => XcmError::AssetNotFound,
|
|
Error::AccountIdConversionFailed => FailedToTransactAsset("AccountIdConversionFailed"),
|
|
Error::AmountToBalanceConversionFailed =>
|
|
FailedToTransactAsset("AmountToBalanceConversionFailed"),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Simple adapter to use a currency as asset transactor. This type can be used as `type AssetTransactor` in
|
|
/// `xcm::Config`.
|
|
///
|
|
/// # Example
|
|
/// ```
|
|
/// use parity_scale_codec::Decode;
|
|
/// use frame_support::{parameter_types, PalletId};
|
|
/// use sp_runtime::traits::{AccountIdConversion, TrailingZeroInput};
|
|
/// use xcm::latest::prelude::*;
|
|
/// use xcm_builder::{ParentIsPreset, CurrencyAdapter, IsConcrete};
|
|
///
|
|
/// /// Our chain's account id.
|
|
/// type AccountId = sp_runtime::AccountId32;
|
|
///
|
|
/// /// Our relay chain's location.
|
|
/// parameter_types! {
|
|
/// pub RelayChain: MultiLocation = Parent.into();
|
|
/// pub CheckingAccount: AccountId = PalletId(*b"checking").into_account();
|
|
/// }
|
|
///
|
|
/// /// Some items that implement `Convert<MultiLocation, AccountId>`. Can be more, but for now we just assume we accept
|
|
/// /// messages from the parent (relay chain).
|
|
/// pub type LocationConverter = (ParentIsPreset<AccountId>);
|
|
///
|
|
/// /// Final currency adapter. This can be used in `xcm::Config` to specify how asset related transactions happen.
|
|
/// pub type AssetTransactor = CurrencyAdapter<
|
|
/// // Use this balance type:
|
|
/// u128,
|
|
/// // The matcher: use the currency when the asset is a concrete asset in our relay chain.
|
|
/// IsConcrete<RelayChain>,
|
|
/// // The local converter: default account of the parent relay chain.
|
|
/// LocationConverter,
|
|
/// // Our chain's account ID type.
|
|
/// AccountId,
|
|
/// // The checking account. Can be any deterministic inaccessible account.
|
|
/// CheckingAccount,
|
|
/// >;
|
|
/// ```
|
|
pub struct CurrencyAdapter<Currency, Matcher, AccountIdConverter, AccountId, CheckedAccount>(
|
|
PhantomData<(Currency, Matcher, AccountIdConverter, AccountId, CheckedAccount)>,
|
|
);
|
|
|
|
impl<
|
|
Matcher: MatchesFungible<Currency::Balance>,
|
|
AccountIdConverter: Convert<MultiLocation, AccountId>,
|
|
Currency: frame_support::traits::Currency<AccountId>,
|
|
AccountId: Clone, // can't get away without it since Currency is generic over it.
|
|
CheckedAccount: Get<Option<AccountId>>,
|
|
> TransactAsset
|
|
for CurrencyAdapter<Currency, Matcher, AccountIdConverter, AccountId, CheckedAccount>
|
|
{
|
|
fn can_check_in(_origin: &MultiLocation, what: &MultiAsset) -> Result {
|
|
log::trace!(target: "xcm::currency_adapter", "can_check_in origin: {:?}, what: {:?}", _origin, what);
|
|
// Check we handle this asset.
|
|
let amount: Currency::Balance =
|
|
Matcher::matches_fungible(what).ok_or(Error::AssetNotFound)?;
|
|
if let Some(checked_account) = CheckedAccount::get() {
|
|
let new_balance = Currency::free_balance(&checked_account)
|
|
.checked_sub(&amount)
|
|
.ok_or(XcmError::NotWithdrawable)?;
|
|
Currency::ensure_can_withdraw(
|
|
&checked_account,
|
|
amount,
|
|
WithdrawReasons::TRANSFER,
|
|
new_balance,
|
|
)
|
|
.map_err(|_| XcmError::NotWithdrawable)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
|
|
fn check_in(_origin: &MultiLocation, what: &MultiAsset) {
|
|
log::trace!(target: "xcm::currency_adapter", "check_in origin: {:?}, what: {:?}", _origin, what);
|
|
if let Some(amount) = Matcher::matches_fungible(what) {
|
|
if let Some(checked_account) = CheckedAccount::get() {
|
|
let ok = Currency::withdraw(
|
|
&checked_account,
|
|
amount,
|
|
WithdrawReasons::TRANSFER,
|
|
AllowDeath,
|
|
)
|
|
.is_ok();
|
|
debug_assert!(
|
|
ok,
|
|
"`can_check_in` must have returned `true` immediately prior; qed"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn check_out(_dest: &MultiLocation, what: &MultiAsset) {
|
|
log::trace!(target: "xcm::currency_adapter", "check_out dest: {:?}, what: {:?}", _dest, what);
|
|
if let Some(amount) = Matcher::matches_fungible(what) {
|
|
if let Some(checked_account) = CheckedAccount::get() {
|
|
Currency::deposit_creating(&checked_account, amount);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn deposit_asset(what: &MultiAsset, who: &MultiLocation) -> Result {
|
|
log::trace!(target: "xcm::currency_adapter", "deposit_asset what: {:?}, who: {:?}", what, who);
|
|
// Check we handle this asset.
|
|
let amount: u128 =
|
|
Matcher::matches_fungible(&what).ok_or(Error::AssetNotFound)?.saturated_into();
|
|
let who =
|
|
AccountIdConverter::convert_ref(who).map_err(|()| Error::AccountIdConversionFailed)?;
|
|
let balance_amount =
|
|
amount.try_into().map_err(|_| Error::AmountToBalanceConversionFailed)?;
|
|
let _imbalance = Currency::deposit_creating(&who, balance_amount);
|
|
Ok(())
|
|
}
|
|
|
|
fn withdraw_asset(what: &MultiAsset, who: &MultiLocation) -> result::Result<Assets, XcmError> {
|
|
log::trace!(target: "xcm::currency_adapter", "withdraw_asset what: {:?}, who: {:?}", what, who);
|
|
// Check we handle this asset.
|
|
let amount: u128 =
|
|
Matcher::matches_fungible(what).ok_or(Error::AssetNotFound)?.saturated_into();
|
|
let who =
|
|
AccountIdConverter::convert_ref(who).map_err(|()| Error::AccountIdConversionFailed)?;
|
|
let balance_amount =
|
|
amount.try_into().map_err(|_| Error::AmountToBalanceConversionFailed)?;
|
|
Currency::withdraw(&who, balance_amount, WithdrawReasons::TRANSFER, AllowDeath)
|
|
.map_err(|e| XcmError::FailedToTransactAsset(e.into()))?;
|
|
Ok(what.clone().into())
|
|
}
|
|
}
|