Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add NonFungibleAdapter #2924

Merged
merged 9 commits into from
Jan 26, 2024
Merged
Show file tree
Hide file tree
Changes from 3 commits
Commits
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
5 changes: 5 additions & 0 deletions polkadot/xcm/xcm-builder/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,11 @@ pub use fungibles_adapter::{
LocalMint, MintLocation, NoChecking, NonLocalMint,
};

mod nonfungible_adapter;
pub use nonfungible_adapter::{
NonFungibleAdapter, NonFungibleMutateAdapter, NonFungibleTransferAdapter,
};

mod nonfungibles_adapter;
pub use nonfungibles_adapter::{
NonFungiblesAdapter, NonFungiblesMutateAdapter, NonFungiblesTransferAdapter,
Expand Down
319 changes: 319 additions & 0 deletions polkadot/xcm/xcm-builder/src/nonfungible_adapter.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,319 @@
// 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/>.

//! Adapters to work with [`frame_support::traits::tokens::nonfungible`] through XCM.

use super::MintLocation;
use frame_support::{
ensure,
traits::{tokens::nonfungible, Get},
};
use sp_std::{marker::PhantomData, prelude::*, result};
use xcm::latest::prelude::*;
seadanda marked this conversation as resolved.
Show resolved Hide resolved
use xcm_executor::traits::{
ConvertLocation, Error as MatchError, MatchesNonFungible, TransactAsset,
};

const LOG_TARGET: &str = "xcm::nonfungible_adapter";

pub struct NonFungibleTransferAdapter<Asset, Matcher, AccountIdConverter, AccountId>(
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
PhantomData<(Asset, Matcher, AccountIdConverter, AccountId)>,
);
impl<
Asset: nonfungible::Transfer<AccountId>,
Matcher: MatchesNonFungible<Asset::ItemId>,
AccountIdConverter: ConvertLocation<AccountId>,
AccountId: Clone, // can't get away without it since Currency is generic over it.
> TransactAsset for NonFungibleTransferAdapter<Asset, Matcher, AccountIdConverter, AccountId>
{
fn transfer_asset(
what: &MultiAsset,
from: &MultiLocation,
to: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
context: &XcmContext,
) -> result::Result<xcm_executor::Assets, XcmError> {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"transfer_asset what: {:?}, from: {:?}, to: {:?}, context: {:?}",
what,
from,
to,
context,
);
// Check we handle this asset.
let instance = Matcher::matches_nonfungible(what).ok_or(MatchError::AssetNotHandled)?;
let destination = AccountIdConverter::convert_location(to)
.ok_or(MatchError::AccountIdConversionFailed)?;
Asset::transfer(&instance, &destination)
.map_err(|e| XcmError::FailedToTransactAsset(e.into()))?;
Ok(what.clone().into())
}
}

pub struct NonFungibleMutateAdapter<Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount>(
PhantomData<(Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount)>,
);

