#![cfg_attr(not(feature = "std"), no_std)]
#![forbid(unsafe_code)]
#![warn(rust_2018_idioms)]
#![feature(let_chains, variant_count, if_let_guard)]
#[cfg(feature = "runtime-benchmarks")]
mod benchmarking;
mod fees;
mod messages;
#[cfg(test)]
mod mock;
#[cfg(test)]
mod tests;
pub mod weights;
#[cfg(not(feature = "std"))]
extern crate alloc;
use codec::{Decode, Encode};
use frame_support::traits::fungible::{Inspect, InspectHold};
use frame_system::pallet_prelude::BlockNumberFor;
pub use pallet::*;
use scale_info::TypeInfo;
use sp_core::U256;
use sp_domains::{DomainAllowlistUpdates, DomainId};
use sp_messenger::messages::{
ChainId, Channel, ChannelId, ChannelState, CrossDomainMessage, FeeModel, MessageId, Nonce,
};
use sp_runtime::traits::{Extrinsic, Hash};
use sp_runtime::DispatchError;
pub(crate) mod verification_errors {
pub(crate) const INVALID_CHANNEL: u8 = 200;
pub(crate) const INVALID_NONCE: u8 = 201;
pub(crate) const NONCE_OVERFLOW: u8 = 202;
}
#[derive(Debug, Encode, Decode, Clone, Eq, PartialEq, TypeInfo, Copy)]
pub enum OutboxMessageResult {
Ok,
Err(DispatchError),
}
pub(crate) type StateRootOf<T> = <<T as frame_system::Config>::Hashing as Hash>::Output;
pub(crate) type BalanceOf<T> =
<<T as Config>::Currency as Inspect<<T as frame_system::Config>::AccountId>>::Balance;
pub(crate) type FungibleHoldId<T> =
<<T as Config>::Currency as InspectHold<<T as frame_system::Config>::AccountId>>::Reason;
#[derive(Debug)]
pub struct ValidatedRelayMessage {
msg_nonce: Nonce,
dst_chain_id: ChainId,
channel_id: ChannelId,
next_nonce: Nonce,
}
#[derive(Debug, Encode, Decode, Clone, Eq, PartialEq, TypeInfo, Copy)]
pub enum ChainAllowlistUpdate {
Add(ChainId),
Remove(ChainId),
}
impl ChainAllowlistUpdate {
fn chain_id(&self) -> ChainId {
match self {
ChainAllowlistUpdate::Add(chain_id) => *chain_id,
ChainAllowlistUpdate::Remove(chain_id) => *chain_id,
}
}
}
#[derive(Debug, Encode, Decode, Clone, Eq, PartialEq, TypeInfo, Copy)]
pub(crate) enum CloseChannelBy<AccountId> {
Owner(AccountId),
Sudo,
}
#[derive(Default, Debug, Encode, Decode, Clone, Eq, PartialEq, TypeInfo, Copy)]
pub struct InitiateChannelParams {
pub max_outgoing_messages: u32,
}
pub trait HoldIdentifier<T: Config> {
fn messenger_channel() -> FungibleHoldId<T>;
}
#[frame_support::pallet]
mod pallet {
use crate::weights::WeightInfo;
use crate::{
BalanceOf, ChainAllowlistUpdate, Channel, ChannelId, ChannelState, CloseChannelBy,
FeeModel, HoldIdentifier, InitiateChannelParams, Nonce, OutboxMessageResult, StateRootOf,
ValidatedRelayMessage, U256,
};
#[cfg(not(feature = "std"))]
use alloc::boxed::Box;
#[cfg(not(feature = "std"))]
use alloc::collections::BTreeSet;
#[cfg(not(feature = "std"))]
use alloc::vec::Vec;
use core::cmp::Ordering;
use frame_support::ensure;
use frame_support::pallet_prelude::*;
use frame_support::traits::fungible::{Inspect, InspectHold, Mutate, MutateHold};
use frame_support::traits::tokens::{Fortitude, Precision, Preservation};
use frame_support::weights::WeightToFee;
use frame_system::pallet_prelude::*;
use sp_core::storage::StorageKey;
use sp_domains::proof_provider_and_verifier::{StorageProofVerifier, VerificationError};
use sp_domains::{DomainAllowlistUpdates, DomainId, DomainOwner};
use sp_messenger::endpoint::{Endpoint, EndpointHandler, EndpointRequest, Sender};
use sp_messenger::messages::{
ChainId, ChannelOpenParams, CrossDomainMessage, Message, MessageId, MessageKey,
MessageWeightTag, Payload, ProtocolMessageRequest, RequestResponse, VersionedPayload,
};
use sp_messenger::{
ChannelNonce, DomainRegistration, InherentError, InherentType, OnXDMRewards, StorageKeys,
INHERENT_IDENTIFIER,
};
use sp_runtime::traits::Zero;
use sp_runtime::{ArithmeticError, Perbill, Saturating};
use sp_subspace_mmr::MmrProofVerifier;
#[cfg(feature = "std")]
use std::collections::BTreeSet;
#[pallet::config]
pub trait Config: frame_system::Config {
type RuntimeEvent: From<Event<Self>> + IsType<<Self as frame_system::Config>::RuntimeEvent>;
type SelfChainId: Get<ChainId>;
fn get_endpoint_handler(endpoint: &Endpoint)
-> Option<Box<dyn EndpointHandler<MessageId>>>;
type Currency: Mutate<Self::AccountId>
+ InspectHold<Self::AccountId>
+ MutateHold<Self::AccountId>;
type WeightInfo: WeightInfo;
type WeightToFee: WeightToFee<Balance = BalanceOf<Self>>;
type OnXDMRewards: OnXDMRewards<BalanceOf<Self>>;
type MmrHash: Parameter + Member + Default + Clone;
type MmrProofVerifier: MmrProofVerifier<
Self::MmrHash,
BlockNumberFor<Self>,
StateRootOf<Self>,
>;
type StorageKeys: StorageKeys;
type DomainOwner: DomainOwner<Self::AccountId>;
type HoldIdentifier: HoldIdentifier<Self>;
#[pallet::constant]
type ChannelReserveFee: Get<BalanceOf<Self>>;
#[pallet::constant]
type ChannelInitReservePortion: Get<Perbill>;
type DomainRegistration: DomainRegistration;
type ChannelFeeModel: Get<FeeModel<BalanceOf<Self>>>;
}
#[pallet::pallet]
#[pallet::without_storage_info]
pub struct Pallet<T>(_);
#[pallet::storage]
#[pallet::getter(fn next_channel_id)]
pub(super) type NextChannelId<T: Config> =
StorageMap<_, Identity, ChainId, ChannelId, ValueQuery>;
#[pallet::storage]
#[pallet::getter(fn channels)]
pub(super) type Channels<T: Config> = StorageDoubleMap<
_,
Identity,
ChainId,
Identity,
ChannelId,
Channel<BalanceOf<T>, T::AccountId>,
OptionQuery,
>;
#[pallet::storage]
#[pallet::getter(fn inbox)]
pub(super) type Inbox<T: Config> = StorageValue<_, Message<BalanceOf<T>>, OptionQuery>;
#[pallet::storage]
#[pallet::getter(fn inbox_fees)]
pub(super) type InboxFee<T: Config> =
StorageMap<_, Identity, (ChainId, MessageId), BalanceOf<T>, OptionQuery>;
#[pallet::storage]
#[pallet::getter(fn outbox_fees)]
pub(super) type OutboxFee<T: Config> =
StorageMap<_, Identity, (ChainId, MessageId), BalanceOf<T>, OptionQuery>;
#[pallet::storage]
#[pallet::getter(fn inbox_responses)]
pub(super) type InboxResponses<T: Config> = CountedStorageMap<
_,
Identity,
(ChainId, ChannelId, Nonce),
Message<BalanceOf<T>>,
OptionQuery,
>;
#[pallet::storage]
#[pallet::getter(fn outbox)]
pub(super) type Outbox<T: Config> = CountedStorageMap<
_,
Identity,
(ChainId, ChannelId, Nonce),
Message<BalanceOf<T>>,
OptionQuery,
>;
#[pallet::storage]
#[pallet::getter(fn outbox_responses)]
pub(super) type OutboxResponses<T: Config> =
StorageValue<_, Message<BalanceOf<T>>, OptionQuery>;
#[pallet::storage]
#[pallet::getter(fn message_weight_tags)]
pub(super) type MessageWeightTags<T: Config> =
StorageValue<_, crate::messages::MessageWeightTags, OptionQuery>;
#[pallet::storage]
#[pallet::getter(fn chain_allowlist)]
pub(super) type ChainAllowlist<T: Config> = StorageValue<_, BTreeSet<ChainId>, ValueQuery>;
#[pallet::storage]
#[pallet::getter(fn domain_chain_allowlist_updates)]
pub(super) type DomainChainAllowlistUpdate<T: Config> =
StorageMap<_, Identity, DomainId, DomainAllowlistUpdates, OptionQuery>;
#[pallet::storage]
pub(super) type UpdatedChannels<T: Config> =
StorageValue<_, BTreeSet<(ChainId, ChannelId)>, ValueQuery>;
#[pallet::event]
#[pallet::generate_deposit(pub (super) fn deposit_event)]
pub enum Event<T: Config> {
ChannelInitiated {
chain_id: ChainId,
channel_id: ChannelId,
},
ChannelClosed {
chain_id: ChainId,
channel_id: ChannelId,
},
ChannelOpen {
chain_id: ChainId,
channel_id: ChannelId,
},
OutboxMessage {
chain_id: ChainId,
channel_id: ChannelId,
nonce: Nonce,
},
OutboxMessageResponse {
chain_id: ChainId,
channel_id: ChannelId,
nonce: Nonce,
},
OutboxMessageResult {
chain_id: ChainId,
channel_id: ChannelId,
nonce: Nonce,
result: OutboxMessageResult,
},
InboxMessage {
chain_id: ChainId,
channel_id: ChannelId,
nonce: Nonce,
},
InboxMessageResponse {
chain_id: ChainId,
channel_id: ChannelId,
nonce: Nonce,
},
}
#[pallet::hooks]
impl<T: Config> Hooks<BlockNumberFor<T>> for Pallet<T> {
fn on_initialize(_n: BlockNumberFor<T>) -> Weight {
UpdatedChannels::<T>::take();
T::DbWeight::get().writes(1)
}
}
#[pallet::validate_unsigned]
impl<T: Config> ValidateUnsigned for Pallet<T> {
type Call = Call<T>;
fn pre_dispatch(call: &Self::Call) -> Result<(), TransactionValidityError> {
match call {
Call::relay_message { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::verify_proof_and_extract_leaf(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
Self::validate_relay_message(xdm, true, consensus_state_root)?;
Ok(())
}
Call::relay_message_response { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::verify_proof_and_extract_leaf(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
Self::validate_relay_message_response(xdm, true, consensus_state_root)?;
Ok(())
}
Call::update_domain_allowlist { .. } => Ok(()),
_ => Err(InvalidTransaction::Call.into()),
}
}
fn validate_unsigned(_source: TransactionSource, call: &Self::Call) -> TransactionValidity {
match call {
Call::relay_message { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::verify_proof_and_extract_leaf(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
let ValidatedRelayMessage {
msg_nonce,
dst_chain_id,
channel_id,
next_nonce,
} = Self::validate_relay_message(xdm, false, consensus_state_root)?;
let mut valid_tx_builder = ValidTransaction::with_tag_prefix("MessengerInbox");
if msg_nonce > next_nonce {
valid_tx_builder = valid_tx_builder.and_requires((
dst_chain_id,
channel_id,
msg_nonce - Nonce::one(),
));
};
valid_tx_builder
.priority(1)
.longevity(TransactionLongevity::MAX)
.and_provides((dst_chain_id, channel_id, msg_nonce))
.propagate(true)
.build()
}
Call::relay_message_response { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::verify_proof_and_extract_leaf(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
let ValidatedRelayMessage {
msg_nonce,
dst_chain_id,
channel_id,
next_nonce,
} = Self::validate_relay_message_response(xdm, false, consensus_state_root)?;
let mut valid_tx_builder =
ValidTransaction::with_tag_prefix("MessengerOutboxResponse");
if msg_nonce > next_nonce {
valid_tx_builder = valid_tx_builder.and_requires((
dst_chain_id,
channel_id,
msg_nonce - Nonce::one(),
));
};
valid_tx_builder
.priority(1)
.longevity(TransactionLongevity::MAX)
.and_provides((dst_chain_id, channel_id, msg_nonce))
.propagate(true)
.build()
}
_ => InvalidTransaction::Call.into(),
}
}
}
#[pallet::error]
pub enum Error<T> {
InvalidChain,
MissingChannel,
InvalidChannelState,
NoOpenChannel,
NoMessageHandler,
OutboxFull,
InvalidMessagePayload,
InvalidMessageDestination,
MessageVerification(VerificationError),
MissingMessage,
WeightTagNotMatch,
BalanceOverflow,
InvalidAllowedChain,
OperationNotAllowed,
NotDomainOwner,
ChainNotAllowed,
InsufficientBalance,
BalanceHold,
ChannelOwner,
BalanceUnlock,
InvalidChannelReserveFee,
}
#[pallet::call]
impl<T: Config> Pallet<T> {
#[pallet::call_index(0)]
#[pallet::weight(T::WeightInfo::initiate_channel())]
pub fn initiate_channel(
origin: OriginFor<T>,
dst_chain_id: ChainId,
params: InitiateChannelParams,
) -> DispatchResult {
let owner = ensure_signed(origin)?;
let hold_id = T::HoldIdentifier::messenger_channel();
let amount = T::ChannelReserveFee::get();
ensure!(
T::Currency::reducible_balance(&owner, Preservation::Preserve, Fortitude::Polite)
>= amount,
Error::<T>::InsufficientBalance
);
T::Currency::hold(&hold_id, &owner, amount).map_err(|_| Error::<T>::BalanceHold)?;
let channel_open_params = ChannelOpenParams {
max_outgoing_messages: params.max_outgoing_messages,
fee_model: T::ChannelFeeModel::get(),
};
let channel_id = Self::do_init_channel(
dst_chain_id,
channel_open_params,
Some(owner.clone()),
true,
amount,
)?;
Self::new_outbox_message(
T::SelfChainId::get(),
dst_chain_id,
channel_id,
VersionedPayload::V0(Payload::Protocol(RequestResponse::Request(
ProtocolMessageRequest::ChannelOpen(channel_open_params),
))),
)?;
Ok(())
}
#[pallet::call_index(1)]
#[pallet::weight(T::WeightInfo::close_channel())]
pub fn close_channel(
origin: OriginFor<T>,
chain_id: ChainId,
channel_id: ChannelId,
) -> DispatchResult {
let close_channel_by = match ensure_signed_or_root(origin)? {
Some(owner) => CloseChannelBy::Owner(owner),
None => CloseChannelBy::Sudo,
};
Self::do_close_channel(chain_id, channel_id, close_channel_by)?;
Self::new_outbox_message(
T::SelfChainId::get(),
chain_id,
channel_id,
VersionedPayload::V0(Payload::Protocol(RequestResponse::Request(
ProtocolMessageRequest::ChannelClose,
))),
)?;
Ok(())
}
#[pallet::call_index(2)]
#[pallet::weight(T::WeightInfo::relay_message().saturating_add(Pallet::< T >::message_weight(& msg.weight_tag)))]
pub fn relay_message(
origin: OriginFor<T>,
msg: CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
) -> DispatchResult {
ensure_none(origin)?;
let inbox_msg = Inbox::<T>::take().ok_or(Error::<T>::MissingMessage)?;
Self::process_inbox_messages(inbox_msg, msg.weight_tag)?;
Ok(())
}
#[pallet::call_index(3)]
#[pallet::weight(T::WeightInfo::relay_message_response().saturating_add(Pallet::< T >::message_weight(& msg.weight_tag)))]
pub fn relay_message_response(
origin: OriginFor<T>,
msg: CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
) -> DispatchResult {
ensure_none(origin)?;
let outbox_resp_msg = OutboxResponses::<T>::take().ok_or(Error::<T>::MissingMessage)?;
Self::process_outbox_message_responses(outbox_resp_msg, msg.weight_tag)?;
Ok(())
}
#[pallet::call_index(4)]
#[pallet::weight(<T as frame_system::Config>::DbWeight::get().reads_writes(1, 1))]
pub fn update_consensus_chain_allowlist(
origin: OriginFor<T>,
update: ChainAllowlistUpdate,
) -> DispatchResult {
ensure_root(origin)?;
ensure!(
T::SelfChainId::get().is_consensus_chain(),
Error::<T>::OperationNotAllowed
);
ensure!(
update.chain_id() != T::SelfChainId::get(),
Error::<T>::InvalidAllowedChain
);
if let ChainAllowlistUpdate::Add(ChainId::Domain(domain_id)) = update {
ensure!(
T::DomainRegistration::is_domain_registered(domain_id),
Error::<T>::InvalidChain
);
}
ChainAllowlist::<T>::mutate(|list| match update {
ChainAllowlistUpdate::Add(chain_id) => list.insert(chain_id),
ChainAllowlistUpdate::Remove(chain_id) => list.remove(&chain_id),
});
Ok(())
}
#[pallet::call_index(5)]
#[pallet::weight(<T as frame_system::Config>::DbWeight::get().reads_writes(1, 1))]
pub fn initiate_domain_update_chain_allowlist(
origin: OriginFor<T>,
domain_id: DomainId,
update: ChainAllowlistUpdate,
) -> DispatchResult {
let domain_owner = ensure_signed(origin)?;
ensure!(
T::DomainOwner::is_domain_owner(domain_id, domain_owner),
Error::<T>::NotDomainOwner
);
ensure!(
T::SelfChainId::get().is_consensus_chain(),
Error::<T>::OperationNotAllowed
);
if let Some(dst_domain_id) = update.chain_id().maybe_domain_chain() {
ensure!(dst_domain_id != domain_id, Error::<T>::InvalidAllowedChain);
}
if let ChainAllowlistUpdate::Add(ChainId::Domain(domain_id)) = update {
ensure!(
T::DomainRegistration::is_domain_registered(domain_id),
Error::<T>::InvalidChain
);
}
DomainChainAllowlistUpdate::<T>::mutate(domain_id, |maybe_domain_updates| {
let mut domain_updates = maybe_domain_updates.take().unwrap_or_default();
match update {
ChainAllowlistUpdate::Add(chain_id) => {
domain_updates.remove_chains.remove(&chain_id);
domain_updates.allow_chains.insert(chain_id);
}
ChainAllowlistUpdate::Remove(chain_id) => {
domain_updates.allow_chains.remove(&chain_id);
domain_updates.remove_chains.insert(chain_id);
}
}
*maybe_domain_updates = Some(domain_updates)
});
Ok(())
}
#[pallet::call_index(6)]
#[pallet::weight((T::DbWeight::get().reads_writes(1, 1), DispatchClass::Mandatory))]
pub fn update_domain_allowlist(
origin: OriginFor<T>,
updates: DomainAllowlistUpdates,
) -> DispatchResult {
ensure_none(origin)?;
ensure!(
!T::SelfChainId::get().is_consensus_chain(),
Error::<T>::OperationNotAllowed
);
let DomainAllowlistUpdates {
allow_chains,
remove_chains,
} = updates;
ChainAllowlist::<T>::mutate(|list| {
remove_chains.into_iter().for_each(|chain_id| {
list.remove(&chain_id);
});
allow_chains.into_iter().for_each(|chain_id| {
list.insert(chain_id);
});
});
Ok(())
}
}
#[pallet::inherent]
impl<T: Config> ProvideInherent for Pallet<T> {
type Call = Call<T>;
type Error = InherentError;
const INHERENT_IDENTIFIER: InherentIdentifier = INHERENT_IDENTIFIER;
fn create_inherent(data: &InherentData) -> Option<Self::Call> {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Messenger inherent data not correctly encoded")
.expect("Messenger inherent data must be provided");
inherent_data
.maybe_updates
.map(|updates| Call::update_domain_allowlist { updates })
}
fn is_inherent_required(data: &InherentData) -> Result<Option<Self::Error>, Self::Error> {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Messenger inherent data not correctly encoded")
.expect("Messenger inherent data must be provided");
Ok(if inherent_data.maybe_updates.is_none() {
None
} else {
Some(InherentError::MissingAllowlistUpdates)
})
}
fn check_inherent(call: &Self::Call, data: &InherentData) -> Result<(), Self::Error> {
let inherent_data = data
.get_data::<InherentType>(&INHERENT_IDENTIFIER)
.expect("Messenger inherent data not correctly encoded")
.expect("Messenger inherent data must be provided");
if let Some(provided_updates) = inherent_data.maybe_updates {
if let Call::update_domain_allowlist { updates } = call {
if updates != &provided_updates {
return Err(InherentError::IncorrectAllowlistUpdates);
}
}
} else {
return Err(InherentError::MissingAllowlistUpdates);
}
Ok(())
}
fn is_inherent(call: &Self::Call) -> bool {
matches!(call, Call::update_domain_allowlist { .. })
}
}
impl<T: Config> Sender<T::AccountId> for Pallet<T> {
type MessageId = MessageId;
fn send_message(
sender: &T::AccountId,
dst_chain_id: ChainId,
req: EndpointRequest,
) -> Result<Self::MessageId, DispatchError> {
let allowed_chains = ChainAllowlist::<T>::get();
ensure!(
allowed_chains.contains(&dst_chain_id),
Error::<T>::ChainNotAllowed
);
let (channel_id, fee_model) =
Self::get_open_channel_for_chain(dst_chain_id).ok_or(Error::<T>::NoOpenChannel)?;
let src_endpoint = req.src_endpoint.clone();
let nonce = Self::new_outbox_message(
T::SelfChainId::get(),
dst_chain_id,
channel_id,
VersionedPayload::V0(Payload::Endpoint(RequestResponse::Request(req))),
)?;
Self::collect_fees_for_message(
sender,
(dst_chain_id, (channel_id, nonce)),
&fee_model,
&src_endpoint,
)?;
Ok((channel_id, nonce))
}
#[cfg(feature = "runtime-benchmarks")]
fn unchecked_open_channel(dst_chain_id: ChainId) -> Result<(), DispatchError> {
let fee_model = FeeModel {
relay_fee: Default::default(),
};
let init_params = ChannelOpenParams {
max_outgoing_messages: 100,
fee_model,
};
ChainAllowlist::<T>::mutate(|list| list.insert(dst_chain_id));
let channel_id =
Self::do_init_channel(dst_chain_id, init_params, None, true, Zero::zero())?;
Self::do_open_channel(dst_chain_id, channel_id)?;
Ok(())
}
}
impl<T: Config> Pallet<T> {
fn message_weight(weight_tag: &MessageWeightTag) -> Weight {
match weight_tag {
MessageWeightTag::ProtocolChannelOpen => T::WeightInfo::do_open_channel(),
MessageWeightTag::ProtocolChannelClose => T::WeightInfo::do_close_channel(),
MessageWeightTag::EndpointRequest(endpoint) => {
T::get_endpoint_handler(endpoint)
.map(|endpoint_handler| endpoint_handler.message_weight())
.unwrap_or(Weight::zero())
}
MessageWeightTag::EndpointResponse(endpoint) => {
T::get_endpoint_handler(endpoint)
.map(|endpoint_handler| endpoint_handler.message_response_weight())
.unwrap_or(Weight::zero())
}
MessageWeightTag::None => Weight::zero(),
}
}
pub fn get_open_channel_for_chain(
dst_chain_id: ChainId,
) -> Option<(ChannelId, FeeModel<BalanceOf<T>>)> {
let mut next_channel_id = NextChannelId::<T>::get(dst_chain_id);
while let Some(channel_id) = next_channel_id.checked_sub(ChannelId::one()) {
if let Some(channel) = Channels::<T>::get(dst_chain_id, channel_id) {
if channel.state == ChannelState::Open {
return Some((channel_id, channel.fee));
}
}
next_channel_id = channel_id
}
None
}
pub(crate) fn do_open_channel(chain_id: ChainId, channel_id: ChannelId) -> DispatchResult {
Channels::<T>::try_mutate(chain_id, channel_id, |maybe_channel| -> DispatchResult {
let channel = maybe_channel.as_mut().ok_or(Error::<T>::MissingChannel)?;
ensure!(
channel.state == ChannelState::Initiated,
Error::<T>::InvalidChannelState
);
channel.state = ChannelState::Open;
Ok(())
})?;
Self::deposit_event(Event::ChannelOpen {
chain_id,
channel_id,
});
Ok(())
}
pub(crate) fn do_close_channel(
chain_id: ChainId,
channel_id: ChannelId,
close_channel_by: CloseChannelBy<T::AccountId>,
) -> DispatchResult {
Channels::<T>::try_mutate(chain_id, channel_id, |maybe_channel| -> DispatchResult {
let channel = maybe_channel.as_mut().ok_or(Error::<T>::MissingChannel)?;
ensure!(
channel.state != ChannelState::Closed,
Error::<T>::InvalidChannelState
);
if let CloseChannelBy::Owner(owner) = close_channel_by {
ensure!(channel.maybe_owner == Some(owner), Error::<T>::ChannelOwner);
}
if let Some(owner) = &channel.maybe_owner {
let hold_id = T::HoldIdentifier::messenger_channel();
let locked_amount = channel.channel_reserve_fee;
let amount_to_release = {
if channel.state == ChannelState::Open {
locked_amount
} else {
let protocol_fee = T::ChannelInitReservePortion::get() * locked_amount;
let release_amount = locked_amount.saturating_sub(protocol_fee);
T::Currency::burn_held(
&hold_id,
owner,
protocol_fee,
Precision::Exact,
Fortitude::Force,
)?;
T::OnXDMRewards::on_chain_protocol_fees(chain_id, protocol_fee);
release_amount
}
};
T::Currency::release(&hold_id, owner, amount_to_release, Precision::Exact)
.map_err(|_| Error::<T>::BalanceUnlock)?;
}
channel.state = ChannelState::Closed;
Ok(())
})?;
Self::deposit_event(Event::ChannelClosed {
chain_id,
channel_id,
});
Ok(())
}
pub(crate) fn do_init_channel(
dst_chain_id: ChainId,
init_params: ChannelOpenParams<BalanceOf<T>>,
maybe_owner: Option<T::AccountId>,
check_allowlist: bool,
channel_reserve_fee: BalanceOf<T>,
) -> Result<ChannelId, DispatchError> {
ensure!(
T::SelfChainId::get() != dst_chain_id,
Error::<T>::InvalidChain,
);
ensure!(
maybe_owner.is_none() || !channel_reserve_fee.is_zero(),
Error::<T>::InvalidChannelReserveFee,
);
if check_allowlist {
let chain_allowlist = ChainAllowlist::<T>::get();
ensure!(
chain_allowlist.contains(&dst_chain_id),
Error::<T>::ChainNotAllowed
);
}
let channel_id = NextChannelId::<T>::get(dst_chain_id);
let next_channel_id = channel_id
.checked_add(U256::one())
.ok_or(DispatchError::Arithmetic(ArithmeticError::Overflow))?;
Channels::<T>::insert(
dst_chain_id,
channel_id,
Channel {
channel_id,
state: ChannelState::Initiated,
next_inbox_nonce: Default::default(),
next_outbox_nonce: Default::default(),
latest_response_received_message_nonce: Default::default(),
max_outgoing_messages: init_params.max_outgoing_messages,
fee: init_params.fee_model,
maybe_owner,
channel_reserve_fee,
},
);
NextChannelId::<T>::insert(dst_chain_id, next_channel_id);
Self::deposit_event(Event::ChannelInitiated {
chain_id: dst_chain_id,
channel_id,
});
Ok(channel_id)
}
pub fn validate_relay_message(
xdm: &CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
pre_dispatch: bool,
consensus_state_root: StateRootOf<T>,
) -> Result<ValidatedRelayMessage, TransactionValidityError> {
let (next_nonce, maybe_channel) =
match Channels::<T>::get(xdm.src_chain_id, xdm.channel_id) {
None => {
log::debug!(
"Initiating new channel: {:?} to chain: {:?}",
xdm.channel_id,
xdm.src_chain_id
);
(Nonce::zero(), None)
}
Some(channel) => {
log::debug!(
"Message to channel: {:?} from chain: {:?}",
xdm.channel_id,
xdm.src_chain_id
);
(channel.next_inbox_nonce, Some(channel))
}
};
let key = StorageKey(
T::StorageKeys::outbox_storage_key(
xdm.src_chain_id,
(T::SelfChainId::get(), xdm.channel_id, xdm.nonce),
)
.ok_or(UnknownTransaction::CannotLookup)?,
);
let msg = Self::do_verify_xdm(next_nonce, key, consensus_state_root, xdm)?;
let is_valid_call = match &msg.payload {
VersionedPayload::V0(payload) => match payload {
Payload::Protocol(RequestResponse::Request(req)) => match req {
ProtocolMessageRequest::ChannelOpen(_) => maybe_channel.is_none(),
ProtocolMessageRequest::ChannelClose => {
if let Some(ref channel) = maybe_channel {
!(channel.state == ChannelState::Closed)
} else {
false
}
}
},
Payload::Endpoint(RequestResponse::Request(_)) => {
if let Some(ref channel) = maybe_channel {
!(channel.state == ChannelState::Initiated)
} else {
false
}
}
_ => false,
},
};
if !is_valid_call {
log::error!("Unexpected call instead of channel open request: {:?}", msg,);
return Err(InvalidTransaction::Call.into());
}
match msg.nonce.cmp(&next_nonce) {
Ordering::Less => return Err(InvalidTransaction::Stale.into()),
Ordering::Greater if pre_dispatch => return Err(InvalidTransaction::Future.into()),
_ => {}
}
let validated_relay_msg = ValidatedRelayMessage {
msg_nonce: msg.nonce,
dst_chain_id: msg.dst_chain_id,
channel_id: msg.channel_id,
next_nonce,
};
if pre_dispatch {
let should_init_channel = maybe_channel.is_none();
Self::pre_dispatch_relay_message(msg, should_init_channel)?;
}
Ok(validated_relay_msg)
}
pub(crate) fn pre_dispatch_relay_message(
msg: Message<BalanceOf<T>>,
should_init_channel: bool,
) -> Result<(), TransactionValidityError> {
if should_init_channel {
if let VersionedPayload::V0(Payload::Protocol(RequestResponse::Request(
ProtocolMessageRequest::ChannelOpen(params),
))) = msg.payload
{
Self::do_init_channel(msg.src_chain_id, params, None, false, Zero::zero())
.map_err(|err| {
log::error!(
"Error initiating channel: {:?} with chain: {:?}: {:?}",
msg.channel_id,
msg.src_chain_id,
err
);
InvalidTransaction::Call
})?;
} else {
log::error!("Unexpected call instead of channel open request: {:?}", msg,);
return Err(InvalidTransaction::Call.into());
}
}
Self::deposit_event(Event::InboxMessage {
chain_id: msg.src_chain_id,
channel_id: msg.channel_id,
nonce: msg.nonce,
});
Inbox::<T>::put(msg);
Ok(())
}
pub fn validate_relay_message_response(
xdm: &CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
pre_dispatch: bool,
consensus_state_root: StateRootOf<T>,
) -> Result<ValidatedRelayMessage, TransactionValidityError> {
let next_nonce =
match Channels::<T>::get(xdm.src_chain_id, xdm.channel_id) {
None => {
log::error!("Unexpected inbox message response: {:?}", xdm,);
return Err(InvalidTransaction::Call.into());
}
Some(channel) => match channel.latest_response_received_message_nonce {
None => Nonce::zero(),
Some(last_nonce) => last_nonce.checked_add(Nonce::one()).ok_or(
InvalidTransaction::Custom(crate::verification_errors::NONCE_OVERFLOW),
)?,
},
};
let key = StorageKey(
T::StorageKeys::inbox_responses_storage_key(
xdm.src_chain_id,
(T::SelfChainId::get(), xdm.channel_id, xdm.nonce),
)
.ok_or(UnknownTransaction::CannotLookup)?,
);
let msg = Self::do_verify_xdm(next_nonce, key, consensus_state_root, xdm)?;
match msg.nonce.cmp(&next_nonce) {
Ordering::Less => return Err(InvalidTransaction::Stale.into()),
Ordering::Greater if pre_dispatch => return Err(InvalidTransaction::Future.into()),
_ => {}
}
let validated_relay_msg = ValidatedRelayMessage {
msg_nonce: msg.nonce,
dst_chain_id: msg.dst_chain_id,
channel_id: msg.channel_id,
next_nonce,
};
if pre_dispatch {
Self::pre_dispatch_relay_message_response(msg)?;
}
Ok(validated_relay_msg)
}
pub(crate) fn pre_dispatch_relay_message_response(
msg: Message<BalanceOf<T>>,
) -> Result<(), TransactionValidityError> {
Self::deposit_event(Event::OutboxMessageResponse {
chain_id: msg.src_chain_id,
channel_id: msg.channel_id,
nonce: msg.nonce,
});
OutboxResponses::<T>::put(msg);
Ok(())
}
pub(crate) fn do_verify_xdm(
next_nonce: Nonce,
storage_key: StorageKey,
consensus_state_root: StateRootOf<T>,
xdm: &CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
) -> Result<Message<BalanceOf<T>>, TransactionValidityError> {
let next_channel_id = NextChannelId::<T>::get(xdm.src_chain_id);
ensure!(
xdm.channel_id <= next_channel_id,
InvalidTransaction::Custom(crate::verification_errors::INVALID_CHANNEL)
);
ensure!(
xdm.nonce >= next_nonce,
InvalidTransaction::Custom(crate::verification_errors::INVALID_NONCE)
);
let state_root = if let Some(domain_proof) = xdm.proof.domain_proof().clone()
&& let Some(domain_id) = xdm.src_chain_id.maybe_domain_chain()
{
let confirmed_domain_block_storage_key =
T::StorageKeys::confirmed_domain_block_storage_key(domain_id)
.ok_or(UnknownTransaction::CannotLookup)?;
StorageProofVerifier::<T::Hashing>::get_decoded_value::<
sp_domains::ExecutionReceipt<
BlockNumberFor<T>,
T::Hash,
BlockNumberFor<T>,
T::Hash,
BalanceOf<T>,
>,
>(
&consensus_state_root,
domain_proof,
StorageKey(confirmed_domain_block_storage_key),
)
.map_err(|err| {
log::error!(
target: "runtime::messenger",
"Failed to verify storage proof for confirmed Domain block: {:?}",
err
);
TransactionValidityError::Invalid(InvalidTransaction::BadProof)
})?
.final_state_root
} else {
consensus_state_root
};
let msg =
StorageProofVerifier::<T::Hashing>::get_decoded_value::<Message<BalanceOf<T>>>(
&state_root,
xdm.proof.message_proof(),
storage_key,
)
.map_err(|err| {
log::error!(
target: "runtime::messenger",
"Failed to verify storage proof for message: {:?}",
err
);
TransactionValidityError::Invalid(InvalidTransaction::BadProof)
})?;
Ok(msg)
}
pub fn outbox_storage_key(message_key: MessageKey) -> Vec<u8> {
Outbox::<T>::hashed_key_for(message_key)
}
pub fn inbox_response_storage_key(message_key: MessageKey) -> Vec<u8> {
InboxResponses::<T>::hashed_key_for(message_key)
}
pub fn channel_storage_key(chain_id: ChainId, channel_id: ChannelId) -> Vec<u8> {
Channels::<T>::hashed_key_for(chain_id, channel_id)
}
pub fn domain_chains_allowlist_update(
domain_id: DomainId,
) -> Option<DomainAllowlistUpdates> {
DomainChainAllowlistUpdate::<T>::get(domain_id).filter(|updates| !updates.is_empty())
}
pub fn domain_allow_list_update_storage_key(domain_id: DomainId) -> Vec<u8> {
DomainChainAllowlistUpdate::<T>::hashed_key_for(domain_id)
}
pub fn updated_channels() -> BTreeSet<(ChainId, ChannelId)> {
UpdatedChannels::<T>::get()
}
pub fn channel_nonce(chain_id: ChainId, channel_id: ChannelId) -> Option<ChannelNonce> {
Channels::<T>::get(chain_id, channel_id).map(|channel| {
let last_inbox_nonce = channel.next_inbox_nonce.checked_sub(U256::one());
ChannelNonce {
relay_msg_nonce: last_inbox_nonce,
relay_response_msg_nonce: channel.latest_response_received_message_nonce,
}
})
}
pub fn pre_dispatch_with_trusted_mmr_proof(
call: &Call<T>,
) -> Result<(), TransactionValidityError> {
match call {
Call::relay_message { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::extract_leaf_without_verifying(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
Self::validate_relay_message(xdm, true, consensus_state_root)?;
Ok(())
}
Call::relay_message_response { msg: xdm } => {
let consensus_state_root = T::MmrProofVerifier::extract_leaf_without_verifying(
xdm.proof.consensus_mmr_proof(),
)
.ok_or(InvalidTransaction::BadProof)?
.state_root();
Self::validate_relay_message_response(xdm, true, consensus_state_root)?;
Ok(())
}
call => Self::pre_dispatch(call),
}
}
}
}
impl<T> Pallet<T>
where
T: Config + frame_system::offchain::SendTransactionTypes<Call<T>>,
{
pub fn outbox_message_unsigned(
msg: CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
) -> Option<T::Extrinsic> {
let call = Call::relay_message { msg };
T::Extrinsic::new(call.into(), None)
}
pub fn inbox_response_message_unsigned(
msg: CrossDomainMessage<BlockNumberFor<T>, T::Hash, T::MmrHash>,
) -> Option<T::Extrinsic> {
let call = Call::relay_message_response { msg };
T::Extrinsic::new(call.into(), None)
}
pub fn should_relay_outbox_message(dst_chain_id: ChainId, msg_id: MessageId) -> bool {
Outbox::<T>::contains_key((dst_chain_id, msg_id.0, msg_id.1))
}
pub fn should_relay_inbox_message_response(dst_chain_id: ChainId, msg_id: MessageId) -> bool {
InboxResponses::<T>::contains_key((dst_chain_id, msg_id.0, msg_id.1))
}
}
impl<T: Config> sp_domains::DomainBundleSubmitted for Pallet<T> {
fn domain_bundle_submitted(domain_id: DomainId) {
DomainChainAllowlistUpdate::<T>::mutate(domain_id, |maybe_updates| {
if let Some(ref mut updates) = maybe_updates {
updates.clear();
}
});
}
}
impl<T: Config> sp_domains::OnDomainInstantiated for Pallet<T> {
fn on_domain_instantiated(domain_id: DomainId) {
DomainChainAllowlistUpdate::<T>::insert(domain_id, DomainAllowlistUpdates::default());
}
}