mirror of
https://github.com/pezkuwichain/pezkuwi-subxt.git
synced 2026-05-07 08:28:05 +00:00
3aae882437
* Fixes * Undiener * Undiener * Undiener * Lockfile * Changes for send returning hash * Include message ID as params to execute_xcm * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Companion fixes * Formatting * Fixes * Formatting * Bump * Bump * Fixes * Formatting * Make the price of UMP/XCMP message sending configurable * cargo fmt * Remove InvertLocation * Formatting * Use ConstantPrice from polkadot-runtime-common * Fix naming * cargo fmt * Fixes * Fixes * Fixes * Add CallDispatcher * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Fixes * Remove unused import * Remove unused import * XCMv3 fixes (#1710) * Fixes XCMv3 related Fixes XCMv3 (removed query_holding) Fixes XCMv3 - should use _depositable_count? Fixes XCMv3 - removed TrustedReserve Fixes - missing weights for statemine/statemint/westmint [DO-NOT-CHERRY-PICK] tmp return query_holding to aviod conficts to master Fixes - missing functions for pallet_xcm_benchmarks::generic::Config Fixes for XCMv3 benchmarking Fix xcm - removed query_holding * ".git/.scripts/bench-bot.sh" xcm statemine assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm statemint assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm westmint assets pallet_xcm_benchmarks::generic * Fix imports * Avoid consuming XCM message for NotApplicable scenario (#1787) * Avoid consuming message for NotApplicable scenario * Avoid consuming message for NotApplicable scenario tests * Add 10 message processing limit to DMP queue * Add 10 message limit to XCMP queue * Always increment the message_processed count whenever a message is processed * Fix formatting * Set an upper limit to the overweight message DMP queue * Add upper limit to XCMP overweight message queue * Fix for missing weight for `fn unpaid_execution()` * Fix - usage of `messages_processed` * Fixes * Fixes * Fixes * cargo fmt * Fixes * Fixes * Fixes * Fixes * Remove unused import * Fixes for gav-xcm-v3 (#1835) * Fix for FungiblesAdapter - trait changes: Contains -> AssetChecking * Fix for missing weight for `fn unpaid_execution()` * Used NonLocalMint for all NonZeroIssuance * Fix * Fixes * Fixes * Fixes * Fixes * Fixes * Fix tests * Fixes * Add SafeCallFilter * Add missing config items * Add TODO * Use () as the PriceForParentDelivery * Fixes * Fixes * Fixes * Fixes * Update transact_origin to transact_origin_and_runtime_call * Add ReachableDest config item to XCM pallet * Update SafeCallFilter to allow remark_with_event in runtime benchmarks * cargo fmt * Update substrate * Fix worst_case_holding * Fix DMQ queue unit tests * Remove unused label * cargo fmt * Actually process incoming XCMs * Fixes * Fixes * Fixes * Fixes - return back Weightless * Added measured benchmarks for `pallet_xcm` (#1968) * Fix Fix Fix * Fix * Fixes for transact benchmark * Fixes add pallet_xcm to benchmarks * Revert remark_with_event * ".git/.scripts/bench-bot.sh" xcm statemine assets pallet_xcm_benchmarks::generic * Fixes * TMP * Fix for reserve_asset_deposited * ".git/.scripts/bench-bot.sh" pallet statemine assets pallet_xcm * Fix * ".git/.scripts/bench-bot.sh" pallet statemint assets pallet_xcm * Fix * ".git/.scripts/bench-bot.sh" pallet westmint assets pallet_xcm * Fix westmint * ".git/.scripts/bench-bot.sh" xcm statemine assets pallet_xcm_benchmarks::generic * Fix * ".git/.scripts/bench-bot.sh" xcm westmint assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm statemint assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" pallet collectives-polkadot collectives pallet_xcm * Fix for collectives * ".git/.scripts/bench-bot.sh" pallet bridge-hub-kusama bridge-hubs pallet_xcm * ".git/.scripts/bench-bot.sh" pallet bridge-hub-rococo bridge-hubs pallet_xcm * Fixes for bridge-hubs * Fixes - return back Weightless * Fix - removed MigrateToTrackInactive for contracts-rococo Co-authored-by: command-bot <> * cargo fmt * Fix benchmarks * Bko gav xcm v3 (#1993) * Fix * ".git/.scripts/bench-bot.sh" xcm statemint assets pallet_xcm_benchmarks::fungible * ".git/.scripts/bench-bot.sh" xcm statemine assets pallet_xcm_benchmarks::fungible * ".git/.scripts/bench-bot.sh" xcm westmint assets pallet_xcm_benchmarks::fungible * ".git/.scripts/bench-bot.sh" xcm statemine assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm statemint assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm westmint assets pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" pallet statemine assets pallet_xcm * ".git/.scripts/bench-bot.sh" pallet westmint assets pallet_xcm * ".git/.scripts/bench-bot.sh" pallet statemint assets pallet_xcm * ".git/.scripts/bench-bot.sh" pallet collectives-polkadot collectives pallet_xcm * ".git/.scripts/bench-bot.sh" pallet bridge-hub-kusama bridge-hubs pallet_xcm * ".git/.scripts/bench-bot.sh" pallet bridge-hub-rococo bridge-hubs pallet_xcm Co-authored-by: command-bot <> * Change AllowUnpaidExecutionFrom to be explicit * xcm-v3 benchmarks, weights, fixes for bridge-hubs (#2035) * Dumy weights to get compile * Change UniversalLocation according to https://github.com/paritytech/polkadot/pull/4097 (Location Inversion Removed) * Fix bridge-hubs weights * ".git/.scripts/bench-bot.sh" pallet statemine assets pallet_xcm * ".git/.scripts/bench-bot.sh" pallet statemint assets pallet_xcm * ".git/.scripts/bench-bot.sh" pallet collectives-polkadot collectives pallet_xcm * ".git/.scripts/bench-bot.sh" pallet westmint assets pallet_xcm * ".git/.scripts/bench-bot.sh" xcm bridge-hub-kusama bridge-hubs pallet_xcm_benchmarks::generic * ".git/.scripts/bench-bot.sh" xcm bridge-hub-kusama bridge-hubs pallet_xcm_benchmarks::fungible * ".git/.scripts/bench-bot.sh" pallet bridge-hub-kusama bridge-hubs pallet_xcm * ".git/.scripts/bench-bot.sh" pallet bridge-hub-rococo bridge-hubs pallet_xcm * ".git/.scripts/bench-bot.sh" xcm bridge-hub-rococo bridge-hubs pallet_xcm_benchmarks::fungible * ".git/.scripts/bench-bot.sh" xcm bridge-hub-rococo bridge-hubs pallet_xcm_benchmarks::generic * Change NetworkId to Option<NetworkId> Co-authored-by: command-bot <> Co-authored-by: Keith Yeung <kungfukeith11@gmail.com> * Add event for showing the hash of an UMP sent message (#1228) * Add UpwardMessageSent event in parachain-system * additional fixes * Message Id * Fix errors from merge * fmt * more fmt * Remove todo * more formatting * Fixes * Fixes * Fixes * Fixes * Allow explicit unpaid executions from the relay chains for system parachains (#2060) * Allow explicit unpaid executions from the relay chains for system parachains * Put origin-filtering barriers into WithComputedOrigin * Use ConstU32<8> * Small nits * formatting * cargo fmt * Allow receiving XCMs from any relay chain plurality * Fixes * update lockfile for {"polkadot", "substrate"} * Update polkadot * Add runtime-benchmarks feature Co-authored-by: Keith Yeung <kungfukeith11@gmail.com> Co-authored-by: Branislav Kontur <bkontur@gmail.com> Co-authored-by: girazoki <gorka.irazoki@gmail.com> Co-authored-by: parity-processbot <>
344 lines
10 KiB
Rust
344 lines
10 KiB
Rust
// Copyright (C) 2021 Parity Technologies (UK) Ltd.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
use super::*;
|
|
use cumulus_primitives_core::XcmpMessageHandler;
|
|
use frame_support::{assert_noop, assert_ok};
|
|
use mock::{new_test_ext, RuntimeCall, RuntimeOrigin, Test, XcmpQueue};
|
|
use sp_runtime::traits::BadOrigin;
|
|
|
|
#[test]
|
|
fn one_message_does_not_panic() {
|
|
new_test_ext().execute_with(|| {
|
|
let message_format = XcmpMessageFormat::ConcatenatedVersionedXcm.encode();
|
|
let messages = vec![(Default::default(), 1u32.into(), message_format.as_slice())];
|
|
|
|
// This shouldn't cause a panic
|
|
XcmpQueue::handle_xcmp_messages(messages.into_iter(), Weight::MAX);
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Invalid incoming blob message data"]
|
|
#[cfg(debug_assertions)]
|
|
fn bad_message_is_handled() {
|
|
new_test_ext().execute_with(|| {
|
|
let bad_data = vec![
|
|
1, 1, 3, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 64, 239, 139, 0,
|
|
0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 255, 255, 0, 0, 0, 0, 0, 0, 0, 37, 0,
|
|
0, 0, 0, 0, 0, 0, 16, 0, 127, 147,
|
|
];
|
|
InboundXcmpMessages::<Test>::insert(ParaId::from(1000), 1, bad_data);
|
|
let format = XcmpMessageFormat::ConcatenatedEncodedBlob;
|
|
// This should exit with an error.
|
|
XcmpQueue::process_xcmp_message(
|
|
1000.into(),
|
|
(1, format),
|
|
&mut 0,
|
|
Weight::from_ref_time(10_000_000_000),
|
|
Weight::from_ref_time(10_000_000_000),
|
|
);
|
|
});
|
|
}
|
|
|
|
/// Tests that a blob message is handled. Currently this isn't implemented and panics when debug assertions
|
|
/// are enabled. When this feature is enabled, this test should be rewritten properly.
|
|
#[test]
|
|
#[should_panic = "Blob messages not handled."]
|
|
#[cfg(debug_assertions)]
|
|
fn handle_blob_message() {
|
|
new_test_ext().execute_with(|| {
|
|
let bad_data = vec![
|
|
1, 1, 1, 1, 3, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 0, 64, 239,
|
|
139, 0, 0, 0, 64, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 255, 255, 0, 0, 0, 0, 0, 0, 0,
|
|
37, 0, 0, 0, 0, 0, 0, 0, 16, 0, 127, 147,
|
|
];
|
|
InboundXcmpMessages::<Test>::insert(ParaId::from(1000), 1, bad_data);
|
|
let format = XcmpMessageFormat::ConcatenatedEncodedBlob;
|
|
XcmpQueue::process_xcmp_message(
|
|
1000.into(),
|
|
(1, format),
|
|
&mut 0,
|
|
Weight::from_ref_time(10_000_000_000),
|
|
Weight::from_ref_time(10_000_000_000),
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
#[should_panic = "Invalid incoming XCMP message data"]
|
|
#[cfg(debug_assertions)]
|
|
fn handle_invalid_data() {
|
|
new_test_ext().execute_with(|| {
|
|
let data = Xcm::<Test>(vec![]).encode();
|
|
InboundXcmpMessages::<Test>::insert(ParaId::from(1000), 1, data);
|
|
let format = XcmpMessageFormat::ConcatenatedVersionedXcm;
|
|
XcmpQueue::process_xcmp_message(
|
|
1000.into(),
|
|
(1, format),
|
|
&mut 0,
|
|
Weight::from_ref_time(10_000_000_000),
|
|
Weight::from_ref_time(10_000_000_000),
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn service_overweight_unknown() {
|
|
new_test_ext().execute_with(|| {
|
|
assert_noop!(
|
|
XcmpQueue::service_overweight(RuntimeOrigin::root(), 0, Weight::from_parts(1000, 1000)),
|
|
Error::<Test>::BadOverweightIndex,
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn service_overweight_bad_xcm_format() {
|
|
new_test_ext().execute_with(|| {
|
|
let bad_xcm = vec![255];
|
|
Overweight::<Test>::insert(0, (ParaId::from(1000), 0, bad_xcm));
|
|
|
|
assert_noop!(
|
|
XcmpQueue::service_overweight(RuntimeOrigin::root(), 0, Weight::from_parts(1000, 1000)),
|
|
Error::<Test>::BadXcm
|
|
);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn suspend_xcm_execution_works() {
|
|
new_test_ext().execute_with(|| {
|
|
QueueSuspended::<Test>::put(true);
|
|
|
|
let xcm =
|
|
VersionedXcm::from(Xcm::<RuntimeCall>(vec![Instruction::<RuntimeCall>::ClearOrigin]))
|
|
.encode();
|
|
let mut message_format = XcmpMessageFormat::ConcatenatedVersionedXcm.encode();
|
|
message_format.extend(xcm.clone());
|
|
let messages = vec![(ParaId::from(999), 1u32.into(), message_format.as_slice())];
|
|
|
|
// This should have executed the incoming XCM, because it came from a system parachain
|
|
XcmpQueue::handle_xcmp_messages(messages.into_iter(), Weight::MAX);
|
|
|
|
let queued_xcm = InboundXcmpMessages::<Test>::get(ParaId::from(999), 1u32);
|
|
assert!(queued_xcm.is_empty());
|
|
|
|
let messages = vec![(ParaId::from(2000), 1u32.into(), message_format.as_slice())];
|
|
|
|
// This shouldn't have executed the incoming XCM
|
|
XcmpQueue::handle_xcmp_messages(messages.into_iter(), Weight::MAX);
|
|
|
|
let queued_xcm = InboundXcmpMessages::<Test>::get(ParaId::from(2000), 1u32);
|
|
assert_eq!(queued_xcm, xcm);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_suspend_threshold_works() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(data.suspend_threshold, 2);
|
|
assert_ok!(XcmpQueue::update_suspend_threshold(RuntimeOrigin::root(), 3));
|
|
assert_noop!(XcmpQueue::update_suspend_threshold(RuntimeOrigin::signed(2), 5), BadOrigin);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(data.suspend_threshold, 3);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_drop_threshold_works() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(data.drop_threshold, 5);
|
|
assert_ok!(XcmpQueue::update_drop_threshold(RuntimeOrigin::root(), 6));
|
|
assert_noop!(XcmpQueue::update_drop_threshold(RuntimeOrigin::signed(2), 7), BadOrigin);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(data.drop_threshold, 6);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_resume_threshold_works() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(data.resume_threshold, 1);
|
|
assert_ok!(XcmpQueue::update_resume_threshold(RuntimeOrigin::root(), 2));
|
|
assert_noop!(XcmpQueue::update_resume_threshold(RuntimeOrigin::signed(7), 3), BadOrigin);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(data.resume_threshold, 2);
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_threshold_weight_works() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(data.threshold_weight, Weight::from_ref_time(100_000));
|
|
assert_ok!(XcmpQueue::update_threshold_weight(
|
|
RuntimeOrigin::root(),
|
|
Weight::from_ref_time(10_000)
|
|
));
|
|
assert_noop!(
|
|
XcmpQueue::update_threshold_weight(
|
|
RuntimeOrigin::signed(5),
|
|
Weight::from_ref_time(10_000_000),
|
|
),
|
|
BadOrigin
|
|
);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(data.threshold_weight, Weight::from_ref_time(10_000));
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_weight_restrict_decay_works() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(data.weight_restrict_decay, Weight::from_ref_time(2));
|
|
assert_ok!(XcmpQueue::update_weight_restrict_decay(
|
|
RuntimeOrigin::root(),
|
|
Weight::from_ref_time(5)
|
|
));
|
|
assert_noop!(
|
|
XcmpQueue::update_weight_restrict_decay(
|
|
RuntimeOrigin::signed(6),
|
|
Weight::from_ref_time(4),
|
|
),
|
|
BadOrigin
|
|
);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(data.weight_restrict_decay, Weight::from_ref_time(5));
|
|
});
|
|
}
|
|
|
|
#[test]
|
|
fn update_xcmp_max_individual_weight() {
|
|
new_test_ext().execute_with(|| {
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
assert_eq!(
|
|
data.xcmp_max_individual_weight,
|
|
Weight::from_parts(20u64 * WEIGHT_REF_TIME_PER_MILLIS, DEFAULT_POV_SIZE),
|
|
);
|
|
assert_ok!(XcmpQueue::update_xcmp_max_individual_weight(
|
|
RuntimeOrigin::root(),
|
|
Weight::from_ref_time(30u64 * WEIGHT_REF_TIME_PER_MILLIS)
|
|
));
|
|
assert_noop!(
|
|
XcmpQueue::update_xcmp_max_individual_weight(
|
|
RuntimeOrigin::signed(3),
|
|
Weight::from_ref_time(10u64 * WEIGHT_REF_TIME_PER_MILLIS)
|
|
),
|
|
BadOrigin
|
|
);
|
|
let data: QueueConfigData = <QueueConfig<Test>>::get();
|
|
|
|
assert_eq!(
|
|
data.xcmp_max_individual_weight,
|
|
Weight::from_ref_time(30u64 * WEIGHT_REF_TIME_PER_MILLIS)
|
|
);
|
|
});
|
|
}
|
|
|
|
/// Validates [`validate`] for required Some(destination) and Some(message)
|
|
struct OkFixedXcmHashWithAssertingRequiredInputsSender;
|
|
impl OkFixedXcmHashWithAssertingRequiredInputsSender {
|
|
const FIXED_XCM_HASH: [u8; 32] = [9; 32];
|
|
|
|
fn fixed_delivery_asset() -> MultiAssets {
|
|
MultiAssets::new()
|
|
}
|
|
|
|
fn expected_delivery_result() -> Result<(XcmHash, MultiAssets), SendError> {
|
|
Ok((Self::FIXED_XCM_HASH, Self::fixed_delivery_asset()))
|
|
}
|
|
}
|
|
impl SendXcm for OkFixedXcmHashWithAssertingRequiredInputsSender {
|
|
type Ticket = ();
|
|
|
|
fn validate(
|
|
destination: &mut Option<MultiLocation>,
|
|
message: &mut Option<Xcm<()>>,
|
|
) -> SendResult<Self::Ticket> {
|
|
assert!(destination.is_some());
|
|
assert!(message.is_some());
|
|
Ok(((), OkFixedXcmHashWithAssertingRequiredInputsSender::fixed_delivery_asset()))
|
|
}
|
|
|
|
fn deliver(_: Self::Ticket) -> Result<XcmHash, SendError> {
|
|
Ok(Self::FIXED_XCM_HASH)
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn xcmp_queue_does_not_consume_dest_or_msg_on_not_applicable() {
|
|
// dummy message
|
|
let message = Xcm(vec![Trap(5)]);
|
|
|
|
// XcmpQueue - check dest is really not applicable
|
|
let dest = (Parent, Parent, Parent);
|
|
let mut dest_wrapper = Some(dest.clone().into());
|
|
let mut msg_wrapper = Some(message.clone());
|
|
assert_eq!(
|
|
Err(SendError::NotApplicable),
|
|
<XcmpQueue as SendXcm>::validate(&mut dest_wrapper, &mut msg_wrapper)
|
|
);
|
|
|
|
// check wrapper were not consumed
|
|
assert_eq!(Some(dest.clone().into()), dest_wrapper.take());
|
|
assert_eq!(Some(message.clone()), msg_wrapper.take());
|
|
|
|
// another try with router chain with asserting sender
|
|
assert_eq!(
|
|
OkFixedXcmHashWithAssertingRequiredInputsSender::expected_delivery_result(),
|
|
send_xcm::<(XcmpQueue, OkFixedXcmHashWithAssertingRequiredInputsSender)>(
|
|
dest.into(),
|
|
message
|
|
)
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn xcmp_queue_consumes_dest_and_msg_on_ok_validate() {
|
|
// dummy message
|
|
let message = Xcm(vec![Trap(5)]);
|
|
|
|
// XcmpQueue - check dest/msg is valid
|
|
let dest = (Parent, X1(Parachain(5555)));
|
|
let mut dest_wrapper = Some(dest.clone().into());
|
|
let mut msg_wrapper = Some(message.clone());
|
|
assert!(<XcmpQueue as SendXcm>::validate(&mut dest_wrapper, &mut msg_wrapper).is_ok());
|
|
|
|
// check wrapper were consumed
|
|
assert_eq!(None, dest_wrapper.take());
|
|
assert_eq!(None, msg_wrapper.take());
|
|
|
|
new_test_ext().execute_with(|| {
|
|
// another try with router chain with asserting sender
|
|
assert_eq!(
|
|
Err(SendError::Transport("NoChannel")),
|
|
send_xcm::<(XcmpQueue, OkFixedXcmHashWithAssertingRequiredInputsSender)>(
|
|
dest.into(),
|
|
message
|
|
)
|
|
);
|
|
});
|
|
}
|