// Copyright 2019-2021 Parity Technologies (UK) Ltd. // 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 . use anyhow::anyhow; use bp_message_dispatch::{CallOrigin, MessagePayload}; use bp_runtime::EncodedOrDecodedCall; use codec::Decode; use frame_support::weights::{DispatchClass, DispatchInfo, Pays, Weight}; use relay_kusama_client::Kusama; use sp_version::RuntimeVersion; use crate::cli::{ bridge, encode_call::{self, Call, CliEncodeCall}, encode_message, send_message::{self, DispatchFeePayment}, CliChain, }; /// Weight of the `system::remark` call at Kusama. /// /// This weight is larger (x2) than actual weight at current Kusama runtime to avoid unsuccessful /// calls in the future. But since it is used only in tests (and on test chains), this is ok. pub(crate) const SYSTEM_REMARK_CALL_WEIGHT: Weight = 2 * 1_345_000; impl CliEncodeCall for Kusama { fn encode_call(call: &Call) -> anyhow::Result> { Ok(match call { Call::Raw { data } => EncodedOrDecodedCall::Encoded(data.0.clone()), Call::Remark { remark_payload, .. } => relay_kusama_client::runtime::Call::System( relay_kusama_client::runtime::SystemCall::remark( remark_payload.as_ref().map(|x| x.0.clone()).unwrap_or_default(), ), ) .into(), Call::BridgeSendMessage { lane, payload, fee, bridge_instance_index } => match *bridge_instance_index { bridge::KUSAMA_TO_POLKADOT_INDEX => { let payload = Decode::decode(&mut &*payload.0)?; relay_kusama_client::runtime::Call::BridgePolkadotMessages( relay_kusama_client::runtime::BridgePolkadotMessagesCall::send_message( lane.0, payload, fee.0, ), ) .into() }, _ => anyhow::bail!( "Unsupported target bridge pallet with instance index: {}", bridge_instance_index ), }, _ => anyhow::bail!("Unsupported Kusama call: {:?}", call), }) } fn get_dispatch_info(call: &EncodedOrDecodedCall) -> anyhow::Result { match *call { EncodedOrDecodedCall::Decoded(relay_kusama_client::runtime::Call::System( relay_kusama_client::runtime::SystemCall::remark(_), )) => Ok(DispatchInfo { weight: crate::chains::kusama::SYSTEM_REMARK_CALL_WEIGHT, class: DispatchClass::Normal, pays_fee: Pays::Yes, }), _ => anyhow::bail!("Unsupported Kusama call: {:?}", call), } } } impl CliChain for Kusama { const RUNTIME_VERSION: RuntimeVersion = bp_kusama::VERSION; type KeyPair = sp_core::sr25519::Pair; type MessagePayload = MessagePayload< bp_kusama::AccountId, bp_polkadot::AccountPublic, bp_polkadot::Signature, Vec, >; fn ss58_format() -> u16 { sp_core::crypto::Ss58AddressFormat::from( sp_core::crypto::Ss58AddressFormatRegistry::KusamaAccount, ) .into() } fn encode_message( message: encode_message::MessagePayload, ) -> anyhow::Result { match message { encode_message::MessagePayload::Raw { data } => MessagePayload::decode(&mut &*data.0) .map_err(|e| anyhow!("Failed to decode Kusama's MessagePayload: {:?}", e)), encode_message::MessagePayload::Call { mut call, mut sender, dispatch_weight } => { type Source = Kusama; type Target = relay_polkadot_client::Polkadot; sender.enforce_chain::(); let spec_version = Target::RUNTIME_VERSION.spec_version; let origin = CallOrigin::SourceAccount(sender.raw_id()); encode_call::preprocess_call::( &mut call, bridge::KUSAMA_TO_POLKADOT_INDEX, ); let call = Target::encode_call(&call)?; let dispatch_weight = dispatch_weight.map(Ok).unwrap_or_else(|| { Err(anyhow::format_err!( "Please specify dispatch weight of the encoded Polkadot call" )) })?; Ok(send_message::message_payload( spec_version, dispatch_weight, origin, &call, DispatchFeePayment::AtSourceChain, )) }, } } }