mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-04-26 20:27:58 +00:00
002d9260f9
**Update:** Pushed additional changes based on the review comments. **This pull request fixes various spelling mistakes in this repository.** Most of the changes are contained in the first **3** commits: - `Fix spelling mistakes in comments and docs` - `Fix spelling mistakes in test names` - `Fix spelling mistakes in error messages, panic messages, logs and tracing` Other source code spelling mistakes are separated into individual commits for easier reviewing: - `Fix the spelling of 'authority'` - `Fix the spelling of 'REASONABLE_HEADERS_IN_JUSTIFICATION_ANCESTRY'` - `Fix the spelling of 'prev_enqueud_messages'` - `Fix the spelling of 'endpoint'` - `Fix the spelling of 'children'` - `Fix the spelling of 'PenpalSiblingSovereignAccount'` - `Fix the spelling of 'PenpalSudoAccount'` - `Fix the spelling of 'insufficient'` - `Fix the spelling of 'PalletXcmExtrinsicsBenchmark'` - `Fix the spelling of 'subtracted'` - `Fix the spelling of 'CandidatePendingAvailability'` - `Fix the spelling of 'exclusive'` - `Fix the spelling of 'until'` - `Fix the spelling of 'discriminator'` - `Fix the spelling of 'nonexistent'` - `Fix the spelling of 'subsystem'` - `Fix the spelling of 'indices'` - `Fix the spelling of 'committed'` - `Fix the spelling of 'topology'` - `Fix the spelling of 'response'` - `Fix the spelling of 'beneficiary'` - `Fix the spelling of 'formatted'` - `Fix the spelling of 'UNKNOWN_PROOF_REQUEST'` - `Fix the spelling of 'succeeded'` - `Fix the spelling of 'reopened'` - `Fix the spelling of 'proposer'` - `Fix the spelling of 'InstantiationNonce'` - `Fix the spelling of 'depositor'` - `Fix the spelling of 'expiration'` - `Fix the spelling of 'phantom'` - `Fix the spelling of 'AggregatedKeyValue'` - `Fix the spelling of 'randomness'` - `Fix the spelling of 'defendant'` - `Fix the spelling of 'AquaticMammal'` - `Fix the spelling of 'transactions'` - `Fix the spelling of 'PassingTracingSubscriber'` - `Fix the spelling of 'TxSignaturePayload'` - `Fix the spelling of 'versioning'` - `Fix the spelling of 'descendant'` - `Fix the spelling of 'overridden'` - `Fix the spelling of 'network'` Let me know if this structure is adequate. **Note:** The usage of the words `Merkle`, `Merkelize`, `Merklization`, `Merkelization`, `Merkleization`, is somewhat inconsistent but I left it as it is. ~~**Note:** In some places the term `Receival` is used to refer to message reception, IMO `Reception` is the correct word here, but I left it as it is.~~ ~~**Note:** In some places the term `Overlayed` is used instead of the more acceptable version `Overlaid` but I also left it as it is.~~ ~~**Note:** In some places the term `Applyable` is used instead of the correct version `Applicable` but I also left it as it is.~~ **Note:** Some usage of British vs American english e.g. `judgement` vs `judgment`, `initialise` vs `initialize`, `optimise` vs `optimize` etc. are both present in different places, but I suppose that's understandable given the number of contributors. ~~**Note:** There is a spelling mistake in `.github/CODEOWNERS` but it triggers errors in CI when I make changes to it, so I left it as it is.~~
337 lines
12 KiB
Rust
337 lines
12 KiB
Rust
// Copyright (C) Parity Technologies (UK) Ltd.
|
|
// This file is part of Polkadot.
|
|
|
|
// Polkadot 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.
|
|
|
|
// Polkadot 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 Polkadot. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
mod mock;
|
|
|
|
use mock::{
|
|
fake_message_hash, kusama_like_with_balances, AccountId, Balance, Balances, BaseXcmWeight,
|
|
System, XcmConfig, CENTS,
|
|
};
|
|
use polkadot_parachain_primitives::primitives::Id as ParaId;
|
|
use sp_runtime::traits::AccountIdConversion;
|
|
use xcm::latest::prelude::*;
|
|
use xcm_executor::XcmExecutor;
|
|
|
|
pub const ALICE: AccountId = AccountId::new([0u8; 32]);
|
|
pub const PARA_ID: u32 = 2000;
|
|
pub const INITIAL_BALANCE: u128 = 100_000_000_000;
|
|
pub const REGISTER_AMOUNT: Balance = 10 * CENTS;
|
|
|
|
// Construct a `BuyExecution` order.
|
|
fn buy_execution<C>() -> Instruction<C> {
|
|
BuyExecution { fees: (Here, REGISTER_AMOUNT).into(), weight_limit: Unlimited }
|
|
}
|
|
|
|
/// Scenario:
|
|
/// A parachain transfers funds on the relay-chain to another parachain's account.
|
|
///
|
|
/// Asserts that the parachain accounts are updated as expected.
|
|
#[test]
|
|
fn withdraw_and_deposit_works() {
|
|
let para_acc: AccountId = ParaId::from(PARA_ID).into_account_truncating();
|
|
let balances = vec![(ALICE, INITIAL_BALANCE), (para_acc.clone(), INITIAL_BALANCE)];
|
|
kusama_like_with_balances(balances).execute_with(|| {
|
|
let other_para_id = 3000;
|
|
let amount = REGISTER_AMOUNT;
|
|
let weight = BaseXcmWeight::get() * 3;
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
DepositAsset {
|
|
assets: AllCounted(1).into(),
|
|
beneficiary: Parachain(other_para_id).into(),
|
|
},
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
let other_para_acc: AccountId = ParaId::from(other_para_id).into_account_truncating();
|
|
assert_eq!(Balances::free_balance(para_acc), INITIAL_BALANCE - amount);
|
|
assert_eq!(Balances::free_balance(other_para_acc), amount);
|
|
});
|
|
}
|
|
|
|
/// Scenario:
|
|
/// Alice simply wants to transfer funds to Bob's account via XCM.
|
|
///
|
|
/// Asserts that the balances are updated correctly and the correct events are fired.
|
|
#[test]
|
|
fn transfer_asset_works() {
|
|
let bob = AccountId::new([1u8; 32]);
|
|
let balances = vec![(ALICE, INITIAL_BALANCE), (bob.clone(), INITIAL_BALANCE)];
|
|
kusama_like_with_balances(balances).execute_with(|| {
|
|
let amount = REGISTER_AMOUNT;
|
|
let weight = BaseXcmWeight::get();
|
|
let message = Xcm(vec![TransferAsset {
|
|
assets: (Here, amount).into(),
|
|
beneficiary: AccountId32 { network: None, id: bob.clone().into() }.into(),
|
|
}]);
|
|
let mut hash = fake_message_hash(&message);
|
|
// Use `prepare_and_execute` here to pass through the barrier
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
AccountId32 { network: None, id: ALICE.into() },
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
weight,
|
|
);
|
|
System::assert_last_event(
|
|
pallet_balances::Event::Transfer { from: ALICE, to: bob.clone(), amount }.into(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
assert_eq!(Balances::free_balance(ALICE), INITIAL_BALANCE - amount);
|
|
assert_eq!(Balances::free_balance(bob), INITIAL_BALANCE + amount);
|
|
});
|
|
}
|
|
|
|
/// Scenario:
|
|
/// A parachain wants to be notified that a transfer worked correctly.
|
|
/// It includes a `QueryHolding` order after the deposit to get notified on success.
|
|
/// This somewhat abuses `QueryHolding` as an indication of execution success. It works because
|
|
/// order execution halts on error (so no `QueryResponse` will be sent if the previous order
|
|
/// failed). The inner response sent due to the query is not used.
|
|
///
|
|
/// Asserts that the balances are updated correctly and the expected XCM is sent.
|
|
#[test]
|
|
fn report_holding_works() {
|
|
use xcm::opaque::latest::prelude::*;
|
|
let para_acc: AccountId = ParaId::from(PARA_ID).into_account_truncating();
|
|
let balances = vec![(ALICE, INITIAL_BALANCE), (para_acc.clone(), INITIAL_BALANCE)];
|
|
kusama_like_with_balances(balances).execute_with(|| {
|
|
let other_para_id = 3000;
|
|
let amount = REGISTER_AMOUNT;
|
|
let weight = BaseXcmWeight::get() * 4;
|
|
let response_info = QueryResponseInfo {
|
|
destination: Parachain(PARA_ID).into(),
|
|
query_id: 1234,
|
|
max_weight: Weight::from_parts(1_000_000_000, 1_000_000_000),
|
|
};
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
DepositAsset {
|
|
assets: AllCounted(1).into(),
|
|
beneficiary: OnlyChild.into(), // invalid destination
|
|
},
|
|
// is not triggered because the deposit fails
|
|
ReportHolding { response_info: response_info.clone(), assets: All.into() },
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(
|
|
r,
|
|
Outcome::Incomplete {
|
|
used: weight - BaseXcmWeight::get(),
|
|
error: XcmError::FailedToTransactAsset("AccountIdConversionFailed")
|
|
}
|
|
);
|
|
// there should be no query response sent for the failed deposit
|
|
assert_eq!(mock::sent_xcm(), vec![]);
|
|
assert_eq!(Balances::free_balance(para_acc.clone()), INITIAL_BALANCE - amount);
|
|
|
|
// now do a successful transfer
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
DepositAsset {
|
|
assets: AllCounted(1).into(),
|
|
beneficiary: Parachain(other_para_id).into(),
|
|
},
|
|
// used to get a notification in case of success
|
|
ReportHolding { response_info: response_info.clone(), assets: AllCounted(1).into() },
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
let other_para_acc: AccountId = ParaId::from(other_para_id).into_account_truncating();
|
|
assert_eq!(Balances::free_balance(other_para_acc), amount);
|
|
assert_eq!(Balances::free_balance(para_acc), INITIAL_BALANCE - 2 * amount);
|
|
let expected_msg = Xcm(vec![QueryResponse {
|
|
query_id: response_info.query_id,
|
|
response: Response::Assets(vec![].into()),
|
|
max_weight: response_info.max_weight,
|
|
querier: Some(Here.into()),
|
|
}]);
|
|
let expected_hash = fake_message_hash(&expected_msg);
|
|
assert_eq!(
|
|
mock::sent_xcm(),
|
|
vec![(Parachain(PARA_ID).into(), expected_msg, expected_hash,)]
|
|
);
|
|
});
|
|
}
|
|
|
|
/// Scenario:
|
|
/// A parachain wants to move KSM from Kusama to Asset Hub.
|
|
/// The parachain sends an XCM to withdraw funds combined with a teleport to the destination.
|
|
///
|
|
/// This way of moving funds from a relay to a parachain will only work for trusted chains.
|
|
/// Reserve based transfer should be used to move KSM to a community parachain.
|
|
///
|
|
/// Asserts that the balances are updated accordingly and the correct XCM is sent.
|
|
#[test]
|
|
fn teleport_to_asset_hub_works() {
|
|
use xcm::opaque::latest::prelude::*;
|
|
let para_acc: AccountId = ParaId::from(PARA_ID).into_account_truncating();
|
|
let balances = vec![(ALICE, INITIAL_BALANCE), (para_acc.clone(), INITIAL_BALANCE)];
|
|
kusama_like_with_balances(balances).execute_with(|| {
|
|
let asset_hub_id = 1000;
|
|
let other_para_id = 3000;
|
|
let amount = REGISTER_AMOUNT;
|
|
let teleport_effects = vec![
|
|
buy_execution(), // unchecked mock value
|
|
DepositAsset {
|
|
assets: AllCounted(1).into(),
|
|
beneficiary: (Parent, Parachain(PARA_ID)).into(),
|
|
},
|
|
];
|
|
let weight = BaseXcmWeight::get() * 3;
|
|
|
|
// teleports are allowed to community chains, even in the absence of trust from their side.
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
InitiateTeleport {
|
|
assets: All.into(),
|
|
dest: Parachain(other_para_id).into(),
|
|
xcm: Xcm(teleport_effects.clone()),
|
|
},
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
let expected_msg = Xcm(vec![ReceiveTeleportedAsset((Parent, amount).into()), ClearOrigin]
|
|
.into_iter()
|
|
.chain(teleport_effects.clone().into_iter())
|
|
.collect());
|
|
let expected_hash = fake_message_hash(&expected_msg);
|
|
assert_eq!(
|
|
mock::sent_xcm(),
|
|
vec![(Parachain(other_para_id).into(), expected_msg, expected_hash,)]
|
|
);
|
|
|
|
// teleports are allowed from asset hub to kusama.
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
InitiateTeleport {
|
|
assets: All.into(),
|
|
dest: Parachain(asset_hub_id).into(),
|
|
xcm: Xcm(teleport_effects.clone()),
|
|
},
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
// 2 * amount because of the other teleport above
|
|
assert_eq!(Balances::free_balance(para_acc), INITIAL_BALANCE - 2 * amount);
|
|
let expected_msg = Xcm(vec![ReceiveTeleportedAsset((Parent, amount).into()), ClearOrigin]
|
|
.into_iter()
|
|
.chain(teleport_effects.clone().into_iter())
|
|
.collect());
|
|
let expected_hash = fake_message_hash(&expected_msg);
|
|
assert_eq!(
|
|
mock::sent_xcm(),
|
|
vec![
|
|
(Parachain(other_para_id).into(), expected_msg.clone(), expected_hash,),
|
|
(Parachain(asset_hub_id).into(), expected_msg, expected_hash,)
|
|
]
|
|
);
|
|
});
|
|
}
|
|
|
|
/// Scenario:
|
|
/// A parachain wants to move KSM from Kusama to the parachain.
|
|
/// It withdraws funds and then deposits them into the reserve account of the destination chain.
|
|
/// to the destination.
|
|
///
|
|
/// Asserts that the balances are updated accordingly and the correct XCM is sent.
|
|
#[test]
|
|
fn reserve_based_transfer_works() {
|
|
use xcm::opaque::latest::prelude::*;
|
|
let para_acc: AccountId = ParaId::from(PARA_ID).into_account_truncating();
|
|
let balances = vec![(ALICE, INITIAL_BALANCE), (para_acc.clone(), INITIAL_BALANCE)];
|
|
kusama_like_with_balances(balances).execute_with(|| {
|
|
let other_para_id = 3000;
|
|
let amount = REGISTER_AMOUNT;
|
|
let transfer_effects = vec![
|
|
buy_execution(), // unchecked mock value
|
|
DepositAsset {
|
|
assets: AllCounted(1).into(),
|
|
beneficiary: (Parent, Parachain(PARA_ID)).into(),
|
|
},
|
|
];
|
|
let message = Xcm(vec![
|
|
WithdrawAsset((Here, amount).into()),
|
|
buy_execution(),
|
|
DepositReserveAsset {
|
|
assets: AllCounted(1).into(),
|
|
dest: Parachain(other_para_id).into(),
|
|
xcm: Xcm(transfer_effects.clone()),
|
|
},
|
|
]);
|
|
let mut hash = fake_message_hash(&message);
|
|
let weight = BaseXcmWeight::get() * 3;
|
|
let r = XcmExecutor::<XcmConfig>::prepare_and_execute(
|
|
Parachain(PARA_ID),
|
|
message,
|
|
&mut hash,
|
|
weight,
|
|
Weight::zero(),
|
|
);
|
|
assert_eq!(r, Outcome::Complete { used: weight });
|
|
assert_eq!(Balances::free_balance(para_acc), INITIAL_BALANCE - amount);
|
|
let expected_msg = Xcm(vec![ReserveAssetDeposited((Parent, amount).into()), ClearOrigin]
|
|
.into_iter()
|
|
.chain(transfer_effects.into_iter())
|
|
.collect());
|
|
let expected_hash = fake_message_hash(&expected_msg);
|
|
assert_eq!(
|
|
mock::sent_xcm(),
|
|
vec![(Parachain(other_para_id).into(), expected_msg, expected_hash,)]
|
|
);
|
|
});
|
|
}
|