impl<
Asset: nonfungible::Mutate<AccountId>,
Matcher: MatchesNonFungible<Asset::ItemId>,
AccountIdConverter: ConvertLocation<AccountId>,
AccountId: Clone + Eq, // can't get away without it since Currency is generic over it.
CheckingAccount: Get<Option<(AccountId, MintLocation)>>,
> NonFungibleMutateAdapter<Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount>
{
fn can_accrue_checked(instance: Asset::ItemId) -> XcmResult {
ensure!(Asset::owner(&instance).is_none(), XcmError::NotDepositable);
Ok(())
}
fn can_reduce_checked(checking_account: AccountId, instance: Asset::ItemId) -> XcmResult {
// This is an asset whose teleports we track.
let owner = Asset::owner(&instance);
ensure!(owner == Some(checking_account), XcmError::NotWithdrawable);
ensure!(Asset::can_transfer(&instance), XcmError::NotWithdrawable);
Ok(())
}
fn accrue_checked(checking_account: AccountId, instance: Asset::ItemId) {
let ok = Asset::mint_into(&instance, &checking_account).is_ok();
debug_assert!(ok, "`mint_into` cannot generally fail; qed");
}
fn reduce_checked(instance: Asset::ItemId) {
let ok = Asset::burn(&instance, None).is_ok();
debug_assert!(ok, "`can_check_in` must have returned `true` immediately prior; qed");
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps these should be proper log messages and not debug_asserts. You are assuming that the user has e.g. called can_check_in prior to calling this function, but there is no guarantee. If the trait doesn't allow you to return an error here, should probably log it.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I used debug_assert here to keep consistent with other adapter implementations.

}

impl<
Asset: nonfungible::Mutate<AccountId>,
Matcher: MatchesNonFungible<Asset::ItemId>,
AccountIdConverter: ConvertLocation<AccountId>,
AccountId: Clone + Eq, // can't get away without it since Currency is generic over it.
CheckingAccount: Get<Option<(AccountId, MintLocation)>>,
> TransactAsset
for NonFungibleMutateAdapter<Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount>
{
fn can_check_in(_origin: &MultiLocation, what: &MultiAsset, context: &XcmContext) -> XcmResult {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"can_check_in origin: {:?}, what: {:?}, context: {:?}",
_origin,
what,
context,
);
// Check we handle this asset.
let instance = Matcher::matches_nonfungible(what).ok_or(MatchError::AssetNotHandled)?;
match CheckingAccount::get() {
// We track this asset's teleports to ensure no more come in than have gone out.
Some((checking_account, MintLocation::Local)) =>
Self::can_reduce_checked(checking_account, instance),
// We track this asset's teleports to ensure no more go out than have come in.
Some((_, MintLocation::NonLocal)) => Self::can_accrue_checked(instance),
_ => Ok(()),
}
}

fn check_in(_origin: &MultiLocation, what: &MultiAsset, context: &XcmContext) {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"check_in origin: {:?}, what: {:?}, context: {:?}",
_origin,
what,
context,
);
if let Some(instance) = Matcher::matches_nonfungible(what) {
match CheckingAccount::get() {
// We track this asset's teleports to ensure no more come in than have gone out.
Some((_, MintLocation::Local)) => Self::reduce_checked(instance),
// We track this asset's teleports to ensure no more go out than have come in.
Some((checking_account, MintLocation::NonLocal)) =>
Self::accrue_checked(checking_account, instance),
_ => (),
}
}
}

fn can_check_out(_dest: &MultiLocation, what: &MultiAsset, context: &XcmContext) -> XcmResult {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"can_check_out dest: {:?}, what: {:?}, context: {:?}",
_dest,
what,
context,
);
// Check we handle this asset.
let instance = Matcher::matches_nonfungible(what).ok_or(MatchError::AssetNotHandled)?;
match CheckingAccount::get() {
// We track this asset's teleports to ensure no more come in than have gone out.
Some((_, MintLocation::Local)) => Self::can_accrue_checked(instance),
// We track this asset's teleports to ensure no more go out than have come in.
Some((checking_account, MintLocation::NonLocal)) =>
Self::can_reduce_checked(checking_account, instance),
_ => Ok(()),
}
}

fn check_out(_dest: &MultiLocation, what: &MultiAsset, context: &XcmContext) {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"check_out dest: {:?}, what: {:?}, context: {:?}",
_dest,
what,
context,
);
if let Some(instance) = Matcher::matches_nonfungible(what) {
match CheckingAccount::get() {
// We track this asset's teleports to ensure no more come in than have gone out.
Some((checking_account, MintLocation::Local)) =>
Self::accrue_checked(checking_account, instance),
// We track this asset's teleports to ensure no more go out than have come in.
Some((_, MintLocation::NonLocal)) => Self::reduce_checked(instance),
_ => (),
}
}
}

fn deposit_asset(
what: &MultiAsset,
who: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
context: Option<&XcmContext>,
) -> XcmResult {
log::trace!(
target: LOG_TARGET,
"deposit_asset what: {:?}, who: {:?}, context: {:?}",
what,
who,
context,
);
// Check we handle this asset.
let instance = Matcher::matches_nonfungible(what).ok_or(MatchError::AssetNotHandled)?;
let who = AccountIdConverter::convert_location(who)
.ok_or(MatchError::AccountIdConversionFailed)?;
Asset::mint_into(&instance, &who).map_err(|e| XcmError::FailedToTransactAsset(e.into()))
}

