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

Polkadot-1.7.2: restricted transfer location changes #1853

Merged
merged 8 commits into from
Jun 6, 2024
Merged
Show file tree
Hide file tree
Changes from 5 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
1 change: 1 addition & 0 deletions Cargo.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

33 changes: 7 additions & 26 deletions libs/types/src/locations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,21 +14,20 @@ use cfg_primitives::AccountId;
use frame_support::RuntimeDebugNoBound;
use parity_scale_codec::{Decode, Encode, MaxEncodedLen};
use scale_info::TypeInfo;
use sp_core::{crypto::AccountId32, H256};
use sp_runtime::traits::{BlakeTwo256, Hash};
use staging_xcm::VersionedLocation;
use sp_core::crypto::AccountId32;
// Please note that if this version change,
// a migration could be required in those places where
// RestrictedTransferLocation is stored
use staging_xcm::v4::Location;

use crate::domain_address::DomainAddress;
/// Location types for destinations that can receive restricted transfers
#[derive(Clone, RuntimeDebugNoBound, Encode, Decode, Eq, PartialEq, MaxEncodedLen, TypeInfo)]
pub enum RestrictedTransferLocation {
/// Local chain account sending destination.
Local(AccountId),
/// XCM MultiLocation sending destinations.
/// Using hash value here as Multilocation is large -- v1 is 512 bytes, but
/// next largest is only 40 bytes other values aren't hashed as we have
/// blake2 hashing on storage map keys, and we don't want the extra overhead
XCM(H256),
/// XCM Location sending destinations.
Xcm(Location),
lemunozm marked this conversation as resolved.
Show resolved Hide resolved
Copy link
Contributor Author

Choose a reason for hiding this comment

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

@onnovisser from this, you do not need to create the hash, just put the location (will be probably a VersionedLocation)

/// DomainAddress sending location from a liquidity pools' instance
Address(DomainAddress),
}
Expand All @@ -38,21 +37,3 @@ impl From<AccountId32> for RestrictedTransferLocation {
Self::Local(value)
}
}

impl From<VersionedLocation> for RestrictedTransferLocation {
fn from(vml: VersionedLocation) -> Self {
lemunozm marked this conversation as resolved.
Show resolved Hide resolved
// using hash here as multilocation is significantly larger than any other enum
// type here -- 592 bytes, vs 40 bytes for domain address (next largest)
Self::XCM(BlakeTwo256::hash(&vml.encode()))

// TODO-1.7: I'm afraid of locations translated from v3 to v4 will
// generate a different hash here. How this affect our current chain
// state?
}
}

