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

Allow two Parachains to swap #4772

Merged
merged 26 commits into from
Feb 12, 2022
Merged
Show file tree
Hide file tree
Changes from 25 commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
fd778cf
add support for parachain to parachain swap
shawntabrizi Jan 23, 2022
3eaa619
enable swaps on kusama
shawntabrizi Jan 23, 2022
31426cf
sanity test in paras_registrar
shawntabrizi Jan 23, 2022
d457803
express more errors
shawntabrizi Jan 23, 2022
331a835
finish up tests
shawntabrizi Jan 23, 2022
bc23e47
fmt
shawntabrizi Jan 23, 2022
3916763
Merge branch 'master' into shawntabrizi-complete-swap
shawntabrizi Jan 23, 2022
578ad43
make fields pub
shawntabrizi Jan 23, 2022
4981fdb
refactor integration tests to use real accounts
shawntabrizi Jan 23, 2022
2527a7c
Merge branch 'master' into shawntabrizi-complete-swap
shawntabrizi Feb 1, 2022
9548b0a
Update Crowdloan Account to FundIndex (#4824)
shawntabrizi Feb 1, 2022
c643273
finish parachain swap test
shawntabrizi Feb 1, 2022
a54d1f2
format
shawntabrizi Feb 1, 2022
c963177
fix warning
shawntabrizi Feb 1, 2022
3d85c33
fix spacing
shawntabrizi Feb 1, 2022
19f7a04
fix formatting
shawntabrizi Feb 1, 2022
ba1dc91
write migrations
shawntabrizi Feb 1, 2022
ed04dc9
add migration
shawntabrizi Feb 1, 2022
61df5dc
fixes
shawntabrizi Feb 1, 2022
603771d
more fixes to migration
shawntabrizi Feb 1, 2022
5d82def
Merge branch 'master' into shawntabrizi-complete-swap
shawntabrizi Feb 1, 2022
0ad9e8f
Update runtime/common/src/crowdloan/mod.rs
shawntabrizi Feb 1, 2022
e38d881
Update runtime/common/src/paras_registrar.rs
shawntabrizi Feb 1, 2022
5c3137b
Merge branch 'master' into shawntabrizi-complete-swap
shawntabrizi Feb 6, 2022
dd2eaca
Update migration.rs
shawntabrizi Feb 7, 2022
6217ef5
extract swap function
shawntabrizi Feb 7, 2022
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
130 changes: 130 additions & 0 deletions runtime/common/src/crowdloan/migration.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
// Copyright 2017-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/>.

use super::*;
use frame_support::generate_storage_alias;

/// Migrations for using fund index to create fund accounts instead of para ID.
pub mod crowdloan_index_migration {
use super::*;

// The old way we generated fund accounts.
fn old_fund_account_id<T: Config>(index: ParaId) -> T::AccountId {
emostov marked this conversation as resolved.
Show resolved Hide resolved
T::PalletId::get().into_sub_account(index)
}

pub fn pre_migrate<T: Config>() -> Result<(), &'static str> {
// `NextTrieIndex` should have a value.
generate_storage_alias!(Crowdloan, NextTrieIndex => Value<FundIndex>);
let next_index = NextTrieIndex::get().unwrap_or_default();
emostov marked this conversation as resolved.
Show resolved Hide resolved
ensure!(next_index > 0, "Next index is zero, which implies no migration is needed.");

log::info!(
target: "runtime",
"next trie index: {:?}",
next_index,
);
emostov marked this conversation as resolved.
Show resolved Hide resolved

// Each fund should have some non-zero balance.
for (para_id, fund) in Funds::<T>::iter() {
let old_fund_account = old_fund_account_id::<T>(para_id);
let total_balance = CurrencyOf::<T>::total_balance(&old_fund_account);

log::info!(
target: "runtime",
"para_id={:?}, old_fund_account={:?}, total_balance={:?}, fund.raised={:?}",
para_id, old_fund_account, total_balance, fund.raised
emostov marked this conversation as resolved.
Show resolved Hide resolved
);

ensure!(
total_balance >= fund.raised,
"Total balance is not equal to the funds raised."
);
ensure!(total_balance > Zero::zero(), "Total balance is equal to zero.");
}

Ok(())
}

/// This migration converts crowdloans to use a crowdloan index rather than the parachain id as a
/// unique identifier. This makes it easier to swap two crowdloans between parachains.
pub fn migrate<T: Config>() -> frame_support::weights::Weight {
let mut weight = 0;

// First migrate `NextTrieIndex` counter to `NextFundIndex`.
generate_storage_alias!(Crowdloan, NextTrieIndex => Value<FundIndex>);

let next_index = NextTrieIndex::take().unwrap_or_default();
NextFundIndex::<T>::set(next_index);

weight = weight.saturating_add(T::DbWeight::get().reads_writes(1, 2));

// Migrate all accounts from `old_fund_account` to `fund_account` using `fund_index`.
for (para_id, fund) in Funds::<T>::iter() {
let old_fund_account = old_fund_account_id::<T>(para_id);
let new_fund_account = Pallet::<T>::fund_account_id(fund.fund_index);

// Funds should only have a free balance and a reserve balance. Both of these are in the
// `Account` storage item, so we just swap them.
let account_info = frame_system::Account::<T>::take(old_fund_account);
frame_system::Account::<T>::insert(new_fund_account, account_info);

weight = weight.saturating_add(T::DbWeight::get().reads_writes(1, 2));
}

weight
}

pub fn post_migrate<T: Config>() -> Result<(), &'static str> {
// `NextTrieIndex` should not have a value, and `NextFundIndex` should.
generate_storage_alias!(Crowdloan, NextTrieIndex => Value<FundIndex>);
ensure!(NextTrieIndex::get().is_none(), "NextTrieIndex still has a value.");

let next_index = NextFundIndex::<T>::get();
log::info!(
target: "runtime",
"next fund index: {:?}",
next_index,
);

ensure!(
next_index > 0,
"NextFundIndex was not migrated or is zero. We assume it cannot be zero else no migration is needed."
);

// Each fund should have balance migrated correctly.
for (para_id, fund) in Funds::<T>::iter() {
// Old fund account is deleted.
let old_fund_account = old_fund_account_id::<T>(para_id);
ensure!(
frame_system::Account::<T>::get(&old_fund_account) == Default::default(),
"Old account wasn't reset to default value."
);

// New fund account has the correct balance.
let new_fund_account = Pallet::<T>::fund_account_id(fund.fund_index);
let total_balance = CurrencyOf::<T>::total_balance(&new_fund_account);

ensure!(
total_balance >= fund.raised,
"Total balance in new account is different than the funds raised."
);
ensure!(total_balance > Zero::zero(), "Total balance in the account is zero.");
}

Ok(())
}
}
Loading