// 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 . //! Rialto chain specification for CLI. use crate::cli::{ bridge, encode_call::{self, Call, CliEncodeCall}, encode_message, send_message::{self, DispatchFeePayment}, CliChain, }; use anyhow::anyhow; use bp_message_dispatch::{CallOrigin, MessagePayload}; use bp_runtime::EncodedOrDecodedCall; use codec::Decode; use frame_support::weights::{DispatchInfo, GetDispatchInfo}; use relay_rialto_client::Rialto; use sp_version::RuntimeVersion; impl CliEncodeCall for Rialto { fn encode_call(call: &Call) -> anyhow::Result> { Ok(match call { Call::Raw { data } => Self::Call::decode(&mut &*data.0)?.into(), Call::Remark { remark_payload, .. } => rialto_runtime::Call::System(rialto_runtime::SystemCall::remark { remark: remark_payload.as_ref().map(|x| x.0.clone()).unwrap_or_default(), }) .into(), Call::Transfer { recipient, amount } => rialto_runtime::Call::Balances(rialto_runtime::BalancesCall::transfer { dest: recipient.raw_id().into(), value: amount.0, }) .into(), Call::BridgeSendMessage { lane, payload, fee, bridge_instance_index } => match *bridge_instance_index { bridge::RIALTO_TO_MILLAU_INDEX => { let payload = Decode::decode(&mut &*payload.0)?; rialto_runtime::Call::BridgeMillauMessages( rialto_runtime::MessagesCall::send_message { lane_id: lane.0, payload, delivery_and_dispatch_fee: fee.0, }, ) .into() }, _ => anyhow::bail!( "Unsupported target bridge pallet with instance index: {}", bridge_instance_index ), }, }) } fn get_dispatch_info(call: &EncodedOrDecodedCall) -> anyhow::Result { Ok(call.to_decoded()?.get_dispatch_info()) } } impl CliChain for Rialto { const RUNTIME_VERSION: RuntimeVersion = rialto_runtime::VERSION; type KeyPair = sp_core::sr25519::Pair; type MessagePayload = MessagePayload< bp_rialto::AccountId, bp_millau::AccountSigner, bp_millau::Signature, Vec, >; fn ss58_format() -> u16 { rialto_runtime::SS58Prefix::get() as u16 } 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 Rialto's MessagePayload: {:?}", e)), encode_message::MessagePayload::Call { mut call, mut sender, dispatch_weight } => { type Source = Rialto; type Target = relay_millau_client::Millau; 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::RIALTO_TO_MILLAU_INDEX, ); let call = Target::encode_call(&call)?; let dispatch_weight = dispatch_weight.map(Ok).unwrap_or_else(|| { call.to_decoded().map(|call| call.get_dispatch_info().weight) })?; Ok(send_message::message_payload( spec_version, dispatch_weight, origin, &call, DispatchFeePayment::AtSourceChain, )) }, } } }