impl From<DomainAddress> for RestrictedTransferLocation {
fn from(da: DomainAddress) -> Self {
Self::Address(da)
}
}
10 changes: 1 addition & 9 deletions pallets/transfer-allowlist/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -109,15 +109,7 @@ pub mod pallet {
type Deposit: Get<DepositBalanceOf<Self>>;

/// Type containing the locations a transfer can be sent to.
type Location: Member
+ Debug
+ Eq
+ PartialEq
+ TypeInfo
+ Encode
+ EncodeLike
+ Decode
+ MaxEncodedLen;
type Location: Member + TypeInfo + Encode + EncodeLike + Decode + MaxEncodedLen;

/// Type for pallet weights
type WeightInfo: WeightInfo;
Expand Down
3 changes: 3 additions & 0 deletions runtime/centrifuge/src/migrations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -31,4 +31,7 @@ pub type UpgradeCentrifuge1029 = (
cumulus_pallet_xcmp_queue::migration::v4::MigrationToV4<crate::Runtime>,
pallet_identity::migration::versioned::V0ToV1<crate::Runtime, IDENTITY_MIGRATION_KEY_LIMIT>,
pallet_uniques::migration::MigrateV0ToV1<crate::Runtime, ()>,
runtime_common::migrations::restricted_location::MigrateRestrictedTransferLocation<
crate::Runtime,
>,
);
1 change: 1 addition & 0 deletions runtime/common/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ repository.workspace = true
documentation.workspace = true

[dependencies]
hex = { workspace = true }
hex-literal = { workspace = true }
log = { workspace = true }
num_enum = { workspace = true }
Expand Down
1 change: 1 addition & 0 deletions runtime/common/src/migrations/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ pub mod increase_storage_version;
pub mod loans;
pub mod nuke;
pub mod precompile_account_codes;
pub mod restricted_location;

pub mod utils {
use frame_support::storage::unhashed;
Expand Down
141 changes: 141 additions & 0 deletions runtime/common/src/migrations/restricted_location.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,141 @@
use cfg_primitives::AccountId;
use cfg_types::{locations::RestrictedTransferLocation, tokens::FilterCurrency};
use frame_support::{
traits::{Get, OnRuntimeUpgrade},
weights::Weight,
};
use pallet_transfer_allowlist::AccountCurrencyTransferAllowance;
use parity_scale_codec::Encode;
use sp_core::H256;
use sp_runtime::traits::{BlakeTwo256, Hash};
use sp_std::vec::Vec;
use staging_xcm::v4;

mod old {
use cfg_primitives::AccountId;
use cfg_types::{domain_address::DomainAddress, tokens::FilterCurrency};
use frame_support::{pallet_prelude::*, storage_alias};
use frame_system::pallet_prelude::*;
use hex::FromHex;
use pallet_transfer_allowlist::AllowanceDetails;
use sp_core::H256;
use staging_xcm::v3;

#[derive(
Clone, RuntimeDebugNoBound, Encode, Decode, Eq, PartialEq, MaxEncodedLen, TypeInfo,
)]
pub enum RestrictedTransferLocation {
Local(AccountId),
Xcm(H256),
Address(DomainAddress),
}

#[storage_alias]
pub type AccountCurrencyTransferAllowance<T: pallet_transfer_allowlist::Config> = StorageNMap<
pallet_transfer_allowlist::Pallet<T>,
(
NMapKey<Twox64Concat, AccountId>,
NMapKey<Twox64Concat, FilterCurrency>,
NMapKey<Blake2_128Concat, RestrictedTransferLocation>,
),
AllowanceDetails<BlockNumberFor<T>>,
OptionQuery,
>;

pub fn location_v3_created_by_apps(_account_id: &AccountId) -> v3::Location {
// Ref: https://github.com/centrifuge/apps/blob/b59bdd34561a4ccd90e0d803c14a3729fc2f3a6d/centrifuge-app/src/utils/usePermissions.tsx#L386
// for account_id == "4dTeMxuPJCK7zQGhFcgCivSJqBs9Wo2SuMSQeYCCuVJ9xrE2"
v3::Location::new(
1,
v3::Junctions::X2(
v3::Junction::Parachain(1000), // AssetHub
v3::Junction::AccountId32 {
network: None,
id: <[u8; 32]>::from_hex(
// Address used by Anemoy to withdraw in AssetHub
"10c03288a534d77418e3c19e745dfbc952423e179e1e3baa89e287092fc7802f",
)
.expect("keep in the array"),
},
),
)
}
}
lemunozm marked this conversation as resolved.
Show resolved Hide resolved

const LOG_PREFIX: &str = "MigrateRestrictedTransferLocation:";

fn migrate_location_key(account_id: &AccountId, hash: H256) -> Option<RestrictedTransferLocation> {
let old_location = old::location_v3_created_by_apps(account_id);
if BlakeTwo256::hash(&old_location.encode()) == hash {
match v4::Location::try_from(old_location) {
Ok(location) => {
log::info!("{LOG_PREFIX} Hash: '{hash}' migrated!");
let new_restricted_location = RestrictedTransferLocation::Xcm(location);

Some(new_restricted_location)
}
Err(_) => {
log::error!("{LOG_PREFIX} Non isometric location v3 -> v4");
None
}
}
} else {
log::error!("{LOG_PREFIX} Hash can not be recovered");
None
}
}

