#![cfg(test)]
use codec::Encode;
use frame_support::weights::Weight;
use polkadot_test_client::{
BlockBuilderExt, ClientBlockImportExt, DefaultTestClientBuilderExt, InitPolkadotBlockBuilder,
TestClientBuilder, TestClientBuilderExt,
};
use polkadot_test_runtime::{pallet_test_notifier, xcm_config::XcmConfig};
use polkadot_test_service::construct_extrinsic;
use sp_runtime::traits::Block;
use sp_state_machine::InspectState;
use xcm::{latest::prelude::*, VersionedResponse, VersionedXcm};
use xcm_executor::traits::WeightBounds;
#[test]
fn basic_buy_fees_message_executes() {
sp_tracing::try_init_simple();
let client = TestClientBuilder::new().build();
let msg = Xcm(vec![
WithdrawAsset((Parent, 100).into()),
BuyExecution { fees: (Parent, 1).into(), weight_limit: Unlimited },
DepositAsset { assets: Wild(AllCounted(1)), beneficiary: Parent.into() },
]);
let mut block_builder = client.init_polkadot_block_builder();
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::Xcm(pallet_xcm::Call::execute {
message: Box::new(VersionedXcm::from(msg)),
max_weight: Weight::from_parts(1_000_000_000, 1024 * 1024),
}),
sp_keyring::Sr25519Keyring::Alice,
0,
);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
assert!(polkadot_test_runtime::System::events().iter().any(|r| matches!(
r.event,
polkadot_test_runtime::RuntimeEvent::Xcm(pallet_xcm::Event::Attempted {
outcome: Outcome::Complete { .. }
}),
)));
});
}
#[test]
fn transact_recursion_limit_works() {
sp_tracing::try_init_simple();
let client = TestClientBuilder::new().build();
let base_xcm = |call: polkadot_test_runtime::RuntimeCall| {
Xcm(vec![
WithdrawAsset((Here, 1_000).into()),
BuyExecution { fees: (Here, 1).into(), weight_limit: Unlimited },
Transact {
origin_kind: OriginKind::Native,
call: call.encode().into(),
fallback_max_weight: None,
},
])
};
let mut call: Option<polkadot_test_runtime::RuntimeCall> = None;
for depth in (1..12).rev() {
let mut msg;
match depth {
11 => {
msg = Xcm(vec![ClearOrigin]);
},
10 => {
let inner_call = call.take().unwrap();
let expected_transact_status =
sp_runtime::DispatchError::Module(sp_runtime::ModuleError {
index: 27,
error: [24, 0, 0, 0],
message: Some("LocalExecutionIncomplete"),
})
.encode()
.into();
msg = base_xcm(inner_call);
msg.inner_mut().push(ExpectTransactStatus(expected_transact_status));
},
d if d >= 1 && d <= 9 => {
let inner_call = call.take().unwrap();
msg = base_xcm(inner_call);
msg.inner_mut().push(ExpectTransactStatus(MaybeErrorCode::Success));
},
_ => unreachable!(),
}
let max_weight = <XcmConfig as xcm_executor::Config>::Weigher::weight(&mut msg).unwrap();
call = Some(polkadot_test_runtime::RuntimeCall::Xcm(pallet_xcm::Call::execute {
message: Box::new(VersionedXcm::from(msg.clone())),
max_weight,
}));
}
let mut block_builder = client.init_polkadot_block_builder();
let execute = construct_extrinsic(&client, call.unwrap(), sp_keyring::Sr25519Keyring::Alice, 0);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
let events = polkadot_test_runtime::System::events();
assert_eq!(
polkadot_test_runtime::System::events()
.iter()
.filter(|r| matches!(
r.event,
polkadot_test_runtime::RuntimeEvent::Xcm(pallet_xcm::Event::Attempted {
outcome: Outcome::Complete { .. }
}),
))
.count(),
10
);
assert!(events.iter().any(|r| matches!(
&r.event,
polkadot_test_runtime::RuntimeEvent::TransactionPayment(
pallet_transaction_payment::Event::TransactionFeePaid {
who: payer,
..
}
) if *payer == sp_keyring::Sr25519Keyring::Alice.into(),
)));
});
}
#[test]
fn query_response_fires() {
use pallet_test_notifier::Event::*;
use pallet_xcm::QueryStatus;
use polkadot_test_runtime::RuntimeEvent::TestNotifier;
sp_tracing::try_init_simple();
let client = TestClientBuilder::new().build();
let mut block_builder = client.init_polkadot_block_builder();
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::TestNotifier(
pallet_test_notifier::Call::prepare_new_query {},
),
sp_keyring::Sr25519Keyring::Alice,
0,
);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
let mut query_id = None;
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
for r in polkadot_test_runtime::System::events().iter() {
match r.event {
TestNotifier(QueryPrepared(q)) => query_id = Some(q),
_ => (),
}
}
});
let query_id = query_id.unwrap();
let mut block_builder = client.init_polkadot_block_builder();
let response = Response::ExecutionResult(None);
let max_weight = Weight::from_parts(1_000_000, 1024 * 1024);
let querier = Some(Here.into());
let msg = Xcm(vec![QueryResponse { query_id, response, max_weight, querier }]);
let msg = Box::new(VersionedXcm::from(msg));
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::Xcm(pallet_xcm::Call::execute {
message: msg,
max_weight: Weight::from_parts(1_000_000_000, 1024 * 1024),
}),
sp_keyring::Sr25519Keyring::Alice,
1,
);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
assert!(polkadot_test_runtime::System::events().iter().any(|r| matches!(
r.event,
polkadot_test_runtime::RuntimeEvent::Xcm(pallet_xcm::Event::ResponseReady {
query_id: q,
response: Response::ExecutionResult(None),
}) if q == query_id,
)));
assert_eq!(
polkadot_test_runtime::Xcm::query(query_id),
Some(QueryStatus::Ready {
response: VersionedResponse::from(Response::ExecutionResult(None)),
at: 2u32.into()
}),
)
});
}
#[test]
fn query_response_elicits_handler() {
use pallet_test_notifier::Event::*;
use polkadot_test_runtime::RuntimeEvent::TestNotifier;
sp_tracing::try_init_simple();
let client = TestClientBuilder::new().build();
let mut block_builder = client.init_polkadot_block_builder();
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::TestNotifier(
pallet_test_notifier::Call::prepare_new_notify_query {},
),
sp_keyring::Sr25519Keyring::Alice,
0,
);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
let mut query_id = None;
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
for r in polkadot_test_runtime::System::events().iter() {
match r.event {
TestNotifier(NotifyQueryPrepared(q)) => query_id = Some(q),
_ => (),
}
}
});
let query_id = query_id.unwrap();
let mut block_builder = client.init_polkadot_block_builder();
let response = Response::ExecutionResult(None);
let max_weight = Weight::from_parts(1_000_000, 1024 * 1024);
let querier = Some(Here.into());
let msg = Xcm(vec![QueryResponse { query_id, response, max_weight, querier }]);
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::Xcm(pallet_xcm::Call::execute {
message: Box::new(VersionedXcm::from(msg)),
max_weight: Weight::from_parts(1_000_000_000, 1024 * 1024),
}),
sp_keyring::Sr25519Keyring::Alice,
1,
);
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
assert!(polkadot_test_runtime::System::events().iter().any(|r| matches!(
&r.event,
TestNotifier(ResponseReceived(
location,
q,
Response::ExecutionResult(None),
)) if *q == query_id && matches!(location.unpack(), (0, [Junction::AccountId32 { .. }])),
)));
});
}
#[test]
fn deposit_reserve_asset_works_for_any_xcm_sender() {
sp_tracing::try_init_simple();
let client = TestClientBuilder::new().build();
let amount_to_send: u128 = 1_000_000_000_000;
let assets: Assets = (Parent, amount_to_send).into();
let fee_asset_item = 0;
let max_assets = assets.len() as u32;
let fees = assets.get(fee_asset_item as usize).unwrap().clone();
let weight_limit = Unlimited;
let reserve = Location::parent();
let dest = Location::new(1, [Parachain(2000)]);
let beneficiary_id = sp_keyring::Sr25519Keyring::Alice.to_account_id();
let beneficiary = Location::new(0, [AccountId32 { network: None, id: beneficiary_id.into() }]);
let fee1 = amount_to_send.saturating_div(2);
let fee2 = amount_to_send.saturating_sub(fee1);
let fees_half_1 = Asset::from((fees.id.clone(), Fungible(fee1)));
let fees_half_2 = Asset::from((fees.id.clone(), Fungible(fee2)));
let reserve_context = <XcmConfig as xcm_executor::Config>::UniversalLocation::get();
let reserve_fees = fees_half_1.reanchored(&reserve, &reserve_context).unwrap();
let dest_fees = fees_half_2.reanchored(&dest, &reserve_context).unwrap();
let assets_reanchored = assets.reanchored(&reserve, &reserve_context).unwrap();
let dest = dest.reanchored(&reserve, &reserve_context).unwrap();
let xcm_on_dest = Xcm(vec![
BuyExecution { fees: dest_fees, weight_limit: weight_limit.clone() },
DepositAsset { assets: Wild(AllCounted(max_assets)), beneficiary },
]);
let msg = Xcm(vec![
WithdrawAsset(assets_reanchored),
ClearOrigin,
BuyExecution { fees: reserve_fees, weight_limit },
DepositReserveAsset { assets: Wild(AllCounted(max_assets)), dest, xcm: xcm_on_dest },
]);
let mut block_builder = client.init_polkadot_block_builder();
let make_para_available =
construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::Sudo(pallet_sudo::Call::sudo {
call: Box::new(polkadot_test_runtime::RuntimeCall::System(
frame_system::Call::set_storage {
items: vec![(
polkadot_runtime_parachains::paras::Heads::<
polkadot_test_runtime::Runtime,
>::hashed_key_for(2000u32),
vec![1, 2, 3],
)],
},
)),
}),
sp_keyring::Sr25519Keyring::Alice,
0,
);
let execute = construct_extrinsic(
&client,
polkadot_test_runtime::RuntimeCall::Xcm(pallet_xcm::Call::execute {
message: Box::new(VersionedXcm::from(msg)),
max_weight: Weight::from_parts(1_000_000_000, 1024 * 1024),
}),
sp_keyring::Sr25519Keyring::Alice,
1,
);
block_builder
.push_polkadot_extrinsic(make_para_available)
.expect("pushes extrinsic");
block_builder.push_polkadot_extrinsic(execute).expect("pushes extrinsic");
let block = block_builder.build().expect("Finalizes the block").block;
let block_hash = block.hash();
futures::executor::block_on(client.import(sp_consensus::BlockOrigin::Own, block))
.expect("imports the block");
client.state_at(block_hash).expect("state should exist").inspect_state(|| {
assert!(polkadot_test_runtime::System::events().iter().any(|r| matches!(
r.event,
polkadot_test_runtime::RuntimeEvent::Xcm(pallet_xcm::Event::Attempted {
outcome: Outcome::Complete { .. }
}),
)));
});
}