479010094e
- Remove missing cli crate from workspace members - Fix TOML array syntax errors in pvf and benchmarking-cli Cargo.toml - Fix Rust import ordering with cargo fmt - Fix feature propagation with zepter (try-runtime, runtime-benchmarks, std)
453 lines
15 KiB
Rust
453 lines
15 KiB
Rust
// Copyright (C) Parity Technologies (UK) Ltd. and Dijital Kurdistan Tech Institute
|
|
// This file is part of Parity Bridges Common.
|
|
|
|
// Parity Bridges Common 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.
|
|
|
|
// Parity Bridges Common 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 Parity Bridges Common. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
//! A module that is responsible for migration of storage.
|
|
|
|
use alloc::vec::Vec;
|
|
use pezframe_support::{
|
|
traits::{Get, StorageVersion},
|
|
weights::Weight,
|
|
};
|
|
|
|
/// The in-code storage version.
|
|
pub const STORAGE_VERSION: StorageVersion = StorageVersion::new(2);
|
|
|
|
/// This module contains data structures that are valid for the initial state of `0`.
|
|
/// (used with v1 migration).
|
|
pub mod v0 {
|
|
use crate::{Config, Pezpallet};
|
|
use codec::{Codec, Decode, Encode, EncodeLike, MaxEncodedLen};
|
|
use core::marker::PhantomData;
|
|
use pezbp_relayers::RewardsAccountOwner;
|
|
use pezbp_runtime::{ChainId, StorageDoubleMapKeyProvider};
|
|
use pezframe_support::{pezpallet_prelude::OptionQuery, Blake2_128Concat, Identity};
|
|
use pezsp_runtime::traits::AccountIdConversion;
|
|
use scale_info::TypeInfo;
|
|
|
|
/// Structure used to identify the account that pays a reward to the relayer.
|
|
#[derive(Copy, Clone, Debug, Decode, Encode, Eq, PartialEq, TypeInfo, MaxEncodedLen)]
|
|
pub struct RewardsAccountParams<LaneId> {
|
|
/// lane_id
|
|
pub lane_id: LaneId,
|
|
/// bridged_chain_id
|
|
pub bridged_chain_id: ChainId,
|
|
/// owner
|
|
pub owner: RewardsAccountOwner,
|
|
}
|
|
|
|
impl<LaneId: Decode + Encode> RewardsAccountParams<LaneId> {
|
|
/// Create a new instance of `RewardsAccountParams`.
|
|
pub const fn new(
|
|
lane_id: LaneId,
|
|
bridged_chain_id: ChainId,
|
|
owner: RewardsAccountOwner,
|
|
) -> Self {
|
|
Self { lane_id, bridged_chain_id, owner }
|
|
}
|
|
}
|
|
|
|
impl<LaneId> pezsp_runtime::TypeId for RewardsAccountParams<LaneId> {
|
|
const TYPE_ID: [u8; 4] = *b"brap";
|
|
}
|
|
|
|
pub(crate) struct RelayerRewardsKeyProvider<AccountId, RewardBalance, LaneId>(
|
|
PhantomData<(AccountId, RewardBalance, LaneId)>,
|
|
);
|
|
|
|
impl<AccountId, RewardBalance, LaneId> StorageDoubleMapKeyProvider
|
|
for RelayerRewardsKeyProvider<AccountId, RewardBalance, LaneId>
|
|
where
|
|
AccountId: 'static + Codec + EncodeLike + Send + Sync,
|
|
RewardBalance: 'static + Codec + EncodeLike + Send + Sync,
|
|
LaneId: Codec + EncodeLike + Send + Sync,
|
|
{
|
|
const MAP_NAME: &'static str = "RelayerRewards";
|
|
|
|
type Hasher1 = Blake2_128Concat;
|
|
type Key1 = AccountId;
|
|
type Hasher2 = Identity;
|
|
type Key2 = RewardsAccountParams<LaneId>;
|
|
type Value = RewardBalance;
|
|
}
|
|
|
|
pub(crate) type RelayerRewardsKeyProviderOf<T, I, LaneId> = RelayerRewardsKeyProvider<
|
|
<T as pezframe_system::Config>::AccountId,
|
|
<T as Config<I>>::RewardBalance,
|
|
LaneId,
|
|
>;
|
|
|
|
#[pezframe_support::storage_alias]
|
|
pub(crate) type RelayerRewards<T: Config<I>, I: 'static, LaneId> = StorageDoubleMap<
|
|
Pezpallet<T, I>,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Hasher1,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Key1,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Hasher2,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Key2,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Value,
|
|
OptionQuery,
|
|
>;
|
|
|
|
/// Reward account generator for `v0`.
|
|
pub struct PayRewardFromAccount<Account, LaneId>(PhantomData<(Account, LaneId)>);
|
|
impl<Account, LaneId> PayRewardFromAccount<Account, LaneId>
|
|
where
|
|
Account: Decode + Encode,
|
|
LaneId: Decode + Encode,
|
|
{
|
|
/// Return account that pays rewards based on the provided parameters.
|
|
pub fn rewards_account(params: RewardsAccountParams<LaneId>) -> Account {
|
|
params.into_sub_account_truncating(b"rewards-account")
|
|
}
|
|
}
|
|
}
|
|
|
|
/// This migration updates `RelayerRewards` where `RewardsAccountParams` was used as the key with
|
|
/// `lane_id` as the first attribute, which affects `into_sub_account_truncating`. We are migrating
|
|
/// this key to use the new `RewardsAccountParams` where `lane_id` is the last attribute.
|
|
pub mod v1 {
|
|
use super::*;
|
|
use crate::{Config, Pezpallet};
|
|
use codec::{Codec, EncodeLike};
|
|
use core::marker::PhantomData;
|
|
use pezbp_messages::LaneIdType;
|
|
use pezbp_relayers::RewardsAccountParams;
|
|
use pezbp_runtime::StorageDoubleMapKeyProvider;
|
|
use pezframe_support::{
|
|
pezpallet_prelude::OptionQuery, traits::UncheckedOnRuntimeUpgrade, Blake2_128Concat,
|
|
Identity,
|
|
};
|
|
use pezsp_arithmetic::traits::Zero;
|
|
|
|
pub(crate) struct RelayerRewardsKeyProvider<AccountId, RewardBalance, LaneId>(
|
|
PhantomData<(AccountId, RewardBalance, LaneId)>,
|
|
);
|
|
|
|
impl<AccountId, RewardBalance, LaneId> StorageDoubleMapKeyProvider
|
|
for RelayerRewardsKeyProvider<AccountId, RewardBalance, LaneId>
|
|
where
|
|
AccountId: 'static + Codec + EncodeLike + Send + Sync,
|
|
RewardBalance: 'static + Codec + EncodeLike + Send + Sync,
|
|
LaneId: Codec + EncodeLike + Send + Sync,
|
|
{
|
|
const MAP_NAME: &'static str = "RelayerRewards";
|
|
|
|
type Hasher1 = Blake2_128Concat;
|
|
type Key1 = AccountId;
|
|
type Hasher2 = Identity;
|
|
type Key2 = v1::RewardsAccountParams<LaneId>;
|
|
type Value = RewardBalance;
|
|
}
|
|
|
|
pub(crate) type RelayerRewardsKeyProviderOf<T, I, LaneId> = RelayerRewardsKeyProvider<
|
|
<T as pezframe_system::Config>::AccountId,
|
|
<T as Config<I>>::RewardBalance,
|
|
LaneId,
|
|
>;
|
|
|
|
#[pezframe_support::storage_alias]
|
|
pub(crate) type RelayerRewards<T: Config<I>, I: 'static, LaneId> = StorageDoubleMap<
|
|
Pezpallet<T, I>,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Hasher1,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Key1,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Hasher2,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Key2,
|
|
<RelayerRewardsKeyProviderOf<T, I, LaneId> as StorageDoubleMapKeyProvider>::Value,
|
|
OptionQuery,
|
|
>;
|
|
|
|
// Copy of `Pezpallet::<T, I>::register_relayer_reward` compatible with v1.
|
|
fn register_relayer_reward_for_v1<
|
|
T: Config<I>,
|
|
I: 'static,
|
|
LaneId: LaneIdType + Send + Sync,
|
|
>(
|
|
rewards_account_params: v1::RewardsAccountParams<LaneId>,
|
|
relayer: &T::AccountId,
|
|
reward_balance: T::RewardBalance,
|
|
) {
|
|
use pezsp_runtime::Saturating;
|
|
|
|
if reward_balance.is_zero() {
|
|
return;
|
|
}
|
|
|
|
v1::RelayerRewards::<T, I, LaneId>::mutate(
|
|
relayer,
|
|
rewards_account_params,
|
|
|old_reward: &mut Option<T::RewardBalance>| {
|
|
let new_reward =
|
|
old_reward.unwrap_or_else(Zero::zero).saturating_add(reward_balance);
|
|
*old_reward = Some(new_reward);
|
|
|
|
tracing::trace!(
|
|
target: crate::LOG_TARGET,
|
|
?relayer,
|
|
?rewards_account_params,
|
|
?new_reward,
|
|
"Relayer can now claim reward"
|
|
);
|
|
},
|
|
);
|
|
}
|
|
|
|
/// Migrates the pezpallet storage to v1.
|
|
pub struct UncheckedMigrationV0ToV1<T, I, LaneId>(PhantomData<(T, I, LaneId)>);
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
const LOG_TARGET: &str = "runtime::bridge-relayers-migration";
|
|
|
|
impl<T: Config<I>, I: 'static, LaneId: LaneIdType + Send + Sync> UncheckedOnRuntimeUpgrade
|
|
for UncheckedMigrationV0ToV1<T, I, LaneId>
|
|
{
|
|
fn on_runtime_upgrade() -> Weight {
|
|
let mut weight = T::DbWeight::get().reads(1);
|
|
|
|
// list all rewards (we cannot do this as one step because of `drain` limitation)
|
|
let mut rewards_to_migrate =
|
|
Vec::with_capacity(v0::RelayerRewards::<T, I, LaneId>::iter().count());
|
|
for (key1, key2, reward) in v0::RelayerRewards::<T, I, LaneId>::drain() {
|
|
rewards_to_migrate.push((key1, key2, reward));
|
|
weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1));
|
|
}
|
|
|
|
// re-register rewards with new format of `RewardsAccountParams`.
|
|
for (key1, key2, reward) in rewards_to_migrate {
|
|
// expand old key
|
|
let v0::RewardsAccountParams { owner, lane_id, bridged_chain_id } = key2;
|
|
|
|
// re-register reward
|
|
register_relayer_reward_for_v1::<T, I, LaneId>(
|
|
v1::RewardsAccountParams::new(lane_id, bridged_chain_id, owner),
|
|
&key1,
|
|
reward,
|
|
);
|
|
weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1));
|
|
}
|
|
|
|
weight
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn pre_upgrade() -> Result<Vec<u8>, pezsp_runtime::DispatchError> {
|
|
use codec::Encode;
|
|
use pezframe_support::BoundedBTreeMap;
|
|
use pezsp_runtime::traits::ConstU32;
|
|
|
|
// collect actual rewards
|
|
let mut rewards: BoundedBTreeMap<
|
|
(T::AccountId, LaneId),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = BoundedBTreeMap::new();
|
|
for (key1, key2, reward) in v0::RelayerRewards::<T, I, LaneId>::iter() {
|
|
tracing::info!(target: LOG_TARGET, ?key1, ?key2, ?reward, "Reward to migrate");
|
|
rewards = rewards
|
|
.try_mutate(|inner| {
|
|
inner
|
|
.entry((key1.clone(), key2.lane_id))
|
|
.and_modify(|value| *value += reward)
|
|
.or_insert(reward);
|
|
})
|
|
.unwrap();
|
|
}
|
|
tracing::info!(target: LOG_TARGET, ?rewards, "Found total rewards to migrate");
|
|
|
|
Ok(rewards.encode())
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn post_upgrade(state: Vec<u8>) -> Result<(), pezsp_runtime::DispatchError> {
|
|
use codec::Decode;
|
|
use pezframe_support::BoundedBTreeMap;
|
|
use pezsp_runtime::traits::ConstU32;
|
|
|
|
let rewards_before: BoundedBTreeMap<
|
|
(T::AccountId, LaneId),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = Decode::decode(&mut &state[..]).unwrap();
|
|
|
|
// collect migrated rewards
|
|
let mut rewards_after: BoundedBTreeMap<
|
|
(T::AccountId, LaneId),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = BoundedBTreeMap::new();
|
|
for (key1, key2, reward) in v1::RelayerRewards::<T, I, LaneId>::iter() {
|
|
tracing::info!(target: LOG_TARGET, ?key1, ?key2, ?reward, "Migrated rewards");
|
|
rewards_after = rewards_after
|
|
.try_mutate(|inner| {
|
|
inner
|
|
.entry((key1.clone(), *key2.lane_id()))
|
|
.and_modify(|value| *value += reward)
|
|
.or_insert(reward);
|
|
})
|
|
.unwrap();
|
|
}
|
|
tracing::info!(target: LOG_TARGET, ?rewards_after, "Found total migrated rewards");
|
|
|
|
pezframe_support::ensure!(
|
|
rewards_before == rewards_after,
|
|
"The rewards were not migrated correctly!."
|
|
);
|
|
|
|
tracing::info!(target: LOG_TARGET, "migrated all.");
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// [`UncheckedMigrationV0ToV1`] wrapped in a
|
|
/// [`VersionedMigration`](pezframe_support::migrations::VersionedMigration), ensuring the
|
|
/// migration is only performed when on-chain version is 0.
|
|
pub type MigrationToV1<T, I, LaneId> = pezframe_support::migrations::VersionedMigration<
|
|
0,
|
|
1,
|
|
UncheckedMigrationV0ToV1<T, I, LaneId>,
|
|
Pezpallet<T, I>,
|
|
<T as pezframe_system::Config>::DbWeight,
|
|
>;
|
|
}
|
|
|
|
/// The pezpallet in version 1 only supported rewards collected under the key of
|
|
/// `RewardsAccountParams`. This migration essentially converts existing `RewardsAccountParams` keys
|
|
/// to the generic type `T::Reward`.
|
|
pub mod v2 {
|
|
use super::*;
|
|
#[cfg(feature = "try-runtime")]
|
|
use crate::RelayerRewards;
|
|
use crate::{Config, Pezpallet};
|
|
use core::marker::PhantomData;
|
|
use pezbp_messages::LaneIdType;
|
|
use pezbp_relayers::RewardsAccountParams;
|
|
use pezframe_support::traits::UncheckedOnRuntimeUpgrade;
|
|
|
|
/// Migrates the pezpallet storage to v2.
|
|
pub struct UncheckedMigrationV1ToV2<T, I, LaneId>(PhantomData<(T, I, LaneId)>);
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
const LOG_TARGET: &str = "runtime::bridge-relayers-migration";
|
|
|
|
impl<T: Config<I>, I: 'static, LaneId: LaneIdType + Send + Sync> UncheckedOnRuntimeUpgrade
|
|
for UncheckedMigrationV1ToV2<T, I, LaneId>
|
|
where
|
|
<T as Config<I>>::Reward: From<RewardsAccountParams<LaneId>>,
|
|
{
|
|
fn on_runtime_upgrade() -> Weight {
|
|
let mut weight = T::DbWeight::get().reads(1);
|
|
|
|
// list all rewards (we cannot do this as one step because of `drain` limitation)
|
|
let mut rewards_to_migrate =
|
|
Vec::with_capacity(v1::RelayerRewards::<T, I, LaneId>::iter().count());
|
|
for (key1, key2, reward) in v1::RelayerRewards::<T, I, LaneId>::drain() {
|
|
rewards_to_migrate.push((key1, key2, reward));
|
|
weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1));
|
|
}
|
|
|
|
// re-register rewards with new format.
|
|
for (key1, key2, reward) in rewards_to_migrate {
|
|
// convert old key to the new
|
|
let new_key2: T::Reward = key2.into();
|
|
|
|
// re-register reward (drained above)
|
|
Pezpallet::<T, I>::register_relayer_reward(new_key2, &key1, reward);
|
|
weight.saturating_accrue(T::DbWeight::get().reads_writes(1, 1));
|
|
}
|
|
|
|
weight
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn pre_upgrade() -> Result<Vec<u8>, pezsp_runtime::DispatchError> {
|
|
use codec::Encode;
|
|
use pezframe_support::BoundedBTreeMap;
|
|
use pezsp_runtime::traits::ConstU32;
|
|
|
|
// collect actual rewards
|
|
let mut rewards: BoundedBTreeMap<
|
|
(T::AccountId, Vec<u8>),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = BoundedBTreeMap::new();
|
|
for (key1, key2, reward) in v1::RelayerRewards::<T, I, LaneId>::iter() {
|
|
let new_key2: T::Reward = key2.into();
|
|
tracing::info!(target: LOG_TARGET, ?key1, ?key2, ?new_key2, ?reward, "Reward to migrate");
|
|
rewards = rewards
|
|
.try_mutate(|inner| {
|
|
inner
|
|
.entry((key1.clone(), new_key2.encode()))
|
|
.and_modify(|value| *value += reward)
|
|
.or_insert(reward);
|
|
})
|
|
.unwrap();
|
|
}
|
|
tracing::info!(target: LOG_TARGET, ?rewards, "Found total rewards to migrate");
|
|
|
|
Ok(rewards.encode())
|
|
}
|
|
|
|
#[cfg(feature = "try-runtime")]
|
|
fn post_upgrade(state: Vec<u8>) -> Result<(), pezsp_runtime::DispatchError> {
|
|
use codec::{Decode, Encode};
|
|
use pezframe_support::BoundedBTreeMap;
|
|
use pezsp_runtime::traits::ConstU32;
|
|
|
|
let rewards_before: BoundedBTreeMap<
|
|
(T::AccountId, Vec<u8>),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = Decode::decode(&mut &state[..]).unwrap();
|
|
|
|
// collect migrated rewards
|
|
let mut rewards_after: BoundedBTreeMap<
|
|
(T::AccountId, Vec<u8>),
|
|
T::RewardBalance,
|
|
ConstU32<{ u32::MAX }>,
|
|
> = BoundedBTreeMap::new();
|
|
for (key1, key2, reward) in v2::RelayerRewards::<T, I>::iter() {
|
|
tracing::info!(target: LOG_TARGET, ?key1, ?key2, ?reward, "Migrated rewards");
|
|
rewards_after = rewards_after
|
|
.try_mutate(|inner| {
|
|
inner
|
|
.entry((key1.clone(), key2.encode()))
|
|
.and_modify(|value| *value += reward)
|
|
.or_insert(reward);
|
|
})
|
|
.unwrap();
|
|
}
|
|
tracing::info!(target: LOG_TARGET, ?rewards_after, "Found total migrated rewards");
|
|
|
|
pezframe_support::ensure!(
|
|
rewards_before == rewards_after,
|
|
"The rewards were not migrated correctly!."
|
|
);
|
|
|
|
tracing::info!(target: LOG_TARGET, "migrated all.");
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
/// [`UncheckedMigrationV1ToV2`] wrapped in a
|
|
/// [`VersionedMigration`](pezframe_support::migrations::VersionedMigration), ensuring the
|
|
/// migration is only performed when on-chain version is 1.
|
|
pub type MigrationToV2<T, I, LaneId> = pezframe_support::migrations::VersionedMigration<
|
|
1,
|
|
2,
|
|
UncheckedMigrationV1ToV2<T, I, LaneId>,
|
|
Pezpallet<T, I>,
|
|
<T as pezframe_system::Config>::DbWeight,
|
|
>;
|
|
}
|