pub struct MigrateRestrictedTransferLocation<T>(sp_std::marker::PhantomData<T>);
impl<T> OnRuntimeUpgrade for MigrateRestrictedTransferLocation<T>
where
T: pallet_transfer_allowlist::Config<
AccountId = AccountId,
CurrencyId = FilterCurrency,
Location = RestrictedTransferLocation,
>,
{
fn on_runtime_upgrade() -> Weight {
log::info!("{LOG_PREFIX} Check keys to migrate...");

let mut weight = Weight::zero();

let key_translations = old::AccountCurrencyTransferAllowance::<T>::iter_keys()
.filter_map(|(account_id, currency_id, old_restricted_location)| {
weight.saturating_accrue(T::DbWeight::get().reads(1));
match old_restricted_location {
old::RestrictedTransferLocation::Xcm(hash) => {
migrate_location_key(&account_id, hash).map(|new_restricted_location| {
(
(account_id.clone(), currency_id, old_restricted_location),
(account_id, currency_id, new_restricted_location),
)
})
}
_ => None,
}
})
.collect::<Vec<_>>();

for (old_key, new_key) in key_translations {
log::info!("{LOG_PREFIX} Remove {old_key:?} and add {new_key:?}");

let value = old::AccountCurrencyTransferAllowance::<T>::get(&old_key);
old::AccountCurrencyTransferAllowance::<T>::remove(old_key);
AccountCurrencyTransferAllowance::<T>::set(new_key, value);

weight.saturating_accrue(T::DbWeight::get().writes(2));
}

weight
}

#[cfg(feature = "try-runtime")]
fn pre_upgrade() -> Result<Vec<u8>, sp_runtime::TryRuntimeError> {
Ok(Vec::new())
}

#[cfg(feature = "try-runtime")]
fn post_upgrade(_pre_state: Vec<u8>) -> Result<(), sp_runtime::TryRuntimeError> {
Ok(())
}
lemunozm marked this conversation as resolved.
Show resolved Hide resolved
}
7 changes: 3 additions & 4 deletions runtime/common/src/transfer_filter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,8 @@ use pallet_restricted_tokens::TransferDetails;
use pallet_restricted_xtokens::TransferEffects;
use parity_scale_codec::{Decode, Encode};
use scale_info::TypeInfo;
use sp_core::Hasher;
use sp_runtime::{
traits::{BlakeTwo256, Convert, DispatchInfoOf, SignedExtension, StaticLookup},
traits::{Convert, DispatchInfoOf, SignedExtension, StaticLookup},
transaction_validity::{InvalidTransaction, TransactionValidityError},
DispatchError, DispatchResult, TokenError,
};
Expand All @@ -49,12 +48,12 @@ impl<
amalgamate_allowance(
T::allowance(
sender.clone(),
RestrictedTransferLocation::XCM(BlakeTwo256::hash(&destination.encode())),
RestrictedTransferLocation::Xcm(destination.clone()),
FilterCurrency::Specific(currency),
),
T::allowance(
sender,
RestrictedTransferLocation::XCM(BlakeTwo256::hash(&destination.encode())),
RestrictedTransferLocation::Xcm(destination),
FilterCurrency::All,
),
)
Expand Down
23 changes: 10 additions & 13 deletions runtime/integration-tests/src/generic/cases/liquidity_pools.rs
Original file line number Diff line number Diff line change
Expand Up @@ -6700,7 +6700,7 @@ mod centrifuge {
}

fn allowed_xcm_location() -> RestrictedTransferLocation {
RestrictedTransferLocation::XCM(BlakeTwo256::hash(&xcm_location().encode()))
RestrictedTransferLocation::Xcm(xcm_location())
}

fn add_allowance<T: Runtime>(
Expand Down Expand Up @@ -7220,18 +7220,15 @@ mod centrifuge {
pallet_transfer_allowlist::Pallet::<T>::add_transfer_allowance(
RawOrigin::Signed(Keyring::Alice.into()).into(),
FilterCurrency::Specific(USDC),
RestrictedTransferLocation::XCM(BlakeTwo256::hash(
&Location::new(
1,
[
Parachain(T::FudgeHandle::SIBLING_ID),
Junction::AccountId32 {
id: Keyring::Alice.into(),
network: None,
}
]
)
.encode()
RestrictedTransferLocation::Xcm(Location::new(
1,
[
Parachain(T::FudgeHandle::SIBLING_ID),
Junction::AccountId32 {
id: Keyring::Alice.into(),
network: None,
}
]
))
)
);
Expand Down
Loading