fn withdraw_asset(
what: &MultiAsset,
who: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
maybe_context: Option<&XcmContext>,
) -> result::Result<xcm_executor::Assets, XcmError> {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
log::trace!(
target: LOG_TARGET,
"withdraw_asset what: {:?}, who: {:?}, maybe_context: {:?}",
what,
who,
maybe_context,
);
// Check we handle this asset.
let who = AccountIdConverter::convert_location(who)
.ok_or(MatchError::AccountIdConversionFailed)?;
let instance = Matcher::matches_nonfungible(what).ok_or(MatchError::AssetNotHandled)?;
Asset::burn(&instance, Some(&who))
.map_err(|e| XcmError::FailedToTransactAsset(e.into()))?;
Ok(what.clone().into())
}
}

pub struct NonFungibleAdapter<Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount>(
PhantomData<(Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount)>,
);
impl<
Asset: nonfungible::Mutate<AccountId> + nonfungible::Transfer<AccountId>,
Matcher: MatchesNonFungible<Asset::ItemId>,
AccountIdConverter: ConvertLocation<AccountId>,
AccountId: Clone + Eq, // can't get away without it since Currency is generic over it.
CheckingAccount: Get<Option<(AccountId, MintLocation)>>,
> TransactAsset
for NonFungibleAdapter<Asset, Matcher, AccountIdConverter, AccountId, CheckingAccount>
{
fn can_check_in(origin: &MultiLocation, what: &MultiAsset, context: &XcmContext) -> XcmResult {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::can_check_in(origin, what, context)
}

fn check_in(origin: &MultiLocation, what: &MultiAsset, context: &XcmContext) {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::check_in(origin, what, context)
}

fn can_check_out(dest: &MultiLocation, what: &MultiAsset, context: &XcmContext) -> XcmResult {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::can_check_out(dest, what, context)
}

fn check_out(dest: &MultiLocation, what: &MultiAsset, context: &XcmContext) {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::check_out(dest, what, context)
}

fn deposit_asset(
what: &MultiAsset,
who: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
context: Option<&XcmContext>,
) -> XcmResult {
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::deposit_asset(what, who, context)
}

fn withdraw_asset(
what: &MultiAsset,
who: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
maybe_context: Option<&XcmContext>,
) -> result::Result<xcm_executor::Assets, XcmError> {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleMutateAdapter::<
Asset,
Matcher,
AccountIdConverter,
AccountId,
CheckingAccount,
>::withdraw_asset(what, who, maybe_context)
}

fn transfer_asset(
what: &MultiAsset,
from: &MultiLocation,
to: &MultiLocation,
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
context: &XcmContext,
) -> result::Result<xcm_executor::Assets, XcmError> {
Szegoo marked this conversation as resolved.
Show resolved Hide resolved
NonFungibleTransferAdapter::<Asset, Matcher, AccountIdConverter, AccountId>::transfer_asset(
what, from, to, context,
)
}
}
2 changes: 1 addition & 1 deletion polkadot/xcm/xcm-builder/src/nonfungibles_adapter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,7 @@
// 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::fungibles`] through XCM.
//! Adapters to work with [`frame_support::traits::tokens::nonfungibles`] through XCM.

use crate::{AssetChecking, MintLocation};
use frame_support::{
Expand Down
13 changes: 13 additions & 0 deletions prdoc/pr_2924.prdoc
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
# Schema: Polkadot SDK PRDoc Schema (prdoc) v1.0.0
# See doc at https://raw.githubusercontent.com/paritytech/polkadot-sdk/master/prdoc/schema_user.json

title: Add NonFungibleAdapter

doc:
- audience: Runtime Dev
description: |
Introduces a new adapter, `NonFungibleAdapter`, to work with `frame_support::traits::tokens::nonfungible`
through XCM.

crates:
- name: staging-xcm-builder
Loading