mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-31 04:01:02 +00:00
pallet-xcm: ensure xcm outcome is always complete, revert effects otherwise (#2405)
On extrinsics/call, ensure local XCM execution is complete/successful. Otherwise, fail the extrinsic so that state changes don't get committed to the db. Added regression tests that fail without the fix. fixes #2237 --------- Co-authored-by: Adrian Catangiu <adrian@parity.io>
This commit is contained in:
@@ -445,7 +445,7 @@ fn trapped_assets_can_be_claimed() {
|
||||
assert_eq!(AssetTraps::<Test>::iter().collect::<Vec<_>>(), vec![]);
|
||||
|
||||
let weight = BaseXcmWeight::get() * 3;
|
||||
assert_ok!(XcmPallet::execute(
|
||||
assert_ok!(<XcmPallet as xcm_builder::ExecuteController<_, _>>::execute(
|
||||
RuntimeOrigin::signed(ALICE),
|
||||
Box::new(VersionedXcm::from(Xcm(vec![
|
||||
ClaimAsset { assets: (Here, SEND_AMOUNT).into(), ticket: Here.into() },
|
||||
@@ -459,6 +459,52 @@ fn trapped_assets_can_be_claimed() {
|
||||
});
|
||||
}
|
||||
|
||||
/// Test failure to complete execution reverts intermediate side-effects.
|
||||
///
|
||||
/// XCM program will withdraw and deposit some assets, then fail execution of a further withdraw.
|
||||
/// Assert that the previous instructions effects are reverted.
|
||||
#[test]
|
||||
fn incomplete_execute_reverts_side_effects() {
|
||||
let balances = vec![(ALICE, INITIAL_BALANCE), (BOB, INITIAL_BALANCE)];
|
||||
new_test_ext_with_balances(balances).execute_with(|| {
|
||||
let weight = BaseXcmWeight::get() * 4;
|
||||
let dest: MultiLocation = Junction::AccountId32 { network: None, id: BOB.into() }.into();
|
||||
assert_eq!(Balances::total_balance(&ALICE), INITIAL_BALANCE);
|
||||
let amount_to_send = INITIAL_BALANCE - ExistentialDeposit::get();
|
||||
let assets: MultiAssets = (Here, amount_to_send).into();
|
||||
let result = XcmPallet::execute(
|
||||
RuntimeOrigin::signed(ALICE),
|
||||
Box::new(VersionedXcm::from(Xcm(vec![
|
||||
// Withdraw + BuyExec + Deposit should work
|
||||
WithdrawAsset(assets.clone()),
|
||||
buy_execution(assets.inner()[0].clone()),
|
||||
DepositAsset { assets: assets.clone().into(), beneficiary: dest },
|
||||
// Withdrawing once more will fail because of InsufficientBalance, and we expect to
|
||||
// revert the effects of the above instructions as well
|
||||
WithdrawAsset(assets),
|
||||
]))),
|
||||
weight,
|
||||
);
|
||||
// all effects are reverted and balances unchanged for either sender or receiver
|
||||
assert_eq!(Balances::total_balance(&ALICE), INITIAL_BALANCE);
|
||||
assert_eq!(Balances::total_balance(&BOB), INITIAL_BALANCE);
|
||||
assert_eq!(
|
||||
result,
|
||||
Err(sp_runtime::DispatchErrorWithPostInfo {
|
||||
post_info: frame_support::dispatch::PostDispatchInfo {
|
||||
actual_weight: None,
|
||||
pays_fee: frame_support::dispatch::Pays::Yes,
|
||||
},
|
||||
error: sp_runtime::DispatchError::Module(sp_runtime::ModuleError {
|
||||
index: 4,
|
||||
error: [24, 0, 0, 0,],
|
||||
message: Some("LocalExecutionIncomplete")
|
||||
})
|
||||
})
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
#[test]
|
||||
fn fake_latest_versioned_multilocation_works() {
|
||||
use codec::Encode;
|
||||
|
||||
Reference in New Issue
Block a user