// This file is part of Bizinikiwi. // Copyright (C) Parity Technologies (UK) Ltd. and Dijital Kurdistan Tech Institute // 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. //! Tests and test utilities for safe mode pezpallet. #![cfg(test)] use super::*; use crate as pezpallet_safe_mode; use pezframe::{ testing_prelude::*, traits::{InsideBoth, InstanceFilter, IsInVec}, }; #[derive_impl(pezframe_system::config_preludes::TestDefaultConfig)] impl pezframe_system::Config for Test { type BaseCallFilter = InsideBoth; type BlockWeights = (); type BlockLength = (); type RuntimeOrigin = RuntimeOrigin; type RuntimeCall = RuntimeCall; type Nonce = u64; type Hash = H256; type Hashing = BlakeTwo256; type AccountId = u64; type Lookup = IdentityLookup; type Block = Block; type RuntimeEvent = RuntimeEvent; type DbWeight = (); type Version = (); type PalletInfo = PalletInfo; type AccountData = pezpallet_balances::AccountData; type OnNewAccount = (); type OnKilledAccount = (); type SystemWeightInfo = (); type SS58Prefix = (); type OnSetCode = (); type MaxConsumers = ConstU32<16>; } /// Identifies a hold on an account's balance. #[derive( Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Encode, Decode, MaxEncodedLen, Debug, TypeInfo, )] pub enum HoldReason { /// The safe-mode pezpallet holds funds since an account either entered or extended the /// safe-mode. SafeMode, } #[derive_impl(pezpallet_balances::config_preludes::TestDefaultConfig)] impl pezpallet_balances::Config for Test { type ExistentialDeposit = ConstU64<2>; type AccountStore = System; } impl pezpallet_utility::Config for Test { type RuntimeEvent = RuntimeEvent; type RuntimeCall = RuntimeCall; type PalletsOrigin = OriginCaller; type WeightInfo = (); } /// Mocked proxies to check that the safe-mode also works with the proxy pezpallet. #[derive( Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Encode, Decode, DecodeWithMemTracking, RuntimeDebug, MaxEncodedLen, scale_info::TypeInfo, )] pub enum ProxyType { Any, JustTransfer, JustUtility, } impl Default for ProxyType { fn default() -> Self { Self::Any } } impl InstanceFilter for ProxyType { fn filter(&self, c: &RuntimeCall) -> bool { match self { ProxyType::Any => true, ProxyType::JustTransfer => { matches!( c, RuntimeCall::Balances(pezpallet_balances::Call::transfer_allow_death { .. }) ) }, ProxyType::JustUtility => matches!(c, RuntimeCall::Utility { .. }), } } fn is_superset(&self, o: &Self) -> bool { self == &ProxyType::Any || self == o } } impl pezpallet_proxy::Config for Test { type RuntimeEvent = RuntimeEvent; type RuntimeCall = RuntimeCall; type Currency = Balances; type ProxyType = ProxyType; type ProxyDepositBase = ConstU64<1>; type ProxyDepositFactor = ConstU64<1>; type MaxProxies = ConstU32<4>; type WeightInfo = (); type CallHasher = BlakeTwo256; type MaxPending = ConstU32<2>; type AnnouncementDepositBase = ConstU64<1>; type AnnouncementDepositFactor = ConstU64<1>; type BlockNumberProvider = pezframe_system::Pezpallet; } /// The calls that can always bypass safe-mode. pub struct WhitelistedCalls; impl Contains for WhitelistedCalls { fn contains(call: &RuntimeCall) -> bool { match call { RuntimeCall::Balances(_) => false, _ => true, } } } parameter_types! { pub const EnterDuration: u64 = 7; pub const ExtendDuration: u64 = 10; pub const EnterDepositAmount: u64 = 100; pub const ExtendDepositAmount: u64 = 100; pub const ReleaseDelay: u64 = 20; pub const SafeModeHoldReason: HoldReason = HoldReason::SafeMode; pub const ForceEnterWeak: u64 = 3; pub const ForceEnterStrong: u64 = 5; pub const ForceExtendWeak: u64 = 11; pub const ForceExtendStrong: u64 = 15; // NOTE: The account ID maps to the duration. Easy for testing. pub ForceEnterOrigins: Vec = vec![ForceEnterWeak::get(), ForceEnterStrong::get()]; pub ForceExtendOrigins: Vec = vec![ForceExtendWeak::get(), ForceExtendStrong::get()]; pub storage Notifications: Vec<(u64, bool)> = vec![]; } pub struct MockedNotify; impl SafeModeNotify for MockedNotify { fn entered() { let mut ns = Notifications::get(); ns.push((>::block_number(), true)); Notifications::set(&ns); } fn exited() { let mut ns = Notifications::get(); ns.push((>::block_number(), false)); Notifications::set(&ns); } } ord_parameter_types! { pub const ForceExitOrigin: u64 = 100; pub const ForceDepositOrigin: u64 = 200; } impl Config for Test { type RuntimeEvent = RuntimeEvent; type Currency = Balances; type RuntimeHoldReason = RuntimeHoldReason; type WhitelistedCalls = WhitelistedCalls; type EnterDuration = EnterDuration; type EnterDepositAmount = EnterDepositAmount; type ExtendDuration = ExtendDuration; type ExtendDepositAmount = ExtendDepositAmount; type ForceEnterOrigin = EnsureSignedBy, u64>; type ForceExtendOrigin = EnsureSignedBy, u64>; type ForceExitOrigin = EnsureSignedBy; type ForceDepositOrigin = EnsureSignedBy; type ReleaseDelay = ReleaseDelay; type Notify = MockedNotify; type WeightInfo = (); } type Block = pezframe_system::mocking::MockBlock; construct_runtime!( pub enum Test { System: pezframe_system, Balances: pezpallet_balances, Utility: pezpallet_utility, Proxy: pezpallet_proxy, SafeMode: pezpallet_safe_mode, } ); pub const BAL_ACC0: u64 = 1234; pub const BAL_ACC1: u64 = 5678; pub fn new_test_ext() -> TestExternalities { let mut t = pezframe_system::GenesisConfig::::default().build_storage().unwrap(); pezpallet_balances::GenesisConfig:: { // The 0 account is NOT a special origin, the rest may be. balances: vec![(0, BAL_ACC0), (1, BAL_ACC1), (2, 5678), (3, 5678), (4, 5678)], ..Default::default() } .assimilate_storage(&mut t) .unwrap(); pezpallet_safe_mode::GenesisConfig:: { entered_until: None } .assimilate_storage(&mut t) .unwrap(); let mut ext = TestExternalities::new(t); ext.execute_with(|| { System::set_block_number(1); }); ext } pub fn next_block() { SafeMode::on_finalize(System::block_number()); Balances::on_finalize(System::block_number()); System::on_finalize(System::block_number()); System::set_block_number(System::block_number() + 1); System::on_initialize(System::block_number()); Balances::on_initialize(System::block_number()); SafeMode::on_initialize(System::block_number()); } pub fn run_to(n: u64) { while System::block_number() < n { next_block(); } }