mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-06-19 03:01:02 +00:00
Bridge: make some headers submissions free (#4102)
supersedes https://github.com/paritytech/parity-bridges-common/pull/2873 Draft because of couple of TODOs: - [x] fix remaining TODOs; - [x] double check that all changes from https://github.com/paritytech/parity-bridges-common/pull/2873 are correctly ported; - [x] create a separate PR (on top of that one or a follow up?) for https://github.com/paritytech/polkadot-sdk/tree/sv-try-new-bridge-fees; - [x] fix compilation issues (haven't checked, but there should be many). --------- Co-authored-by: Adrian Catangiu <adrian@parity.io>
This commit is contained in:
committed by
GitHub
parent
4f3d43a0c4
commit
a633e954f3
@@ -14,25 +14,45 @@
|
||||
// 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/>.
|
||||
|
||||
use crate::{Config, Pallet, RelayBlockNumber};
|
||||
use crate::{Config, GrandpaPalletOf, Pallet, RelayBlockHash, RelayBlockNumber};
|
||||
use bp_header_chain::HeaderChain;
|
||||
use bp_parachains::BestParaHeadHash;
|
||||
use bp_polkadot_core::parachains::{ParaHash, ParaId};
|
||||
use bp_runtime::OwnedBridgeModule;
|
||||
use frame_support::{dispatch::CallableCallFor, traits::IsSubType};
|
||||
use bp_runtime::{HeaderId, OwnedBridgeModule};
|
||||
use frame_support::{
|
||||
dispatch::CallableCallFor,
|
||||
traits::{Get, IsSubType},
|
||||
};
|
||||
use pallet_bridge_grandpa::SubmitFinalityProofHelper;
|
||||
use sp_runtime::{
|
||||
transaction_validity::{InvalidTransaction, TransactionValidity, ValidTransaction},
|
||||
traits::Zero,
|
||||
transaction_validity::{InvalidTransaction, TransactionValidityError},
|
||||
RuntimeDebug,
|
||||
};
|
||||
|
||||
/// Info about a `SubmitParachainHeads` call which tries to update a single parachain.
|
||||
#[derive(PartialEq, RuntimeDebug)]
|
||||
pub struct SubmitParachainHeadsInfo {
|
||||
/// Number of the finalized relay block that has been used to prove parachain finality.
|
||||
pub at_relay_block_number: RelayBlockNumber,
|
||||
/// Number and hash of the finalized relay block that has been used to prove parachain
|
||||
/// finality.
|
||||
pub at_relay_block: HeaderId<RelayBlockHash, RelayBlockNumber>,
|
||||
/// Parachain identifier.
|
||||
pub para_id: ParaId,
|
||||
/// Hash of the bundled parachain head.
|
||||
pub para_head_hash: ParaHash,
|
||||
/// If `true`, then the call must be free (assuming that everything else is valid) to
|
||||
/// be treated as valid.
|
||||
pub is_free_execution_expected: bool,
|
||||
}
|
||||
|
||||
/// Verified `SubmitParachainHeadsInfo`.
|
||||
#[derive(PartialEq, RuntimeDebug)]
|
||||
pub struct VerifiedSubmitParachainHeadsInfo {
|
||||
/// Base call information.
|
||||
pub base: SubmitParachainHeadsInfo,
|
||||
/// A difference between bundled bridged relay chain header and relay chain header number
|
||||
/// used to prove best bridged parachain header, known to us before the call.
|
||||
pub improved_by: RelayBlockNumber,
|
||||
}
|
||||
|
||||
/// Helper struct that provides methods for working with the `SubmitParachainHeads` call.
|
||||
@@ -41,40 +61,117 @@ pub struct SubmitParachainHeadsHelper<T: Config<I>, I: 'static> {
|
||||
}
|
||||
|
||||
impl<T: Config<I>, I: 'static> SubmitParachainHeadsHelper<T, I> {
|
||||
/// Check if the para head provided by the `SubmitParachainHeads` is better than the best one
|
||||
/// we know.
|
||||
pub fn is_obsolete(update: &SubmitParachainHeadsInfo) -> bool {
|
||||
let stored_best_head = match crate::ParasInfo::<T, I>::get(update.para_id) {
|
||||
Some(stored_best_head) => stored_best_head,
|
||||
None => return false,
|
||||
/// Check that is called from signed extension and takes the `is_free_execution_expected`
|
||||
/// into account.
|
||||
pub fn check_obsolete_from_extension(
|
||||
update: &SubmitParachainHeadsInfo,
|
||||
) -> Result<RelayBlockNumber, TransactionValidityError> {
|
||||
// first do all base checks
|
||||
let improved_by = Self::check_obsolete(update)?;
|
||||
|
||||
// if we don't expect free execution - no more checks
|
||||
if !update.is_free_execution_expected {
|
||||
return Ok(improved_by);
|
||||
}
|
||||
|
||||
// reject if no more free slots remaining in the block
|
||||
if !SubmitFinalityProofHelper::<T, T::BridgesGrandpaPalletInstance>::has_free_header_slots()
|
||||
{
|
||||
log::trace!(
|
||||
target: crate::LOG_TARGET,
|
||||
"The free parachain {:?} head can't be updated: no more free slots \
|
||||
left in the block.",
|
||||
update.para_id,
|
||||
);
|
||||
|
||||
return Err(InvalidTransaction::Call.into());
|
||||
}
|
||||
|
||||
// if free headers interval is not configured and call is expected to execute
|
||||
// for free => it is a relayer error, it should've been able to detect that.
|
||||
let free_headers_interval = match T::FreeHeadersInterval::get() {
|
||||
Some(free_headers_interval) => free_headers_interval,
|
||||
None => return Ok(improved_by),
|
||||
};
|
||||
|
||||
if stored_best_head.best_head_hash.at_relay_block_number >= update.at_relay_block_number {
|
||||
// reject if we are importing parachain headers too often
|
||||
if improved_by < free_headers_interval {
|
||||
log::trace!(
|
||||
target: crate::LOG_TARGET,
|
||||
"The parachain head can't be updated. The parachain head for {:?} \
|
||||
was already updated at better relay chain block {} >= {}.",
|
||||
"The free parachain {:?} head can't be updated: it improves previous
|
||||
best head by {} while at least {} is expected.",
|
||||
update.para_id,
|
||||
stored_best_head.best_head_hash.at_relay_block_number,
|
||||
update.at_relay_block_number
|
||||
improved_by,
|
||||
free_headers_interval,
|
||||
);
|
||||
return true
|
||||
|
||||
return Err(InvalidTransaction::Stale.into());
|
||||
}
|
||||
|
||||
if stored_best_head.best_head_hash.head_hash == update.para_head_hash {
|
||||
Ok(improved_by)
|
||||
}
|
||||
|
||||
/// Check if the para head provided by the `SubmitParachainHeads` is better than the best one
|
||||
/// we know.
|
||||
pub fn check_obsolete(
|
||||
update: &SubmitParachainHeadsInfo,
|
||||
) -> Result<RelayBlockNumber, TransactionValidityError> {
|
||||
// check if we know better parachain head already
|
||||
let improved_by = match crate::ParasInfo::<T, I>::get(update.para_id) {
|
||||
Some(stored_best_head) => {
|
||||
let improved_by = match update
|
||||
.at_relay_block
|
||||
.0
|
||||
.checked_sub(stored_best_head.best_head_hash.at_relay_block_number)
|
||||
{
|
||||
Some(improved_by) if improved_by > Zero::zero() => improved_by,
|
||||
_ => {
|
||||
log::trace!(
|
||||
target: crate::LOG_TARGET,
|
||||
"The parachain head can't be updated. The parachain head for {:?} \
|
||||
was already updated at better relay chain block {} >= {}.",
|
||||
update.para_id,
|
||||
stored_best_head.best_head_hash.at_relay_block_number,
|
||||
update.at_relay_block.0
|
||||
);
|
||||
return Err(InvalidTransaction::Stale.into())
|
||||
},
|
||||
};
|
||||
|
||||
if stored_best_head.best_head_hash.head_hash == update.para_head_hash {
|
||||
log::trace!(
|
||||
target: crate::LOG_TARGET,
|
||||
"The parachain head can't be updated. The parachain head hash for {:?} \
|
||||
was already updated to {} at block {} < {}.",
|
||||
update.para_id,
|
||||
update.para_head_hash,
|
||||
stored_best_head.best_head_hash.at_relay_block_number,
|
||||
update.at_relay_block.0
|
||||
);
|
||||
return Err(InvalidTransaction::Stale.into())
|
||||
}
|
||||
|
||||
improved_by
|
||||
},
|
||||
None => RelayBlockNumber::MAX,
|
||||
};
|
||||
|
||||
// let's check if our chain had no reorgs and we still know the relay chain header
|
||||
// used to craft the proof
|
||||
if GrandpaPalletOf::<T, I>::finalized_header_state_root(update.at_relay_block.1).is_none() {
|
||||
log::trace!(
|
||||
target: crate::LOG_TARGET,
|
||||
"The parachain head can't be updated. The parachain head hash for {:?} \
|
||||
was already updated to {} at block {} < {}.",
|
||||
"The parachain {:?} head can't be updated. Relay chain header {}/{} used to create \
|
||||
parachain proof is missing from the storage.",
|
||||
update.para_id,
|
||||
update.para_head_hash,
|
||||
stored_best_head.best_head_hash.at_relay_block_number,
|
||||
update.at_relay_block_number
|
||||
update.at_relay_block.0,
|
||||
update.at_relay_block.1,
|
||||
);
|
||||
return true
|
||||
|
||||
return Err(InvalidTransaction::Call.into())
|
||||
}
|
||||
|
||||
false
|
||||
Ok(improved_by)
|
||||
}
|
||||
|
||||
/// Check if the `SubmitParachainHeads` was successfully executed.
|
||||
@@ -83,7 +180,7 @@ impl<T: Config<I>, I: 'static> SubmitParachainHeadsHelper<T, I> {
|
||||
Some(stored_best_head) =>
|
||||
stored_best_head.best_head_hash ==
|
||||
BestParaHeadHash {
|
||||
at_relay_block_number: update.at_relay_block_number,
|
||||
at_relay_block_number: update.at_relay_block.0,
|
||||
head_hash: update.para_head_hash,
|
||||
},
|
||||
None => false,
|
||||
@@ -98,22 +195,36 @@ pub trait CallSubType<T: Config<I, RuntimeCall = Self>, I: 'static>:
|
||||
/// Create a new instance of `SubmitParachainHeadsInfo` from a `SubmitParachainHeads` call with
|
||||
/// one single parachain entry.
|
||||
fn one_entry_submit_parachain_heads_info(&self) -> Option<SubmitParachainHeadsInfo> {
|
||||
if let Some(crate::Call::<T, I>::submit_parachain_heads {
|
||||
ref at_relay_block,
|
||||
ref parachains,
|
||||
..
|
||||
}) = self.is_sub_type()
|
||||
{
|
||||
if let &[(para_id, para_head_hash)] = parachains.as_slice() {
|
||||
return Some(SubmitParachainHeadsInfo {
|
||||
at_relay_block_number: at_relay_block.0,
|
||||
match self.is_sub_type() {
|
||||
Some(crate::Call::<T, I>::submit_parachain_heads {
|
||||
ref at_relay_block,
|
||||
ref parachains,
|
||||
..
|
||||
}) => match ¶chains[..] {
|
||||
&[(para_id, para_head_hash)] => Some(SubmitParachainHeadsInfo {
|
||||
at_relay_block: HeaderId(at_relay_block.0, at_relay_block.1),
|
||||
para_id,
|
||||
para_head_hash,
|
||||
})
|
||||
}
|
||||
is_free_execution_expected: false,
|
||||
}),
|
||||
_ => None,
|
||||
},
|
||||
Some(crate::Call::<T, I>::submit_parachain_heads_ex {
|
||||
ref at_relay_block,
|
||||
ref parachains,
|
||||
is_free_execution_expected,
|
||||
..
|
||||
}) => match ¶chains[..] {
|
||||
&[(para_id, para_head_hash)] => Some(SubmitParachainHeadsInfo {
|
||||
at_relay_block: HeaderId(at_relay_block.0, at_relay_block.1),
|
||||
para_id,
|
||||
para_head_hash,
|
||||
is_free_execution_expected: *is_free_execution_expected,
|
||||
}),
|
||||
_ => None,
|
||||
},
|
||||
_ => None,
|
||||
}
|
||||
|
||||
None
|
||||
}
|
||||
|
||||
/// Create a new instance of `SubmitParachainHeadsInfo` from a `SubmitParachainHeads` call with
|
||||
@@ -133,24 +244,23 @@ pub trait CallSubType<T: Config<I, RuntimeCall = Self>, I: 'static>:
|
||||
/// block production, or "eat" significant portion of block production time literally
|
||||
/// for nothing. In addition, the single-parachain-head-per-transaction is how the
|
||||
/// pallet will be used in our environment.
|
||||
fn check_obsolete_submit_parachain_heads(&self) -> TransactionValidity
|
||||
fn check_obsolete_submit_parachain_heads(
|
||||
&self,
|
||||
) -> Result<Option<VerifiedSubmitParachainHeadsInfo>, TransactionValidityError>
|
||||
where
|
||||
Self: Sized,
|
||||
{
|
||||
let update = match self.one_entry_submit_parachain_heads_info() {
|
||||
Some(update) => update,
|
||||
None => return Ok(ValidTransaction::default()),
|
||||
None => return Ok(None),
|
||||
};
|
||||
|
||||
if Pallet::<T, I>::ensure_not_halted().is_err() {
|
||||
return InvalidTransaction::Call.into()
|
||||
return Err(InvalidTransaction::Call.into())
|
||||
}
|
||||
|
||||
if SubmitParachainHeadsHelper::<T, I>::is_obsolete(&update) {
|
||||
return InvalidTransaction::Stale.into()
|
||||
}
|
||||
|
||||
Ok(ValidTransaction::default())
|
||||
SubmitParachainHeadsHelper::<T, I>::check_obsolete_from_extension(&update)
|
||||
.map(|improved_by| Some(VerifiedSubmitParachainHeadsInfo { base: update, improved_by }))
|
||||
}
|
||||
}
|
||||
|
||||
@@ -164,9 +274,10 @@ where
|
||||
#[cfg(test)]
|
||||
mod tests {
|
||||
use crate::{
|
||||
mock::{run_test, RuntimeCall, TestRuntime},
|
||||
CallSubType, PalletOperatingMode, ParaInfo, ParasInfo, RelayBlockNumber,
|
||||
mock::{run_test, FreeHeadersInterval, RuntimeCall, TestRuntime},
|
||||
CallSubType, PalletOperatingMode, ParaInfo, ParasInfo, RelayBlockHash, RelayBlockNumber,
|
||||
};
|
||||
use bp_header_chain::StoredHeaderData;
|
||||
use bp_parachains::BestParaHeadHash;
|
||||
use bp_polkadot_core::parachains::{ParaHash, ParaHeadsProof, ParaId};
|
||||
use bp_runtime::BasicOperatingMode;
|
||||
@@ -175,15 +286,37 @@ mod tests {
|
||||
num: RelayBlockNumber,
|
||||
parachains: Vec<(ParaId, ParaHash)>,
|
||||
) -> bool {
|
||||
RuntimeCall::Parachains(crate::Call::<TestRuntime, ()>::submit_parachain_heads {
|
||||
at_relay_block: (num, Default::default()),
|
||||
RuntimeCall::Parachains(crate::Call::<TestRuntime, ()>::submit_parachain_heads_ex {
|
||||
at_relay_block: (num, [num as u8; 32].into()),
|
||||
parachains,
|
||||
parachain_heads_proof: ParaHeadsProof { storage_proof: Vec::new() },
|
||||
is_free_execution_expected: false,
|
||||
})
|
||||
.check_obsolete_submit_parachain_heads()
|
||||
.is_ok()
|
||||
}
|
||||
|
||||
fn validate_free_submit_parachain_heads(
|
||||
num: RelayBlockNumber,
|
||||
parachains: Vec<(ParaId, ParaHash)>,
|
||||
) -> bool {
|
||||
RuntimeCall::Parachains(crate::Call::<TestRuntime, ()>::submit_parachain_heads_ex {
|
||||
at_relay_block: (num, [num as u8; 32].into()),
|
||||
parachains,
|
||||
parachain_heads_proof: ParaHeadsProof { storage_proof: Vec::new() },
|
||||
is_free_execution_expected: true,
|
||||
})
|
||||
.check_obsolete_submit_parachain_heads()
|
||||
.is_ok()
|
||||
}
|
||||
|
||||
fn insert_relay_block(num: RelayBlockNumber) {
|
||||
pallet_bridge_grandpa::ImportedHeaders::<TestRuntime, crate::Instance1>::insert(
|
||||
RelayBlockHash::from([num as u8; 32]),
|
||||
StoredHeaderData { number: num, state_root: RelayBlockHash::from([10u8; 32]) },
|
||||
);
|
||||
}
|
||||
|
||||
fn sync_to_relay_header_10() {
|
||||
ParasInfo::<TestRuntime, ()>::insert(
|
||||
ParaId(1),
|
||||
@@ -244,6 +377,7 @@ mod tests {
|
||||
// when current best finalized is #10 and we're trying to import header#15 => tx is
|
||||
// accepted
|
||||
sync_to_relay_header_10();
|
||||
insert_relay_block(15);
|
||||
assert!(validate_submit_parachain_heads(15, vec![(ParaId(1), [2u8; 32].into())]));
|
||||
});
|
||||
}
|
||||
@@ -260,4 +394,65 @@ mod tests {
|
||||
));
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn extension_rejects_initial_parachain_head_if_missing_relay_chain_header() {
|
||||
run_test(|| {
|
||||
// when relay chain header is unknown => "obsolete"
|
||||
assert!(!validate_submit_parachain_heads(10, vec![(ParaId(1), [1u8; 32].into())]));
|
||||
// when relay chain header is unknown => "ok"
|
||||
insert_relay_block(10);
|
||||
assert!(validate_submit_parachain_heads(10, vec![(ParaId(1), [1u8; 32].into())]));
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn extension_rejects_free_parachain_head_if_missing_relay_chain_header() {
|
||||
run_test(|| {
|
||||
sync_to_relay_header_10();
|
||||
// when relay chain header is unknown => "obsolete"
|
||||
assert!(!validate_submit_parachain_heads(15, vec![(ParaId(2), [15u8; 32].into())]));
|
||||
// when relay chain header is unknown => "ok"
|
||||
insert_relay_block(15);
|
||||
assert!(validate_submit_parachain_heads(15, vec![(ParaId(2), [15u8; 32].into())]));
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn extension_rejects_free_parachain_head_if_no_free_slots_remaining() {
|
||||
run_test(|| {
|
||||
// when current best finalized is #10 and we're trying to import header#15 => tx should
|
||||
// be accepted
|
||||
sync_to_relay_header_10();
|
||||
insert_relay_block(15);
|
||||
// ... but since we have specified `is_free_execution_expected = true`, it'll be
|
||||
// rejected
|
||||
assert!(!validate_free_submit_parachain_heads(15, vec![(ParaId(1), [2u8; 32].into())]));
|
||||
// ... if we have specify `is_free_execution_expected = false`, it'll be accepted
|
||||
assert!(validate_submit_parachain_heads(15, vec![(ParaId(1), [2u8; 32].into())]));
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn extension_rejects_free_parachain_head_if_improves_by_is_below_expected() {
|
||||
run_test(|| {
|
||||
// when current best finalized is #10 and we're trying to import header#15 => tx should
|
||||
// be accepted
|
||||
sync_to_relay_header_10();
|
||||
insert_relay_block(10 + FreeHeadersInterval::get() - 1);
|
||||
insert_relay_block(10 + FreeHeadersInterval::get());
|
||||
// try to submit at 10 + FreeHeadersInterval::get() - 1 => failure
|
||||
let relay_header = 10 + FreeHeadersInterval::get() - 1;
|
||||
assert!(!validate_free_submit_parachain_heads(
|
||||
relay_header,
|
||||
vec![(ParaId(1), [2u8; 32].into())]
|
||||
));
|
||||
// try to submit at 10 + FreeHeadersInterval::get() => ok
|
||||
let relay_header = 10 + FreeHeadersInterval::get();
|
||||
assert!(validate_free_submit_parachain_heads(
|
||||
relay_header,
|
||||
vec![(ParaId(1), [2u8; 32].into())]
|
||||
));
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@@ -32,6 +32,7 @@ use bp_parachains::{parachain_head_storage_key_at_source, ParaInfo, ParaStoredHe
|
||||
use bp_polkadot_core::parachains::{ParaHash, ParaHead, ParaHeadsProof, ParaId};
|
||||
use bp_runtime::{Chain, HashOf, HeaderId, HeaderIdOf, Parachain, StorageProofError};
|
||||
use frame_support::{dispatch::PostDispatchInfo, DefaultNoBound};
|
||||
use pallet_bridge_grandpa::SubmitFinalityProofHelper;
|
||||
use sp_std::{marker::PhantomData, vec::Vec};
|
||||
|
||||
#[cfg(feature = "runtime-benchmarks")]
|
||||
@@ -92,7 +93,8 @@ pub mod pallet {
|
||||
BoundedStorageValue<<T as Config<I>>::MaxParaHeadDataSize, ParaStoredHeaderData>;
|
||||
/// Weight info of the given parachains pallet.
|
||||
pub type WeightInfoOf<T, I> = <T as Config<I>>::WeightInfo;
|
||||
type GrandpaPalletOf<T, I> =
|
||||
/// Bridge GRANDPA pallet that is used to verify parachain proofs.
|
||||
pub type GrandpaPalletOf<T, I> =
|
||||
pallet_bridge_grandpa::Pallet<T, <T as Config<I>>::BridgesGrandpaPalletInstance>;
|
||||
|
||||
#[pallet::event]
|
||||
@@ -192,6 +194,21 @@ pub mod pallet {
|
||||
///
|
||||
/// The GRANDPA pallet instance must be configured to import headers of relay chain that
|
||||
/// we're interested in.
|
||||
///
|
||||
/// The associated GRANDPA pallet is also used to configure free parachain heads
|
||||
/// submissions. The parachain head submission will be free if:
|
||||
///
|
||||
/// 1) the submission contains exactly one parachain head update that succeeds;
|
||||
///
|
||||
/// 2) the difference between relay chain block numbers, used to prove new parachain head
|
||||
/// and previous best parachain head is larger than the `FreeHeadersInterval`, configured
|
||||
/// at the associated GRANDPA pallet;
|
||||
///
|
||||
/// 3) there are slots for free submissions, remaining at the block. This is also configured
|
||||
/// at the associated GRANDPA pallet using `MaxFreeHeadersPerBlock` parameter.
|
||||
///
|
||||
/// First parachain head submission is also free for the submitted, if free submissions
|
||||
/// are yet accepted to this block.
|
||||
type BridgesGrandpaPalletInstance: 'static;
|
||||
|
||||
/// Name of the original `paras` pallet in the `construct_runtime!()` call at the bridged
|
||||
@@ -335,10 +352,83 @@ pub mod pallet {
|
||||
at_relay_block: (RelayBlockNumber, RelayBlockHash),
|
||||
parachains: Vec<(ParaId, ParaHash)>,
|
||||
parachain_heads_proof: ParaHeadsProof,
|
||||
) -> DispatchResultWithPostInfo {
|
||||
Self::submit_parachain_heads_ex(
|
||||
origin,
|
||||
at_relay_block,
|
||||
parachains,
|
||||
parachain_heads_proof,
|
||||
false,
|
||||
)
|
||||
}
|
||||
|
||||
/// Change `PalletOwner`.
|
||||
///
|
||||
/// May only be called either by root, or by `PalletOwner`.
|
||||
#[pallet::call_index(1)]
|
||||
#[pallet::weight((T::DbWeight::get().reads_writes(1, 1), DispatchClass::Operational))]
|
||||
pub fn set_owner(origin: OriginFor<T>, new_owner: Option<T::AccountId>) -> DispatchResult {
|
||||
<Self as OwnedBridgeModule<_>>::set_owner(origin, new_owner)
|
||||
}
|
||||
|
||||
/// Halt or resume all pallet operations.
|
||||
///
|
||||
/// May only be called either by root, or by `PalletOwner`.
|
||||
#[pallet::call_index(2)]
|
||||
#[pallet::weight((T::DbWeight::get().reads_writes(1, 1), DispatchClass::Operational))]
|
||||
pub fn set_operating_mode(
|
||||
origin: OriginFor<T>,
|
||||
operating_mode: BasicOperatingMode,
|
||||
) -> DispatchResult {
|
||||
<Self as OwnedBridgeModule<_>>::set_operating_mode(origin, operating_mode)
|
||||
}
|
||||
|
||||
/// Submit proof of one or several parachain heads.
|
||||
///
|
||||
/// The proof is supposed to be proof of some `Heads` entries from the
|
||||
/// `polkadot-runtime-parachains::paras` pallet instance, deployed at the bridged chain.
|
||||
/// The proof is supposed to be crafted at the `relay_header_hash` that must already be
|
||||
/// imported by corresponding GRANDPA pallet at this chain.
|
||||
///
|
||||
/// The call fails if:
|
||||
///
|
||||
/// - the pallet is halted;
|
||||
///
|
||||
/// - the relay chain block `at_relay_block` is not imported by the associated bridge
|
||||
/// GRANDPA pallet.
|
||||
///
|
||||
/// The call may succeed, but some heads may not be updated e.g. because pallet knows
|
||||
/// better head or it isn't tracked by the pallet.
|
||||
///
|
||||
/// The `is_free_execution_expected` parameter is not really used inside the call. It is
|
||||
/// used by the transaction extension, which should be registered at the runtime level. If
|
||||
/// this parameter is `true`, the transaction will be treated as invalid, if the call won't
|
||||
/// be executed for free. If transaction extension is not used by the runtime, this
|
||||
/// parameter is not used at all.
|
||||
#[pallet::call_index(3)]
|
||||
#[pallet::weight(WeightInfoOf::<T, I>::submit_parachain_heads_weight(
|
||||
T::DbWeight::get(),
|
||||
parachain_heads_proof,
|
||||
parachains.len() as _,
|
||||
))]
|
||||
pub fn submit_parachain_heads_ex(
|
||||
origin: OriginFor<T>,
|
||||
at_relay_block: (RelayBlockNumber, RelayBlockHash),
|
||||
parachains: Vec<(ParaId, ParaHash)>,
|
||||
parachain_heads_proof: ParaHeadsProof,
|
||||
_is_free_execution_expected: bool,
|
||||
) -> DispatchResultWithPostInfo {
|
||||
Self::ensure_not_halted().map_err(Error::<T, I>::BridgeModule)?;
|
||||
ensure_signed(origin)?;
|
||||
|
||||
let total_parachains = parachains.len();
|
||||
let free_headers_interval =
|
||||
T::FreeHeadersInterval::get().unwrap_or(RelayBlockNumber::MAX);
|
||||
// the pallet allows two kind of free submissions
|
||||
// 1) if distance between all parachain heads is gte than the [`T::FreeHeadersInterval`]
|
||||
// 2) if all heads are the first heads of their parachains
|
||||
let mut free_parachain_heads = 0;
|
||||
|
||||
// we'll need relay chain header to verify that parachains heads are always increasing.
|
||||
let (relay_block_number, relay_block_hash) = at_relay_block;
|
||||
let relay_block = pallet_bridge_grandpa::ImportedHeaders::<
|
||||
@@ -358,6 +448,7 @@ pub mod pallet {
|
||||
parachains.len() as _,
|
||||
);
|
||||
|
||||
let mut is_updated_something = false;
|
||||
let mut storage = GrandpaPalletOf::<T, I>::storage_proof_checker(
|
||||
relay_block_hash,
|
||||
parachain_heads_proof.storage_proof,
|
||||
@@ -414,6 +505,7 @@ pub mod pallet {
|
||||
}
|
||||
|
||||
// convert from parachain head into stored parachain head data
|
||||
let parachain_head_size = parachain_head.0.len();
|
||||
let parachain_head_data =
|
||||
match T::ParaStoredHeaderDataBuilder::try_build(parachain, ¶chain_head) {
|
||||
Some(parachain_head_data) => parachain_head_data,
|
||||
@@ -430,13 +522,30 @@ pub mod pallet {
|
||||
|
||||
let update_result: Result<_, ()> =
|
||||
ParasInfo::<T, I>::try_mutate(parachain, |stored_best_head| {
|
||||
let is_free = parachain_head_size <
|
||||
T::ParaStoredHeaderDataBuilder::max_free_head_size() as usize &&
|
||||
match stored_best_head {
|
||||
Some(ref best_head)
|
||||
if at_relay_block.0.saturating_sub(
|
||||
best_head.best_head_hash.at_relay_block_number,
|
||||
) >= free_headers_interval =>
|
||||
true,
|
||||
Some(_) => false,
|
||||
None => true,
|
||||
};
|
||||
let artifacts = Pallet::<T, I>::update_parachain_head(
|
||||
parachain,
|
||||
stored_best_head.take(),
|
||||
relay_block_number,
|
||||
HeaderId(relay_block_number, relay_block_hash),
|
||||
parachain_head_data,
|
||||
parachain_head_hash,
|
||||
)?;
|
||||
|
||||
is_updated_something = true;
|
||||
if is_free {
|
||||
free_parachain_heads = free_parachain_heads + 1;
|
||||
}
|
||||
|
||||
*stored_best_head = Some(artifacts.best_head);
|
||||
Ok(artifacts.prune_happened)
|
||||
});
|
||||
@@ -467,28 +576,21 @@ pub mod pallet {
|
||||
Error::<T, I>::HeaderChainStorageProof(HeaderChainError::StorageProof(e))
|
||||
})?;
|
||||
|
||||
Ok(PostDispatchInfo { actual_weight: Some(actual_weight), pays_fee: Pays::Yes })
|
||||
}
|
||||
// check if we allow this submission for free
|
||||
let is_free = total_parachains == 1
|
||||
&& free_parachain_heads == total_parachains
|
||||
&& SubmitFinalityProofHelper::<T, T::BridgesGrandpaPalletInstance>::has_free_header_slots();
|
||||
let pays_fee = if is_free {
|
||||
log::trace!(target: LOG_TARGET, "Parachain heads update transaction is free");
|
||||
pallet_bridge_grandpa::on_free_header_imported::<T, T::BridgesGrandpaPalletInstance>(
|
||||
);
|
||||
Pays::No
|
||||
} else {
|
||||
log::trace!(target: LOG_TARGET, "Parachain heads update transaction is paid");
|
||||
Pays::Yes
|
||||
};
|
||||
|
||||
/// Change `PalletOwner`.
|
||||
///
|
||||
/// May only be called either by root, or by `PalletOwner`.
|
||||
#[pallet::call_index(1)]
|
||||
#[pallet::weight((T::DbWeight::get().reads_writes(1, 1), DispatchClass::Operational))]
|
||||
pub fn set_owner(origin: OriginFor<T>, new_owner: Option<T::AccountId>) -> DispatchResult {
|
||||
<Self as OwnedBridgeModule<_>>::set_owner(origin, new_owner)
|
||||
}
|
||||
|
||||
/// Halt or resume all pallet operations.
|
||||
///
|
||||
/// May only be called either by root, or by `PalletOwner`.
|
||||
#[pallet::call_index(2)]
|
||||
#[pallet::weight((T::DbWeight::get().reads_writes(1, 1), DispatchClass::Operational))]
|
||||
pub fn set_operating_mode(
|
||||
origin: OriginFor<T>,
|
||||
operating_mode: BasicOperatingMode,
|
||||
) -> DispatchResult {
|
||||
<Self as OwnedBridgeModule<_>>::set_operating_mode(origin, operating_mode)
|
||||
Ok(PostDispatchInfo { actual_weight: Some(actual_weight), pays_fee })
|
||||
}
|
||||
}
|
||||
|
||||
@@ -545,18 +647,20 @@ pub mod pallet {
|
||||
pub(super) fn update_parachain_head(
|
||||
parachain: ParaId,
|
||||
stored_best_head: Option<ParaInfo>,
|
||||
new_at_relay_block_number: RelayBlockNumber,
|
||||
new_at_relay_block: HeaderId<RelayBlockHash, RelayBlockNumber>,
|
||||
new_head_data: ParaStoredHeaderData,
|
||||
new_head_hash: ParaHash,
|
||||
) -> Result<UpdateParachainHeadArtifacts, ()> {
|
||||
// check if head has been already updated at better relay chain block. Without this
|
||||
// check, we may import heads in random order
|
||||
let update = SubmitParachainHeadsInfo {
|
||||
at_relay_block_number: new_at_relay_block_number,
|
||||
at_relay_block: new_at_relay_block,
|
||||
para_id: parachain,
|
||||
para_head_hash: new_head_hash,
|
||||
// doesn't actually matter here
|
||||
is_free_execution_expected: false,
|
||||
};
|
||||
if SubmitParachainHeadsHelper::<T, I>::is_obsolete(&update) {
|
||||
if SubmitParachainHeadsHelper::<T, I>::check_obsolete(&update).is_err() {
|
||||
Self::deposit_event(Event::RejectedObsoleteParachainHead {
|
||||
parachain,
|
||||
parachain_head_hash: new_head_hash,
|
||||
@@ -596,7 +700,7 @@ pub mod pallet {
|
||||
ImportedParaHashes::<T, I>::try_get(parachain, next_imported_hash_position);
|
||||
let updated_best_para_head = ParaInfo {
|
||||
best_head_hash: BestParaHeadHash {
|
||||
at_relay_block_number: new_at_relay_block_number,
|
||||
at_relay_block_number: new_at_relay_block.0,
|
||||
head_hash: new_head_hash,
|
||||
},
|
||||
next_imported_hash_position: (next_imported_hash_position + 1) %
|
||||
@@ -610,9 +714,10 @@ pub mod pallet {
|
||||
ImportedParaHeads::<T, I>::insert(parachain, new_head_hash, updated_head_data);
|
||||
log::trace!(
|
||||
target: LOG_TARGET,
|
||||
"Updated head of parachain {:?} to {}",
|
||||
"Updated head of parachain {:?} to {} at relay block {}",
|
||||
parachain,
|
||||
new_head_hash,
|
||||
new_at_relay_block.0,
|
||||
);
|
||||
|
||||
// remove old head
|
||||
@@ -696,14 +801,28 @@ impl<T: Config<I>, I: 'static, C: Parachain<Hash = ParaHash>> HeaderChain<C>
|
||||
pub fn initialize_for_benchmarks<T: Config<I>, I: 'static, PC: Parachain<Hash = ParaHash>>(
|
||||
header: HeaderOf<PC>,
|
||||
) {
|
||||
use bp_runtime::HeaderIdProvider;
|
||||
use sp_runtime::traits::Header;
|
||||
|
||||
let relay_head =
|
||||
pallet_bridge_grandpa::BridgedHeader::<T, T::BridgesGrandpaPalletInstance>::new(
|
||||
0,
|
||||
Default::default(),
|
||||
Default::default(),
|
||||
Default::default(),
|
||||
Default::default(),
|
||||
);
|
||||
let parachain = ParaId(PC::PARACHAIN_ID);
|
||||
let parachain_head = ParaHead(header.encode());
|
||||
let updated_head_data = T::ParaStoredHeaderDataBuilder::try_build(parachain, ¶chain_head)
|
||||
.expect("failed to build stored parachain head in benchmarks");
|
||||
pallet_bridge_grandpa::initialize_for_benchmarks::<T, T::BridgesGrandpaPalletInstance>(
|
||||
relay_head.clone(),
|
||||
);
|
||||
Pallet::<T, I>::update_parachain_head(
|
||||
parachain,
|
||||
None,
|
||||
0,
|
||||
relay_head.id(),
|
||||
updated_head_data,
|
||||
parachain_head.hash(),
|
||||
)
|
||||
@@ -714,9 +833,9 @@ pub fn initialize_for_benchmarks<T: Config<I>, I: 'static, PC: Parachain<Hash =
|
||||
pub(crate) mod tests {
|
||||
use super::*;
|
||||
use crate::mock::{
|
||||
run_test, test_relay_header, BigParachainHeader, RegularParachainHasher,
|
||||
RegularParachainHeader, RelayBlockHeader, RuntimeEvent as TestEvent, RuntimeOrigin,
|
||||
TestRuntime, UNTRACKED_PARACHAIN_ID,
|
||||
run_test, test_relay_header, BigParachain, BigParachainHeader, FreeHeadersInterval,
|
||||
RegularParachainHasher, RegularParachainHeader, RelayBlockHeader,
|
||||
RuntimeEvent as TestEvent, RuntimeOrigin, TestRuntime, UNTRACKED_PARACHAIN_ID,
|
||||
};
|
||||
use bp_test_utils::prepare_parachain_heads_proof;
|
||||
use codec::Encode;
|
||||
@@ -736,8 +855,9 @@ pub(crate) mod tests {
|
||||
use frame_support::{
|
||||
assert_noop, assert_ok,
|
||||
dispatch::DispatchResultWithPostInfo,
|
||||
pallet_prelude::Pays,
|
||||
storage::generator::{StorageDoubleMap, StorageMap},
|
||||
traits::{Get, OnInitialize},
|
||||
traits::Get,
|
||||
weights::Weight,
|
||||
};
|
||||
use frame_system::{EventRecord, Pallet as System, Phase};
|
||||
@@ -749,6 +869,7 @@ pub(crate) mod tests {
|
||||
type DbWeight = <TestRuntime as frame_system::Config>::DbWeight;
|
||||
|
||||
pub(crate) fn initialize(state_root: RelayBlockHash) -> RelayBlockHash {
|
||||
pallet_bridge_grandpa::FreeHeadersRemaining::<TestRuntime, BridgesGrandpaPalletInstance>::set(Some(100));
|
||||
pallet_bridge_grandpa::Pallet::<TestRuntime, BridgesGrandpaPalletInstance>::initialize(
|
||||
RuntimeOrigin::root(),
|
||||
bp_header_chain::InitializationData {
|
||||
@@ -770,10 +891,6 @@ pub(crate) mod tests {
|
||||
num: RelayBlockNumber,
|
||||
state_root: RelayBlockHash,
|
||||
) -> (ParaHash, GrandpaJustification<RelayBlockHeader>) {
|
||||
pallet_bridge_grandpa::Pallet::<TestRuntime, BridgesGrandpaPalletInstance>::on_initialize(
|
||||
0,
|
||||
);
|
||||
|
||||
let header = test_relay_header(num, state_root);
|
||||
let hash = header.hash();
|
||||
let justification = make_default_justification(&header);
|
||||
@@ -783,6 +900,7 @@ pub(crate) mod tests {
|
||||
Box::new(header),
|
||||
justification.clone(),
|
||||
TEST_GRANDPA_SET_ID,
|
||||
false,
|
||||
)
|
||||
);
|
||||
|
||||
@@ -908,7 +1026,7 @@ pub(crate) mod tests {
|
||||
run_test(|| {
|
||||
initialize(state_root);
|
||||
|
||||
// we're trying to update heads of parachains 1, 2 and 3
|
||||
// we're trying to update heads of parachains 1 and 3
|
||||
let expected_weight =
|
||||
WeightInfo::submit_parachain_heads_weight(DbWeight::get(), &proof, 2);
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
@@ -918,9 +1036,10 @@ pub(crate) mod tests {
|
||||
proof,
|
||||
);
|
||||
assert_ok!(result);
|
||||
assert_eq!(result.expect("checked above").pays_fee, Pays::Yes);
|
||||
assert_eq!(result.expect("checked above").actual_weight, Some(expected_weight));
|
||||
|
||||
// but only 1 and 2 are updated, because proof is missing head of parachain#2
|
||||
// 1 and 3 are updated, because proof is missing head of parachain#2
|
||||
assert_eq!(ParasInfo::<TestRuntime>::get(ParaId(1)), Some(initial_best_head(1)));
|
||||
assert_eq!(ParasInfo::<TestRuntime>::get(ParaId(2)), None);
|
||||
assert_eq!(
|
||||
@@ -989,7 +1108,9 @@ pub(crate) mod tests {
|
||||
run_test(|| {
|
||||
// start with relay block #0 and import head#5 of parachain#1
|
||||
initialize(state_root_5);
|
||||
assert_ok!(import_parachain_1_head(0, state_root_5, parachains_5, proof_5));
|
||||
let result = import_parachain_1_head(0, state_root_5, parachains_5, proof_5);
|
||||
// first parachain head is imported for free
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::No);
|
||||
assert_eq!(
|
||||
ParasInfo::<TestRuntime>::get(ParaId(1)),
|
||||
Some(ParaInfo {
|
||||
@@ -1024,7 +1145,9 @@ pub(crate) mod tests {
|
||||
|
||||
// import head#10 of parachain#1 at relay block #1
|
||||
let (relay_1_hash, justification) = proceed(1, state_root_10);
|
||||
assert_ok!(import_parachain_1_head(1, state_root_10, parachains_10, proof_10));
|
||||
let result = import_parachain_1_head(1, state_root_10, parachains_10, proof_10);
|
||||
// second parachain head is imported for fee
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
assert_eq!(
|
||||
ParasInfo::<TestRuntime>::get(ParaId(1)),
|
||||
Some(ParaInfo {
|
||||
@@ -1647,4 +1770,143 @@ pub(crate) mod tests {
|
||||
);
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn may_be_free_for_submitting_filtered_heads() {
|
||||
run_test(|| {
|
||||
let (state_root, proof, parachains) =
|
||||
prepare_parachain_heads_proof::<RegularParachainHeader>(vec![(2, head_data(2, 5))]);
|
||||
// start with relay block #0 and import head#5 of parachain#2
|
||||
initialize(state_root);
|
||||
// first submission is free
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(0, test_relay_header(0, state_root).hash()),
|
||||
parachains.clone(),
|
||||
proof.clone(),
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::No);
|
||||
// next submission is NOT free, because we haven't updated anything
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(0, test_relay_header(0, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
// then we submit new head, proved at relay block `FreeHeadersInterval - 1` => Pays::Yes
|
||||
let (state_root, proof, parachains) = prepare_parachain_heads_proof::<
|
||||
RegularParachainHeader,
|
||||
>(vec![(2, head_data(2, 50))]);
|
||||
let relay_block_number = FreeHeadersInterval::get() - 1;
|
||||
proceed(relay_block_number, state_root);
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(relay_block_number, test_relay_header(relay_block_number, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
// then we submit new head, proved after `FreeHeadersInterval` => Pays::No
|
||||
let (state_root, proof, parachains) = prepare_parachain_heads_proof::<
|
||||
RegularParachainHeader,
|
||||
>(vec![(2, head_data(2, 100))]);
|
||||
let relay_block_number = relay_block_number + FreeHeadersInterval::get();
|
||||
proceed(relay_block_number, state_root);
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(relay_block_number, test_relay_header(relay_block_number, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::No);
|
||||
// then we submit new BIG head, proved after `FreeHeadersInterval` => Pays::Yes
|
||||
// then we submit new head, proved after `FreeHeadersInterval` => Pays::No
|
||||
let mut large_head = head_data(2, 100);
|
||||
large_head.0.extend(&[42u8; BigParachain::MAX_HEADER_SIZE as _]);
|
||||
let (state_root, proof, parachains) =
|
||||
prepare_parachain_heads_proof::<RegularParachainHeader>(vec![(2, large_head)]);
|
||||
let relay_block_number = relay_block_number + FreeHeadersInterval::get();
|
||||
proceed(relay_block_number, state_root);
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(relay_block_number, test_relay_header(relay_block_number, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
})
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn grandpa_and_parachain_pallets_share_free_headers_counter() {
|
||||
run_test(|| {
|
||||
initialize(Default::default());
|
||||
// set free headers limit to `4`
|
||||
let mut free_headers_remaining = 4;
|
||||
pallet_bridge_grandpa::FreeHeadersRemaining::<TestRuntime, BridgesGrandpaPalletInstance>::set(
|
||||
Some(free_headers_remaining),
|
||||
);
|
||||
// import free GRANDPA and parachain headers
|
||||
let mut relay_block_number = 0;
|
||||
for i in 0..2 {
|
||||
// import free GRANDPA header
|
||||
let (state_root, proof, parachains) = prepare_parachain_heads_proof::<
|
||||
RegularParachainHeader,
|
||||
>(vec![(2, head_data(2, 5 + i))]);
|
||||
relay_block_number = relay_block_number + FreeHeadersInterval::get();
|
||||
proceed(relay_block_number, state_root);
|
||||
assert_eq!(
|
||||
pallet_bridge_grandpa::FreeHeadersRemaining::<
|
||||
TestRuntime,
|
||||
BridgesGrandpaPalletInstance,
|
||||
>::get(),
|
||||
Some(free_headers_remaining - 1),
|
||||
);
|
||||
free_headers_remaining = free_headers_remaining - 1;
|
||||
// import free parachain header
|
||||
assert_ok!(Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(relay_block_number, test_relay_header(relay_block_number, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
),);
|
||||
assert_eq!(
|
||||
pallet_bridge_grandpa::FreeHeadersRemaining::<
|
||||
TestRuntime,
|
||||
BridgesGrandpaPalletInstance,
|
||||
>::get(),
|
||||
Some(free_headers_remaining - 1),
|
||||
);
|
||||
free_headers_remaining = free_headers_remaining - 1;
|
||||
}
|
||||
// try to import free GRANDPA header => non-free execution
|
||||
let (state_root, proof, parachains) =
|
||||
prepare_parachain_heads_proof::<RegularParachainHeader>(vec![(2, head_data(2, 7))]);
|
||||
relay_block_number = relay_block_number + FreeHeadersInterval::get();
|
||||
let result = pallet_bridge_grandpa::Pallet::<TestRuntime, BridgesGrandpaPalletInstance>::submit_finality_proof_ex(
|
||||
RuntimeOrigin::signed(1),
|
||||
Box::new(test_relay_header(relay_block_number, state_root)),
|
||||
make_default_justification(&test_relay_header(relay_block_number, state_root)),
|
||||
TEST_GRANDPA_SET_ID,
|
||||
false,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
// try to import free parachain header => non-free execution
|
||||
let result = Pallet::<TestRuntime>::submit_parachain_heads(
|
||||
RuntimeOrigin::signed(1),
|
||||
(relay_block_number, test_relay_header(relay_block_number, state_root).hash()),
|
||||
parachains,
|
||||
proof,
|
||||
);
|
||||
assert_eq!(result.unwrap().pays_fee, Pays::Yes);
|
||||
assert_eq!(
|
||||
pallet_bridge_grandpa::FreeHeadersRemaining::<
|
||||
TestRuntime,
|
||||
BridgesGrandpaPalletInstance,
|
||||
>::get(),
|
||||
Some(0),
|
||||
);
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
@@ -70,6 +70,7 @@ impl Chain for Parachain1 {
|
||||
|
||||
impl Parachain for Parachain1 {
|
||||
const PARACHAIN_ID: u32 = 1;
|
||||
const MAX_HEADER_SIZE: u32 = 1_024;
|
||||
}
|
||||
|
||||
pub struct Parachain2;
|
||||
@@ -96,6 +97,7 @@ impl Chain for Parachain2 {
|
||||
|
||||
impl Parachain for Parachain2 {
|
||||
const PARACHAIN_ID: u32 = 2;
|
||||
const MAX_HEADER_SIZE: u32 = 1_024;
|
||||
}
|
||||
|
||||
pub struct Parachain3;
|
||||
@@ -122,6 +124,7 @@ impl Chain for Parachain3 {
|
||||
|
||||
impl Parachain for Parachain3 {
|
||||
const PARACHAIN_ID: u32 = 3;
|
||||
const MAX_HEADER_SIZE: u32 = 1_024;
|
||||
}
|
||||
|
||||
// this parachain is using u128 as block number and stored head data size exceeds limit
|
||||
@@ -149,6 +152,7 @@ impl Chain for BigParachain {
|
||||
|
||||
impl Parachain for BigParachain {
|
||||
const PARACHAIN_ID: u32 = 4;
|
||||
const MAX_HEADER_SIZE: u32 = 2_048;
|
||||
}
|
||||
|
||||
construct_runtime! {
|
||||
@@ -168,12 +172,14 @@ impl frame_system::Config for TestRuntime {
|
||||
|
||||
parameter_types! {
|
||||
pub const HeadersToKeep: u32 = 5;
|
||||
pub const FreeHeadersInterval: u32 = 15;
|
||||
}
|
||||
|
||||
impl pallet_bridge_grandpa::Config<pallet_bridge_grandpa::Instance1> for TestRuntime {
|
||||
type RuntimeEvent = RuntimeEvent;
|
||||
type BridgedChain = TestBridgedChain;
|
||||
type MaxFreeMandatoryHeadersPerBlock = ConstU32<2>;
|
||||
type MaxFreeHeadersPerBlock = ConstU32<2>;
|
||||
type FreeHeadersInterval = FreeHeadersInterval;
|
||||
type HeadersToKeep = HeadersToKeep;
|
||||
type WeightInfo = ();
|
||||
}
|
||||
@@ -181,7 +187,8 @@ impl pallet_bridge_grandpa::Config<pallet_bridge_grandpa::Instance1> for TestRun
|
||||
impl pallet_bridge_grandpa::Config<pallet_bridge_grandpa::Instance2> for TestRuntime {
|
||||
type RuntimeEvent = RuntimeEvent;
|
||||
type BridgedChain = TestBridgedChain;
|
||||
type MaxFreeMandatoryHeadersPerBlock = ConstU32<2>;
|
||||
type MaxFreeHeadersPerBlock = ConstU32<2>;
|
||||
type FreeHeadersInterval = FreeHeadersInterval;
|
||||
type HeadersToKeep = HeadersToKeep;
|
||||
type WeightInfo = ();
|
||||
}
|
||||
|
||||
@@ -36,6 +36,20 @@ pub const EXTRA_STORAGE_PROOF_SIZE: u32 = 1024;
|
||||
|
||||
/// Extended weight info.
|
||||
pub trait WeightInfoExt: WeightInfo {
|
||||
// Our configuration assumes that the runtime has special signed extensions used to:
|
||||
//
|
||||
// 1) boost priority of `submit_parachain_heads` transactions;
|
||||
//
|
||||
// 2) slash relayer if he submits an invalid transaction.
|
||||
//
|
||||
// We read and update storage values of other pallets (`pallet-bridge-relayers` and
|
||||
// balances/assets pallet). So we need to add this weight to the weight of our call.
|
||||
// Hence two following methods.
|
||||
|
||||
/// Extra weight that is added to the `submit_finality_proof` call weight by signed extensions
|
||||
/// that are declared at runtime level.
|
||||
fn submit_parachain_heads_overhead_from_runtime() -> Weight;
|
||||
|
||||
/// Storage proof overhead, that is included in every storage proof.
|
||||
///
|
||||
/// The relayer would pay some extra fee for additional proof bytes, since they mean
|
||||
@@ -65,7 +79,10 @@ pub trait WeightInfoExt: WeightInfo {
|
||||
let pruning_weight =
|
||||
Self::parachain_head_pruning_weight(db_weight).saturating_mul(parachains_count as u64);
|
||||
|
||||
base_weight.saturating_add(proof_size_overhead).saturating_add(pruning_weight)
|
||||
base_weight
|
||||
.saturating_add(proof_size_overhead)
|
||||
.saturating_add(pruning_weight)
|
||||
.saturating_add(Self::submit_parachain_heads_overhead_from_runtime())
|
||||
}
|
||||
|
||||
/// Returns weight of single parachain head storage update.
|
||||
@@ -95,12 +112,20 @@ pub trait WeightInfoExt: WeightInfo {
|
||||
}
|
||||
|
||||
impl WeightInfoExt for () {
|
||||
fn submit_parachain_heads_overhead_from_runtime() -> Weight {
|
||||
Weight::zero()
|
||||
}
|
||||
|
||||
fn expected_extra_storage_proof_size() -> u32 {
|
||||
EXTRA_STORAGE_PROOF_SIZE
|
||||
}
|
||||
}
|
||||
|
||||
impl<T: frame_system::Config> WeightInfoExt for BridgeWeight<T> {
|
||||
fn submit_parachain_heads_overhead_from_runtime() -> Weight {
|
||||
Weight::zero()
|
||||
}
|
||||
|
||||
fn expected_extra_storage_proof_size() -> u32 {
|
||||
EXTRA_STORAGE_PROOF_SIZE
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user