// This file is part of Substrate. // Copyright (C) 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. //! Tests and test utilities for safe mode pallet. #![cfg(test)] use super::*; use crate as pallet_safe_mode; use frame_support::{ parameter_types, traits::{ConstU64, Everything, InsideBoth, InstanceFilter, IsInVec, SafeModeNotify}, }; use frame_system::EnsureSignedBy; use sp_core::H256; use sp_runtime::{ traits::{BlakeTwo256, IdentityLookup}, BuildStorage, }; impl frame_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 BlockHashCount = ConstU64<250>; type DbWeight = (); type Version = (); type PalletInfo = PalletInfo; type AccountData = pallet_balances::AccountData; type OnNewAccount = (); type OnKilledAccount = (); type SystemWeightInfo = (); type SS58Prefix = (); type OnSetCode = (); type MaxConsumers = frame_support::traits::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 pallet holds funds since an account either entered or extended the safe-mode. SafeMode, } impl pallet_balances::Config for Test { type Balance = u64; type DustRemoval = (); type RuntimeEvent = RuntimeEvent; type ExistentialDeposit = ConstU64<2>; type AccountStore = System; type WeightInfo = (); type MaxLocks = (); type MaxReserves = ConstU32<10>; type ReserveIdentifier = [u8; 8]; type RuntimeHoldReason = RuntimeHoldReason; type RuntimeFreezeReason = RuntimeFreezeReason; type FreezeIdentifier = (); type MaxHolds = ConstU32<10>; type MaxFreezes = ConstU32<0>; } impl pallet_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 pallet. #[derive( Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Encode, Decode, 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(pallet_balances::Call::transfer_allow_death { .. }) ) }, ProxyType::JustUtility => matches!(c, RuntimeCall::Utility { .. }), } } fn is_superset(&self, o: &Self) -> bool { self == &ProxyType::Any || self == o } } impl pallet_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>; } /// 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); } } frame_support::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 = frame_system::mocking::MockBlock; frame_support::construct_runtime!( pub enum Test { System: frame_system, Balances: pallet_balances, Utility: pallet_utility, Proxy: pallet_proxy, SafeMode: pallet_safe_mode, } ); pub const BAL_ACC0: u64 = 1234; pub const BAL_ACC1: u64 = 5678; pub fn new_test_ext() -> sp_io::TestExternalities { let mut t = frame_system::GenesisConfig::::default().build_storage().unwrap(); pallet_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)], } .assimilate_storage(&mut t) .unwrap(); pallet_safe_mode::GenesisConfig:: { entered_until: None } .assimilate_storage(&mut t) .unwrap(); let mut ext = sp_io::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(); } }